2012-04-14 20:52:50 +02:00
|
|
|
;; notmuch-tag.el --- tag messages within emacs
|
|
|
|
;;
|
2013-03-23 12:29:54 +01:00
|
|
|
;; Copyright © Damien Cassou
|
2012-04-14 20:52:50 +02:00
|
|
|
;; Copyright © Carl Worth
|
|
|
|
;;
|
|
|
|
;; This file is part of Notmuch.
|
|
|
|
;;
|
|
|
|
;; Notmuch 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.
|
|
|
|
;;
|
|
|
|
;; Notmuch 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 Notmuch. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
;;
|
|
|
|
;; Authors: Carl Worth <cworth@cworth.org>
|
2013-03-23 12:29:54 +01:00
|
|
|
;; Damien Cassou <damien.cassou@gmail.com>
|
|
|
|
;;
|
|
|
|
;;; Code:
|
|
|
|
;;
|
2012-04-14 20:52:50 +02:00
|
|
|
|
2013-03-23 12:29:54 +01:00
|
|
|
(require 'cl)
|
2012-04-14 20:52:50 +02:00
|
|
|
(require 'crm)
|
|
|
|
(require 'notmuch-lib)
|
|
|
|
|
2013-03-23 12:29:54 +01:00
|
|
|
(defcustom notmuch-tag-formats
|
|
|
|
'(("unread" (propertize tag 'face '(:foreground "red")))
|
|
|
|
("flagged" (notmuch-tag-format-image-data tag (notmuch-tag-star-icon))))
|
|
|
|
"Custom formats for individual tags.
|
|
|
|
|
|
|
|
This gives a list that maps from tag names to lists of formatting
|
|
|
|
expressions. The car of each element gives a tag name and the
|
|
|
|
cdr gives a list of Elisp expressions that modify the tag. If
|
|
|
|
the list is empty, the tag will simply be hidden. Otherwise,
|
|
|
|
each expression will be evaluated in order: for the first
|
|
|
|
expression, the variable `tag' will be bound to the tag name; for
|
|
|
|
each later expression, the variable `tag' will be bound to the
|
|
|
|
result of the previous expression. In this way, each expression
|
|
|
|
can build on the formatting performed by the previous expression.
|
|
|
|
The result of the last expression will displayed in place of the
|
|
|
|
tag.
|
|
|
|
|
|
|
|
For example, to replace a tag with another string, simply use
|
|
|
|
that string as a formatting expression. To change the foreground
|
|
|
|
of a tag to red, use the expression
|
|
|
|
(propertize tag 'face '(:foreground \"red\"))
|
|
|
|
|
|
|
|
See also `notmuch-tag-format-image', which can help replace tags
|
|
|
|
with images."
|
|
|
|
|
|
|
|
:group 'notmuch-search
|
|
|
|
:group 'notmuch-show
|
|
|
|
:type '(alist :key-type (string :tag "Tag")
|
|
|
|
:extra-offset -3
|
|
|
|
:value-type
|
|
|
|
(radio :format "%v"
|
|
|
|
(const :tag "Hidden" nil)
|
|
|
|
(set :tag "Modified"
|
|
|
|
(string :tag "Display as")
|
|
|
|
(list :tag "Face" :extra-offset -4
|
|
|
|
(const :format "" :inline t
|
|
|
|
(propertize tag 'face))
|
|
|
|
(list :format "%v"
|
|
|
|
(const :format "" quote)
|
|
|
|
custom-face-edit))
|
|
|
|
(list :format "%v" :extra-offset -4
|
|
|
|
(const :format "" :inline t
|
|
|
|
(notmuch-tag-format-image-data tag))
|
|
|
|
(choice :tag "Image"
|
|
|
|
(const :tag "Star"
|
|
|
|
(notmuch-tag-star-icon))
|
|
|
|
(const :tag "Empty star"
|
|
|
|
(notmuch-tag-star-empty-icon))
|
|
|
|
(const :tag "Tag"
|
|
|
|
(notmuch-tag-tag-icon))
|
|
|
|
(string :tag "Custom")))
|
|
|
|
(sexp :tag "Custom")))))
|
|
|
|
|
|
|
|
(defun notmuch-tag-format-image-data (tag data)
|
|
|
|
"Replace TAG with image DATA, if available.
|
|
|
|
|
|
|
|
This function returns a propertized string that will display image
|
|
|
|
DATA in place of TAG.This is designed for use in
|
|
|
|
`notmuch-tag-formats'.
|
|
|
|
|
|
|
|
DATA is the content of an SVG picture (e.g., as returned by
|
|
|
|
`notmuch-tag-star-icon')."
|
|
|
|
(propertize tag 'display
|
|
|
|
`(image :type svg
|
|
|
|
:data ,data
|
|
|
|
:ascent center
|
|
|
|
:mask heuristic)))
|
|
|
|
|
|
|
|
(defun notmuch-tag-star-icon ()
|
|
|
|
"Return SVG data representing a star icon.
|
|
|
|
This can be used with `notmuch-tag-format-image-data'."
|
|
|
|
"<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"no\"?>
|
|
|
|
<svg version=\"1.1\" width=\"16\" height=\"16\">
|
|
|
|
<g transform=\"translate(-242.81601,-315.59635)\">
|
|
|
|
<path
|
|
|
|
d=\"m 290.25762,334.31206 -17.64143,-11.77975 -19.70508,7.85447 5.75171,-20.41814 -13.55925,-16.31348 21.19618,-0.83936 11.325,-17.93675 7.34825,19.89939 20.55849,5.22795 -16.65471,13.13786 z\"
|
|
|
|
transform=\"matrix(0.2484147,-0.02623394,0.02623394,0.2484147,174.63605,255.37691)\"
|
|
|
|
style=\"fill:#ffff00;fill-rule:evenodd;stroke:#000000;stroke-width:1px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1\" />
|
|
|
|
</g>
|
|
|
|
</svg>")
|
|
|
|
|
|
|
|
(defun notmuch-tag-star-empty-icon ()
|
|
|
|
"Return SVG data representing an empty star icon.
|
|
|
|
This can be used with `notmuch-tag-format-image-data'."
|
|
|
|
"<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"no\"?>
|
|
|
|
<svg version=\"1.1\" width=\"16\" height=\"16\">
|
|
|
|
<g transform=\"translate(-242.81601,-315.59635)\">
|
|
|
|
<path
|
|
|
|
d=\"m 290.25762,334.31206 -17.64143,-11.77975 -19.70508,7.85447 5.75171,-20.41814 -13.55925,-16.31348 21.19618,-0.83936 11.325,-17.93675 7.34825,19.89939 20.55849,5.22795 -16.65471,13.13786 z\"
|
|
|
|
transform=\"matrix(0.2484147,-0.02623394,0.02623394,0.2484147,174.63605,255.37691)\"
|
|
|
|
style=\"fill:#d6d6d1;fill-opacity:1;fill-rule:evenodd;stroke:#000000;stroke-width:1px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1\" />
|
|
|
|
</g>
|
|
|
|
</svg>")
|
|
|
|
|
|
|
|
(defun notmuch-tag-tag-icon ()
|
|
|
|
"Return SVG data representing a tag icon.
|
|
|
|
This can be used with `notmuch-tag-format-image-data'."
|
|
|
|
"<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"no\"?>
|
|
|
|
<svg version=\"1.1\" width=\"16\" height=\"16\">
|
|
|
|
<g transform=\"translate(0,-1036.3622)\">
|
|
|
|
<path
|
|
|
|
d=\"m 0.44642857,1040.9336 12.50000043,0 2.700893,3.6161 -2.700893,3.616 -12.50000043,0 z\"
|
|
|
|
style=\"fill:#ffff00;fill-opacity:1;fill-rule:evenodd;stroke:#000000;stroke-width:0.25;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1\" />
|
|
|
|
</g>
|
|
|
|
</svg>")
|
|
|
|
|
|
|
|
(defun notmuch-tag-format-tag (tag)
|
|
|
|
"Format TAG by looking into `notmuch-tag-formats'."
|
|
|
|
(let ((formats (assoc tag notmuch-tag-formats)))
|
|
|
|
(cond
|
|
|
|
((null formats) ;; - Tag not in `notmuch-tag-formats',
|
|
|
|
tag) ;; the format is the tag itself.
|
|
|
|
((null (cdr formats)) ;; - Tag was deliberately hidden,
|
|
|
|
nil) ;; no format must be returned
|
|
|
|
(t ;; - Tag was found and has formats,
|
|
|
|
(let ((tag tag)) ;; we must apply all the formats.
|
|
|
|
(dolist (format (cdr formats) tag)
|
|
|
|
(setq tag (eval format))))))))
|
|
|
|
|
|
|
|
(defun notmuch-tag-format-tags (tags)
|
|
|
|
"Return a string representing formatted TAGS."
|
|
|
|
(notmuch-combine-face-text-property-string
|
|
|
|
(mapconcat #'identity
|
|
|
|
;; nil indicated that the tag was deliberately hidden
|
|
|
|
(delq nil (mapcar #'notmuch-tag-format-tag tags))
|
|
|
|
" ")
|
|
|
|
'notmuch-tag-face
|
|
|
|
t))
|
|
|
|
|
2012-04-14 20:52:50 +02:00
|
|
|
(defcustom notmuch-before-tag-hook nil
|
|
|
|
"Hooks that are run before tags of a message are modified.
|
|
|
|
|
|
|
|
'tags' will contain the tags that are about to be added or removed as
|
|
|
|
a list of strings of the form \"+TAG\" or \"-TAG\".
|
|
|
|
'query' will be a string containing the search query that determines
|
|
|
|
the messages that are about to be tagged"
|
|
|
|
|
|
|
|
:type 'hook
|
|
|
|
:options '(notmuch-hl-line-mode)
|
|
|
|
:group 'notmuch-hooks)
|
|
|
|
|
|
|
|
(defcustom notmuch-after-tag-hook nil
|
|
|
|
"Hooks that are run after tags of a message are modified.
|
|
|
|
|
|
|
|
'tags' will contain the tags that were added or removed as
|
|
|
|
a list of strings of the form \"+TAG\" or \"-TAG\".
|
|
|
|
'query' will be a string containing the search query that determines
|
|
|
|
the messages that were tagged"
|
|
|
|
:type 'hook
|
|
|
|
:options '(notmuch-hl-line-mode)
|
|
|
|
:group 'notmuch-hooks)
|
|
|
|
|
|
|
|
(defvar notmuch-select-tag-history nil
|
|
|
|
"Variable to store minibuffer history for
|
|
|
|
`notmuch-select-tag-with-completion' function.")
|
|
|
|
|
|
|
|
(defvar notmuch-read-tag-changes-history nil
|
|
|
|
"Variable to store minibuffer history for
|
|
|
|
`notmuch-read-tag-changes' function.")
|
|
|
|
|
|
|
|
(defun notmuch-tag-completions (&optional search-terms)
|
|
|
|
(if (null search-terms)
|
|
|
|
(setq search-terms (list "*")))
|
|
|
|
(split-string
|
|
|
|
(with-output-to-string
|
|
|
|
(with-current-buffer standard-output
|
|
|
|
(apply 'call-process notmuch-command nil t
|
|
|
|
nil "search" "--output=tags" "--exclude=false" search-terms)))
|
|
|
|
"\n+" t))
|
|
|
|
|
|
|
|
(defun notmuch-select-tag-with-completion (prompt &rest search-terms)
|
|
|
|
(let ((tag-list (notmuch-tag-completions search-terms)))
|
|
|
|
(completing-read prompt tag-list nil nil nil 'notmuch-select-tag-history)))
|
|
|
|
|
|
|
|
(defun notmuch-read-tag-changes (&optional initial-input &rest search-terms)
|
|
|
|
(let* ((all-tag-list (notmuch-tag-completions))
|
|
|
|
(add-tag-list (mapcar (apply-partially 'concat "+") all-tag-list))
|
|
|
|
(remove-tag-list (mapcar (apply-partially 'concat "-")
|
|
|
|
(if (null search-terms)
|
|
|
|
all-tag-list
|
|
|
|
(notmuch-tag-completions search-terms))))
|
|
|
|
(tag-list (append add-tag-list remove-tag-list))
|
|
|
|
(crm-separator " ")
|
|
|
|
;; By default, space is bound to "complete word" function.
|
|
|
|
;; Re-bind it to insert a space instead. Note that <tab>
|
|
|
|
;; 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
|
|
|
|
'notmuch-read-tag-changes-history))))
|
|
|
|
|
|
|
|
(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<)))
|
|
|
|
|
2012-04-14 22:49:07 +02:00
|
|
|
(defun notmuch-tag (query &optional tag-changes)
|
2012-04-14 20:52:50 +02:00
|
|
|
"Add/remove tags in TAG-CHANGES to messages matching QUERY.
|
|
|
|
|
2012-04-14 22:49:07 +02:00
|
|
|
QUERY should be a string containing the search-terms.
|
|
|
|
TAG-CHANGES can take multiple forms. If TAG-CHANGES is a list of
|
|
|
|
strings of the form \"+tag\" or \"-tag\" then those are the tag
|
|
|
|
changes applied. If TAG-CHANGES is a string then it is
|
|
|
|
interpreted as a single tag change. If TAG-CHANGES is the string
|
|
|
|
\"-\" or \"+\", or null, then the user is prompted to enter the
|
|
|
|
tag changes.
|
2012-04-14 20:52:50 +02:00
|
|
|
|
|
|
|
Note: Other code should always use this function alter tags of
|
|
|
|
messages instead of running (notmuch-call-notmuch-process \"tag\" ..)
|
|
|
|
directly, so that hooks specified in notmuch-before-tag-hook and
|
|
|
|
notmuch-after-tag-hook will be run."
|
|
|
|
;; Perform some validation
|
2012-04-14 22:49:07 +02:00
|
|
|
(if (string-or-null-p tag-changes)
|
|
|
|
(if (or (string= tag-changes "-") (string= tag-changes "+") (null tag-changes))
|
|
|
|
(setq tag-changes (notmuch-read-tag-changes tag-changes query))
|
|
|
|
(setq tag-changes (list tag-changes))))
|
2012-04-14 20:52:50 +02:00
|
|
|
(mapc (lambda (tag-change)
|
|
|
|
(unless (string-match-p "^[-+]\\S-+$" tag-change)
|
|
|
|
(error "Tag must be of the form `+this_tag' or `-that_tag'")))
|
|
|
|
tag-changes)
|
|
|
|
(unless (null tag-changes)
|
|
|
|
(run-hooks 'notmuch-before-tag-hook)
|
|
|
|
(apply 'notmuch-call-notmuch-process "tag"
|
|
|
|
(append tag-changes (list "--" query)))
|
2012-04-14 22:49:07 +02:00
|
|
|
(run-hooks 'notmuch-after-tag-hook))
|
|
|
|
;; in all cases we return tag-changes as a list
|
|
|
|
tag-changes)
|
2012-04-14 20:52:50 +02:00
|
|
|
|
2012-09-06 17:32:37 +02:00
|
|
|
(defun notmuch-tag-change-list (tags &optional reverse)
|
|
|
|
"Convert TAGS into a list of tag changes.
|
|
|
|
|
|
|
|
Add a \"+\" prefix to any tag in TAGS list that doesn't already
|
|
|
|
begin with a \"+\" or a \"-\". If REVERSE is non-nil, replace all
|
|
|
|
\"+\" prefixes with \"-\" and vice versa in the result."
|
|
|
|
(mapcar (lambda (str)
|
|
|
|
(let ((s (if (string-match "^[+-]" str) str (concat "+" str))))
|
|
|
|
(if reverse
|
|
|
|
(concat (if (= (string-to-char s) ?-) "+" "-")
|
|
|
|
(substring s 1))
|
|
|
|
s)))
|
|
|
|
tags))
|
|
|
|
|
|
|
|
|
2012-04-14 20:52:50 +02:00
|
|
|
;;
|
|
|
|
|
|
|
|
(provide 'notmuch-tag)
|
2013-05-14 10:13:53 +02:00
|
|
|
|
|
|
|
;; Local Variables:
|
|
|
|
;; byte-compile-warnings: (not cl-functions)
|
|
|
|
;; End:
|