;;; helm-org.el --- Helm for org headlines and keywords completion -*- lexical-binding: t -*-
|
|
;; Copyright (C) 2012 ~ 2018 Thierry Volpiatto <thierry.volpiatto@gmail.com>
|
|
;; This program is free software; you can redistribute it and/or modify
|
;; it under the terms of the GNU General Public License as published by
|
;; the Free Software Foundation, either version 3 of the License, or
|
;; (at your option) any later version.
|
|
;; This program is distributed in the hope that it will be useful,
|
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
|
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
;; GNU General Public License for more details.
|
|
;; You should have received a copy of the GNU General Public License
|
;; along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
;;; Code:
|
(require 'cl-lib)
|
(require 'helm)
|
(require 'helm-utils)
|
(require 'org)
|
|
;; Load org-with-point-at macro when compiling
|
(eval-when-compile
|
(require 'org-macs))
|
|
(declare-function org-agenda-switch-to "org-agenda.el")
|
|
(defgroup helm-org nil
|
"Org related functions for helm."
|
:group 'helm)
|
|
(defcustom helm-org-headings-fontify nil
|
"Fontify org buffers before parsing them.
|
This reflect fontification in helm-buffer when non--nil.
|
NOTE: This will be slow on large org buffers."
|
:group 'helm-org
|
:type 'boolean)
|
|
(defcustom helm-org-format-outline-path nil
|
"Show all org level as path."
|
:group 'helm-org
|
:type 'boolean)
|
|
(defcustom helm-org-show-filename nil
|
"Show org filenames in `helm-org-agenda-files-headings' when non--nil.
|
Note this have no effect in `helm-org-in-buffer-headings'."
|
:group 'helm-org
|
:type 'boolean)
|
|
(defcustom helm-org-headings-min-depth 1
|
"Minimum depth of org headings to start with."
|
:group 'helm-org
|
:type 'integer)
|
|
(defcustom helm-org-headings-max-depth 8
|
"Go down to this maximum depth of org headings."
|
:group 'helm-org
|
:type 'integer)
|
|
(defcustom helm-org-headings-actions
|
'(("Go to heading" . helm-org-goto-marker)
|
("Open in indirect buffer `C-c i'" . helm-org--open-heading-in-indirect-buffer)
|
("Refile heading(s) (marked-to-selected|current-to-selected) `C-c w`" . helm-org--refile-heading-to)
|
("Insert link to this heading `C-c l`" . helm-org-insert-link-to-heading-at-marker))
|
"Default actions alist for
|
`helm-source-org-headings-for-files'."
|
:group 'helm-org
|
:type '(alist :key-type string :value-type function))
|
|
(defcustom helm-org-truncate-lines t
|
"Truncate org-header-lines when non-nil"
|
:type 'boolean
|
:group 'helm-org)
|
|
(defcustom helm-org-ignore-autosaves nil
|
"Ignore autosave files when starting `helm-org-agenda-files-headings'."
|
:type 'boolean
|
:group 'helm-org)
|
|
|
;;; Org capture templates
|
;;
|
;;
|
(defvar org-capture-templates)
|
(defun helm-source-org-capture-templates ()
|
(helm-build-sync-source "Org Capture Templates:"
|
:candidates (cl-loop for template in org-capture-templates
|
collect (cons (nth 1 template) (nth 0 template)))
|
:action '(("Do capture" . (lambda (template-shortcut)
|
(org-capture nil template-shortcut))))))
|
|
;;; Org headings
|
;;
|
;;
|
(defun helm-org-goto-marker (marker)
|
(switch-to-buffer (marker-buffer marker))
|
(goto-char (marker-position marker))
|
(org-show-context)
|
(re-search-backward "^\\*+ " nil t)
|
(org-show-entry))
|
|
(defun helm-org--open-heading-in-indirect-buffer (marker)
|
(helm-org-goto-marker marker)
|
(org-tree-to-indirect-buffer)
|
|
;; Put the non-indirect buffer at the bottom of the prev-buffers
|
;; list so it won't be selected when the indirect buffer is killed
|
(set-window-prev-buffers nil (append (cdr (window-prev-buffers))
|
(car (window-prev-buffers)))))
|
|
(defun helm-org-run-open-heading-in-indirect-buffer ()
|
"Open selected Org heading in an indirect buffer."
|
(interactive)
|
(with-helm-alive-p
|
(helm-exit-and-execute-action #'helm-org--open-heading-in-indirect-buffer)))
|
(put 'helm-org-run-open-heading-in-indirect-buffer 'helm-only t)
|
|
(defvar helm-org-headings-map
|
(let ((map (make-sparse-keymap)))
|
(set-keymap-parent map helm-map)
|
(define-key map (kbd "C-c i") 'helm-org-run-open-heading-in-indirect-buffer)
|
(define-key map (kbd "C-c w") 'helm-org-run-refile-heading-to)
|
(define-key map (kbd "C-c l") 'helm-org-run-insert-link-to-heading-at-marker)
|
map)
|
"Keymap for `helm-source-org-headings-for-files'.")
|
|
(defclass helm-org-headings-class (helm-source-sync)
|
((parents
|
:initarg :parents
|
:initform nil
|
:custom boolean)
|
(match :initform
|
(lambda (candidate)
|
(string-match
|
helm-pattern
|
(helm-aif (get-text-property 0 'helm-real-display candidate)
|
it
|
candidate))))
|
(help-message :initform 'helm-org-headings-help-message)
|
(action :initform 'helm-org-headings-actions)
|
(keymap :initform 'helm-org-headings-map)
|
(group :initform 'helm-org)))
|
|
(defmethod helm--setup-source :after ((source helm-org-headings-class))
|
(let ((parents (slot-value source 'parents)))
|
(setf (slot-value source 'candidate-transformer)
|
(lambda (candidates)
|
(let ((cands (helm-org-get-candidates candidates parents)))
|
(if parents (nreverse cands) cands))))))
|
|
(defun helm-source-org-headings-for-files (filenames &optional parents)
|
(helm-make-source "Org Headings" 'helm-org-headings-class
|
:filtered-candidate-transformer 'helm-org-startup-visibility
|
:parents parents
|
:candidates filenames))
|
|
(defun helm-org-startup-visibility (candidates _source)
|
"Indent headings and hide leading stars displayed in the helm buffer.
|
If `org-startup-indented' and `org-hide-leading-stars' are nil, do
|
nothing to CANDIDATES."
|
(cl-loop for i in candidates
|
collect
|
;; Transformation is not needed if these variables are t.
|
(if (or helm-org-show-filename helm-org-format-outline-path)
|
(cons
|
(car i) (cdr i))
|
(cons
|
(if helm-org-headings-fontify
|
(when (string-match "^\\(\\**\\)\\(\\* \\)\\(.*\n?\\)" (car i))
|
(replace-match "\\1\\2\\3" t nil (car i)))
|
(when (string-match "^\\(\\**\\)\\(\\* \\)\\(.*\n?\\)" (car i))
|
(let ((foreground (org-find-invisible-foreground)))
|
(with-helm-current-buffer
|
(cond
|
;; org-startup-indented is t, and org-hide-leading-stars is t
|
;; Or: #+STARTUP: indent hidestars
|
((and org-startup-indented org-hide-leading-stars)
|
(with-helm-buffer
|
(require 'org-indent)
|
(org-indent-mode 1)
|
(replace-match
|
(format "%s\\2\\3"
|
(propertize (replace-match "\\1" t nil (car i))
|
'face `(:foreground ,foreground)))
|
t nil (car i))))
|
;; org-startup-indented is nil, org-hide-leading-stars is t
|
;; Or: #+STARTUP: noindent hidestars
|
((and (not org-startup-indented) org-hide-leading-stars)
|
(with-helm-buffer
|
(replace-match
|
(format "%s\\2\\3"
|
(propertize (replace-match "\\1" t nil (car i))
|
'face `(:foreground ,foreground)))
|
t nil (car i))))
|
;; org-startup-indented is nil, and org-hide-leading-stars is nil
|
;; Or: #+STARTUP: noindent showstars
|
(t
|
(with-helm-buffer
|
(replace-match "\\1\\2\\3" t nil (car i)))))))))
|
(cdr i)))))
|
|
(defun helm-org-get-candidates (filenames &optional parents)
|
(apply #'append
|
(mapcar (lambda (filename)
|
(helm-org--get-candidates-in-file
|
filename
|
helm-org-headings-fontify
|
(or parents (null helm-org-show-filename))
|
parents))
|
filenames)))
|
|
(defun helm-org--get-candidates-in-file (filename &optional fontify nofname parents)
|
(with-current-buffer (pcase filename
|
((pred bufferp) filename)
|
((pred stringp) (find-file-noselect filename t)))
|
(let ((match-fn (if fontify
|
#'match-string
|
#'match-string-no-properties))
|
(search-fn (lambda ()
|
(re-search-forward
|
org-complex-heading-regexp nil t)))
|
(file (unless (or (bufferp filename) nofname)
|
(concat (helm-basename filename) ":"))))
|
(when parents
|
(add-function :around (var search-fn)
|
(lambda (old-fn &rest args)
|
(when (org-up-heading-safe)
|
(apply old-fn args)))))
|
(save-excursion
|
(save-restriction
|
(unless (and (bufferp filename)
|
(buffer-base-buffer filename))
|
;; Only widen direct buffers, not indirect ones.
|
(widen))
|
(unless parents (goto-char (point-min)))
|
;; clear cache for new version of org-get-outline-path
|
(and (boundp 'org-outline-path-cache)
|
(setq org-outline-path-cache nil))
|
(cl-loop with width = (window-width (helm-window))
|
while (funcall search-fn)
|
for beg = (point-at-bol)
|
for end = (point-at-eol)
|
when (and fontify
|
(null (text-property-any
|
beg end 'fontified t)))
|
do (jit-lock-fontify-now beg end)
|
for level = (length (match-string-no-properties 1))
|
for heading = (funcall match-fn 4)
|
if (and (>= level helm-org-headings-min-depth)
|
(<= level helm-org-headings-max-depth))
|
collect `(,(propertize
|
(if helm-org-format-outline-path
|
(org-format-outline-path
|
;; org-get-outline-path changed in signature and behaviour since org's
|
;; commit 105a4466971. Let's fall-back to the new version in case
|
;; of wrong-number-of-arguments error.
|
(condition-case nil
|
(append (apply #'org-get-outline-path
|
(unless parents
|
(list t level heading)))
|
(list heading))
|
(wrong-number-of-arguments
|
(org-get-outline-path t t)))
|
width file)
|
(if file
|
(concat file (funcall match-fn 0))
|
(funcall match-fn 0)))
|
'helm-real-display heading)
|
. ,(point-marker))))))))
|
|
(defun helm-org-insert-link-to-heading-at-marker (marker)
|
(with-current-buffer (marker-buffer marker)
|
(let ((heading-name (save-excursion (goto-char (marker-position marker))
|
(nth 4 (org-heading-components))))
|
(file-name (buffer-file-name)))
|
(with-helm-current-buffer
|
(org-insert-link
|
file-name (concat "file:" file-name "::*" heading-name))))))
|
|
(defun helm-org-run-insert-link-to-heading-at-marker ()
|
(interactive)
|
(with-helm-alive-p
|
(helm-exit-and-execute-action
|
'helm-org-insert-link-to-heading-at-marker)))
|
|
(defun helm-org--refile-heading-to (marker)
|
"Refile headings to heading at MARKER.
|
If multiple candidates are marked in the Helm session, they will
|
all be refiled. If no headings are marked, the selected heading
|
will be refiled."
|
(let* ((victims (with-helm-buffer (helm-marked-candidates)))
|
(buffer (marker-buffer marker))
|
(filename (buffer-file-name buffer))
|
(rfloc (list nil filename nil marker)))
|
(when (and (= 1 (length victims))
|
(equal (helm-get-selection) (car victims)))
|
;; No candidates are marked; we are refiling the entry at point
|
;; to the selected heading
|
(setq victims (list (point))))
|
;; Probably best to check that everything returned a value
|
(when (and victims buffer filename rfloc)
|
(cl-loop for victim in victims
|
do (org-with-point-at victim
|
(org-refile nil nil rfloc))))))
|
|
(defun helm-org-in-buffer-preselect ()
|
(if (org-on-heading-p)
|
(buffer-substring-no-properties (point-at-bol) (point-at-eol))
|
(save-excursion
|
(outline-previous-visible-heading 1)
|
(buffer-substring-no-properties (point-at-bol) (point-at-eol)))))
|
|
(defun helm-org-run-refile-heading-to ()
|
(interactive)
|
(with-helm-alive-p
|
(helm-exit-and-execute-action 'helm-org--refile-heading-to)))
|
(put 'helm-org-run-refile-heading-to 'helm-only t)
|
|
;;;###autoload
|
(defun helm-org-agenda-files-headings ()
|
"Preconfigured helm for org files headings."
|
(interactive)
|
(let ((autosaves (cl-loop for f in (org-agenda-files)
|
when (file-exists-p
|
(expand-file-name
|
(concat "#" (helm-basename f) "#")
|
(helm-basedir f)))
|
collect (helm-basename f))))
|
(when (or (null autosaves)
|
helm-org-ignore-autosaves
|
(y-or-n-p (format "%s have auto save data, continue?"
|
(mapconcat 'identity autosaves ", "))))
|
(helm :sources (helm-source-org-headings-for-files (org-agenda-files))
|
:candidate-number-limit 99999
|
:truncate-lines helm-org-truncate-lines
|
:buffer "*helm org headings*"))))
|
|
;;;###autoload
|
(defun helm-org-in-buffer-headings ()
|
"Preconfigured helm for org buffer headings."
|
(interactive)
|
(let (helm-org-show-filename)
|
(helm :sources (helm-source-org-headings-for-files
|
(list (current-buffer)))
|
:candidate-number-limit 99999
|
:preselect (helm-org-in-buffer-preselect)
|
:truncate-lines helm-org-truncate-lines
|
:buffer "*helm org inbuffer*")))
|
|
;;;###autoload
|
(defun helm-org-parent-headings ()
|
"Preconfigured helm for org headings that are parents of the
|
current heading."
|
(interactive)
|
;; Use a large max-depth to ensure all parents are displayed.
|
(let ((helm-org-headings-min-depth 1)
|
(helm-org-headings-max-depth 50))
|
(helm :sources (helm-source-org-headings-for-files
|
(list (current-buffer)) t)
|
:candidate-number-limit 99999
|
:truncate-lines helm-org-truncate-lines
|
:buffer "*helm org parent headings*")))
|
|
;;;###autoload
|
(defun helm-org-capture-templates ()
|
"Preconfigured helm for org templates."
|
(interactive)
|
(helm :sources (helm-source-org-capture-templates)
|
:candidate-number-limit 99999
|
:truncate-lines helm-org-truncate-lines
|
:buffer "*helm org capture templates*"))
|
|
;;; Org tag completion
|
|
;; Based on code from Anders Johansson posted on 3 Mar 2016 at
|
;; <https://groups.google.com/d/msg/emacs-helm/tA6cn6TUdRY/G1S3TIdzBwAJ>
|
|
(defvar crm-separator)
|
|
;;;###autoload
|
(defun helm-org-completing-read-tags (prompt collection pred req initial
|
hist def inherit-input-method _name _buffer)
|
"Completing read function for Org tags.
|
|
This function is used as a `completing-read' function in
|
`helm-completing-read-handlers-alist' by `org-set-tags' and
|
`org-capture'.
|
|
NOTE: Org tag completion will work only if you disable org fast tag
|
selection, see (info \"(org) setting tags\")."
|
(if (not (string= "Tags: " prompt))
|
;; Not a tags prompt. Use normal completion by calling
|
;; `org-icompleting-read' again without this function in
|
;; `helm-completing-read-handlers-alist'
|
(let ((helm-completing-read-handlers-alist
|
(rassq-delete-all
|
'helm-org-completing-read-tags
|
(copy-alist helm-completing-read-handlers-alist))))
|
(org-icompleting-read
|
prompt collection pred req initial hist def inherit-input-method))
|
;; Tags prompt
|
(let* ((curr (and (stringp initial)
|
(not (string= initial ""))
|
(org-split-string initial ":")))
|
(table (delete curr
|
(org-uniquify
|
(mapcar 'car org-last-tags-completion-table))))
|
(crm-separator ":\\|,\\|\\s-"))
|
(cl-letf (((symbol-function 'crm-complete-word)
|
'self-insert-command))
|
(mapconcat 'identity
|
(completing-read-multiple
|
prompt table pred nil initial hist def)
|
":")))))
|
|
(provide 'helm-org)
|
|
;; Local Variables:
|
;; byte-compile-warnings: (not obsolete)
|
;; coding: utf-8
|
;; indent-tabs-mode: nil
|
;; End:
|
|
;;; helm-org.el ends here
|