emacs: make "+" and "-" tagging operations in notmuch-search more flexible

Before the change, "+" and "-" tagging operations in notmuch-search
view accepted only a single tag.  The patch makes them use the
recently added `notmuch-read-tag-changes' function (renamed
`notmuch-select-tags-with-completion'), which allows to enter multiple
tags with "+" and "-" prefixes.  So after the change, "+" and "-"
bindings in notmuch-search view allow to both add and remove multiple
tags.  The only difference between "+" and "-" is the minibuffer
initial input ("+" and "-" respectively).
This commit is contained in:
Dmitry Kurochkin 2012-02-05 11:13:44 +04:00 committed by David Bremner
parent b3eafaf725
commit 2beaefa2ec

View file

@ -76,38 +76,56 @@ For example:
(defvar notmuch-query-history nil (defvar notmuch-query-history nil
"Variable to store minibuffer history for notmuch queries") "Variable to store minibuffer history for notmuch queries")
(defun notmuch-tag-completions (&optional prefixes search-terms) (defun notmuch-tag-completions (&optional search-terms)
(let ((tag-list (split-string
(split-string (with-output-to-string
(with-output-to-string (with-current-buffer standard-output
(with-current-buffer standard-output (apply 'call-process notmuch-command nil t
(apply 'call-process notmuch-command nil t nil "search-tags" search-terms)))
nil "search-tags" search-terms))) "\n+" t))
"\n+" t)))
(if (null prefixes)
tag-list
(apply #'append
(mapcar (lambda (tag)
(mapcar (lambda (prefix)
(concat prefix tag)) prefixes))
tag-list)))))
(defun notmuch-select-tag-with-completion (prompt &rest search-terms) (defun notmuch-select-tag-with-completion (prompt &rest search-terms)
(let ((tag-list (notmuch-tag-completions nil search-terms))) (let ((tag-list (notmuch-tag-completions search-terms)))
(completing-read prompt tag-list))) (completing-read prompt tag-list)))
(defun notmuch-select-tags-with-completion (prompt &optional prefixes &rest search-terms) (defun notmuch-read-tag-changes (&optional initial-input &rest search-terms)
(let ((tag-list (notmuch-tag-completions prefixes search-terms)) (let* ((all-tag-list (notmuch-tag-completions))
(crm-separator " ") (add-tag-list (mapcar (apply-partially 'concat "+") all-tag-list))
;; By default, space is bound to "complete word" function. (remove-tag-list (mapcar (apply-partially 'concat "-")
;; Re-bind it to insert a space instead. Note that <tab> (if (null search-terms)
;; still does the completion. all-tag-list
(crm-local-completion-map (notmuch-tag-completions search-terms))))
(let ((map (make-sparse-keymap))) (tag-list (append add-tag-list remove-tag-list))
(set-keymap-parent map crm-local-completion-map) (crm-separator " ")
(define-key map " " 'self-insert-command) ;; By default, space is bound to "complete word" function.
map))) ;; Re-bind it to insert a space instead. Note that <tab>
(delete "" (completing-read-multiple prompt tag-list)))) ;; still does the completion.
(crm-local-completion-map
(let ((map (make-sparse-keymap)))
(set-keymap-parent map crm-local-completion-map)
(define-key map " " 'self-insert-command)
map)))
(delete "" (completing-read-multiple "Tags (+add -drop): "
tag-list nil nil initial-input))))
(defun notmuch-update-tags (tags tag-changes)
"Return a copy of TAGS with additions and removals from TAG-CHANGES.
TAG-CHANGES must be a list of tags names, each prefixed with
either a \"+\" to indicate the tag should be added to TAGS if not
present or a \"-\" to indicate that the tag should be removed
from TAGS if present."
(let ((result-tags (copy-sequence tags)))
(dolist (tag-change tag-changes)
(let ((op (string-to-char tag-change))
(tag (unless (string= tag-change "") (substring tag-change 1))))
(case op
(?+ (unless (member tag result-tags)
(push tag result-tags)))
(?- (setq result-tags (delete tag result-tags)))
(otherwise
(error "Changed tag must be of the form `+this_tag' or `-that_tag'")))))
(sort result-tags 'string<)))
(defun notmuch-foreach-mime-part (function mm-handle) (defun notmuch-foreach-mime-part (function mm-handle)
(cond ((stringp (car mm-handle)) (cond ((stringp (car mm-handle))
@ -447,6 +465,10 @@ Complete list of currently available key bindings:
"Return a list of threads for the current region" "Return a list of threads for the current region"
(notmuch-search-properties-in-region 'notmuch-search-thread-id beg end)) (notmuch-search-properties-in-region 'notmuch-search-thread-id beg end))
(defun notmuch-search-find-thread-id-region-search (beg end)
"Return a search string for threads for the current region"
(mapconcat 'identity (notmuch-search-find-thread-id-region beg end) " or "))
(defun notmuch-search-find-authors () (defun notmuch-search-find-authors ()
"Return the authors for the current thread" "Return the authors for the current thread"
(get-text-property (point) 'notmuch-search-authors)) (get-text-property (point) 'notmuch-search-authors))
@ -584,74 +606,53 @@ the messages that were tagged"
(forward-line 1)) (forward-line 1))
output))) output)))
(defun notmuch-search-add-tag-thread (tag) (defun notmuch-search-tag-thread (&rest tags)
(notmuch-search-add-tag-region tag (point) (point))) "Change tags for the currently selected thread.
(defun notmuch-search-add-tag-region (tag beg end) See `notmuch-search-tag-region' for details."
(let ((search-id-string (mapconcat 'identity (notmuch-search-find-thread-id-region beg end) " or "))) (apply 'notmuch-search-tag-region (point) (point) tags))
(notmuch-tag search-id-string (concat "+" tag))
(defun notmuch-search-tag-region (beg end &rest tags)
"Change tags for threads in the given region.
TAGS is a list of tag operations for `notmuch-tag'. The tags are
added or removed for all threads in the region from BEG to END."
(let ((search-string (notmuch-search-find-thread-id-region-search beg end)))
(apply 'notmuch-tag search-string tags)
(save-excursion (save-excursion
(let ((last-line (line-number-at-pos end)) (let ((last-line (line-number-at-pos end))
(max-line (- (line-number-at-pos (point-max)) 2))) (max-line (- (line-number-at-pos (point-max)) 2)))
(goto-char beg) (goto-char beg)
(while (<= (line-number-at-pos) (min last-line max-line)) (while (<= (line-number-at-pos) (min last-line max-line))
(notmuch-search-set-tags (delete-dups (sort (cons tag (notmuch-search-get-tags)) 'string<))) (notmuch-search-set-tags
(notmuch-update-tags (notmuch-search-get-tags) tags))
(forward-line)))))) (forward-line))))))
(defun notmuch-search-remove-tag-thread (tag) (defun notmuch-search-tag (&optional initial-input)
(notmuch-search-remove-tag-region tag (point) (point))) "Change tags for the currently selected thread or region."
(interactive)
(let* ((beg (if (region-active-p) (region-beginning) (point)))
(end (if (region-active-p) (region-end) (point)))
(search-string (notmuch-search-find-thread-id-region-search beg end))
(tags (notmuch-read-tag-changes initial-input search-string)))
(apply 'notmuch-search-tag-region beg end tags)))
(defun notmuch-search-remove-tag-region (tag beg end) (defun notmuch-search-add-tag ()
(let ((search-id-string (mapconcat 'identity (notmuch-search-find-thread-id-region beg end) " or "))) "Same as `notmuch-search-tag' but sets initial input to '+'."
(notmuch-tag search-id-string (concat "-" tag)) (interactive)
(save-excursion (notmuch-search-tag "+"))
(let ((last-line (line-number-at-pos end))
(max-line (- (line-number-at-pos (point-max)) 2)))
(goto-char beg)
(while (<= (line-number-at-pos) (min last-line max-line))
(notmuch-search-set-tags (delete tag (notmuch-search-get-tags)))
(forward-line))))))
(defun notmuch-search-add-tag (tag) (defun notmuch-search-remove-tag ()
"Add a tag to the currently selected thread or region. "Same as `notmuch-search-tag' but sets initial input to '-'."
(interactive)
The tag is added to all messages in the currently selected thread (notmuch-search-tag "-"))
or threads in the current region."
(interactive
(list (notmuch-select-tag-with-completion "Tag to add: ")))
(save-excursion
(if (region-active-p)
(let* ((beg (region-beginning))
(end (region-end)))
(notmuch-search-add-tag-region tag beg end))
(notmuch-search-add-tag-thread tag))))
(defun notmuch-search-remove-tag (tag)
"Remove a tag from the currently selected thread or region.
The tag is removed from all messages in the currently selected
thread or threads in the current region."
(interactive
(list (notmuch-select-tag-with-completion
"Tag to remove: "
(if (region-active-p)
(mapconcat 'identity
(notmuch-search-find-thread-id-region (region-beginning) (region-end))
" ")
(notmuch-search-find-thread-id)))))
(save-excursion
(if (region-active-p)
(let* ((beg (region-beginning))
(end (region-end)))
(notmuch-search-remove-tag-region tag beg end))
(notmuch-search-remove-tag-thread tag))))
(defun notmuch-search-archive-thread () (defun notmuch-search-archive-thread ()
"Archive the currently selected thread (remove its \"inbox\" tag). "Archive the currently selected thread (remove its \"inbox\" tag).
This function advances the next thread when finished." This function advances the next thread when finished."
(interactive) (interactive)
(notmuch-search-remove-tag-thread "inbox") (notmuch-search-tag-thread "-inbox")
(notmuch-search-next-thread)) (notmuch-search-next-thread))
(defvar notmuch-search-process-filter-data nil (defvar notmuch-search-process-filter-data nil
@ -886,9 +887,7 @@ will prompt for tags to be added or removed. Tags prefixed with
Each character of the tag name may consist of alphanumeric Each character of the tag name may consist of alphanumeric
characters as well as `_.+-'. characters as well as `_.+-'.
" "
(interactive (notmuch-select-tags-with-completion (interactive (notmuch-read-tag-changes))
"Operations (+add -drop): notmuch tag "
'("+" "-")))
(apply 'notmuch-tag notmuch-search-query-string actions)) (apply 'notmuch-tag notmuch-search-query-string actions))
(defun notmuch-search-buffer-title (query) (defun notmuch-search-buffer-title (query)