summaryrefslogtreecommitdiffhomepage
path: root/lib
diff options
context:
space:
mode:
Diffstat (limited to 'lib')
-rw-r--r--lib/net/jao-notmuch.el216
1 files changed, 216 insertions, 0 deletions
diff --git a/lib/net/jao-notmuch.el b/lib/net/jao-notmuch.el
new file mode 100644
index 0000000..5916840
--- /dev/null
+++ b/lib/net/jao-notmuch.el
@@ -0,0 +1,216 @@
+;;; jao-notmuch.el --- Extensions for notmuch -*- lexical-binding: t; -*-
+
+;; Copyright (C) 2021 jao
+
+;; Author: jao <mail@jao.io>
+;; Keywords: mail
+
+;; 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 <https://www.gnu.org/licenses/>.
+
+;;; Commentary:
+
+;; Extensions to vanilla notmuch, mostly for tree view
+
+;;; Code:
+
+(require 'notmuch)
+(require 'notmuch-tree)
+(require 'notmuch-show)
+
+
+;; Moving messages around
+
+(defvar jao-notmuch-mailboxes nil)
+(defvar jao-notmuch-mailboxes-rx nil)
+
+(defun jao-notmuch--path-to-mailbox (full-path)
+ (unless jao-notmuch-mailboxes-rx
+ (setq jao-notmuch-mailboxes-rx (regexp-opt jao-notmuch-mailboxes)))
+ (if (string-match jao-notmuch-mailboxes-rx full-path)
+ (match-string 0 full-path)
+ (user-error "Message not in any mailbox!")))
+
+(defun jao-notmuch-msg-props ()
+ (if-let ((p (save-excursion
+ (text-property-search-forward :notmuch-message-properties))))
+ (prop-match-value p)
+ (user-error "No message at point")))
+
+(defun jao-notmuch--move (full-path &optional d)
+ (let* ((ff (jao-notmuch--path-to-mailbox full-path))
+ (d (or d (completing-read (format "From %s to: " ff)
+ (remove ff jao-notmuch-mailboxes)
+ nil
+ t)))
+ (dest (string-replace ff d full-path))
+ (dest (replace-regexp-in-string ",U=.+$" "" dest))
+ (fnd (file-name-nondirectory full-path))
+ (dnd (file-name-nondirectory dest))
+ (prompt (format "%s/%s -> %s/%s? " ff fnd d dnd)))
+ (when (y-or-n-p prompt)
+ (let ((c (notmuch-tag-change-list (split-string ff "/") t))
+ (cc (notmuch-tag-change-list (split-string d "/"))))
+ (notmuch-tree-tag (append c cc)))
+ (notmuch-tree-close-message-window)
+ (rename-file full-path dest)
+ (shell-command-to-string "notmuch new")
+ (notmuch-refresh-this-buffer))))
+
+(defun jao-notmuch-move-message ()
+ (interactive)
+ (let* ((p (jao-notmuch-msg-props))
+ (f (car (plist-get p :filename))))
+ (jao-notmuch--move (or f ""))))
+
+(defun jao-notmuch-tree-spam ()
+ (interactive)
+ (let* ((p (jao-notmuch-msg-props))
+ (f (car (plist-get p :filename)))
+ (mbox (car (split-string (jao-notmuch--path-to-mailbox f) "/"))))
+ (notmuch-tree-tag '("-new" "-unread"))
+ (jao-notmuch--move f (concat mbox "/spam"))))
+
+
+;; Targetting the displayed message from the tree view
+
+(defvar-local jao-notmuch--tree-buffer nil)
+
+(defun jao-notmuch-goto-message-buffer (&optional and-click)
+ (interactive "P")
+ (when (window-live-p notmuch-tree-message-window)
+ (let ((b (current-buffer)))
+ (select-window notmuch-tree-message-window)
+ (setq-local jao-notmuch--tree-buffer b)
+ (or (not and-click)
+ (cond ((eww--url-at-point) (shr-browse-url) 'url)
+ ((button-at (point)) (push-button) 'button))))))
+
+(defun jao-notmuch-click-message-buffer ()
+ (interactive)
+ (let ((b (current-buffer)))
+ (unless (eq 'url (jao-notmuch-goto-message-buffer t))
+ (pop-to-buffer b))))
+
+(defun jao-notmuch-goto-index-buffer ()
+ (interactive)
+ (if (buffer-live-p jao-notmuch--tree-buffer)
+ (pop-to-buffer jao-notmuch--tree-buffer)
+ (user-error "No index for this buffer")))
+
+(defun jao-notmuch-browse-urls ()
+ (interactive)
+ (when (or (derived-mode-p 'notmuch-show-mode)
+ (jao-notmuch-goto-message-buffer))
+ (notmuch-show-browse-urls)))
+
+(defun jao-notmuch-toggle-mime-parts ()
+ (interactive)
+ (when (jao-notmuch-goto-message-buffer)
+ (goto-char (point-min))
+ (let ((notmuch-show-text/html-blocked-images nil)
+ (shr-inhibit-images nil)
+ (shr-blocked-images nil))
+ (save-excursion
+ (when (re-search-forward "\\[ multipart/alternative \\]" nil t)
+ (while (forward-button 1 nil nil t)
+ (notmuch-show-toggle-part-invisibility)))))
+ (jao-notmuch-goto-index-buffer)))
+
+(defun jao-notmuch-toggle-images ()
+ (interactive)
+ (save-window-excursion
+ (jao-notmuch-goto-message-buffer)
+ (when (derived-mode-p 'notmuch-show-mode)
+ (let ((notmuch-show-text/html-blocked-images nil)
+ (shr-inhibit-images nil)
+ (shr-blocked-images nil))
+ (notmuch-refresh-this-buffer)))))
+
+
+;; Keeping track of unread messages in current tree view
+
+(defun jao-notmuch--unread-count ()
+ (save-excursion
+ (goto-char (point-min))
+ (let ((p) (cnt))
+ (while (setq p (text-property-search-forward
+ :notmuch-message-properties))
+ (unless cnt (setq cnt 0))
+ (let ((tags (plist-get (prop-match-value p) :tags)))
+ (when (or (member "unread" tags) (member "new" tags))
+ (setq cnt (1+ cnt)))))
+ cnt)))
+
+(defun jao-notmuch-tree-update-buffer-name (&optional n)
+ (when-let ((n (or n (jao-notmuch--unread-count))))
+ (rename-buffer (format "*%s - {%s messages left}*"
+ notmuch-tree-basic-query
+ n))
+ (jao-minibuffer-refresh)))
+
+(defun jao-notmuch-tree-next (thread &optional no-exit)
+ "Next message or thread in forest or exit if none."
+ (interactive "P")
+ (if thread
+ (progn (notmuch-tree-next-thread)
+ (notmuch-tree-show-message nil))
+ (notmuch-tree-next-matching-message (not no-exit)))
+ (jao-notmuch-tree-update-buffer-name))
+
+;; Tagging
+
+(defun jao-notmuch-tag-jump-and-next (reverse)
+ (interactive "P")
+ (notmuch-tag-jump reverse)
+ (jao-notmuch-tree-next nil t))
+
+(defun jao-notmuch-tree--tag-and-next (tags reverse whole-thread)
+ (let ((c (notmuch-tag-change-list tags reverse)))
+ (if whole-thread (notmuch-tree-tag-thread c) (notmuch-tree-tag c)))
+ (jao-notmuch-tree-next whole-thread t))
+
+(defun jao-notmuch-tree-delete-message (undelete)
+ (interactive "P")
+ (jao-notmuch-tree--tag-and-next '("+deleted" "-new" "-unread") undelete nil))
+
+(defun jao-notmuch-tree-delete-thread ()
+ (interactive)
+ (jao-notmuch-tree--tag-and-next '("+deleted" "-new" "-unread") nil t))
+
+(defun jao-notmuch-tree-read-thread ()
+ (interactive)
+ (jao-notmuch-tree--tag-and-next '("-unread" "-new") nil t))
+
+;; Scrolling the shown message
+
+(defun jao-notmuch-tree-scroll-or-next ()
+ "Scroll or next message in forest or exit if none."
+ (interactive)
+ (if (notmuch-tree-scroll-message-window)
+ (notmuch-tree-next-matching-message t)
+ (when (not (window-live-p notmuch-tree-message-window))
+ (notmuch-tree-show-message nil)))
+ (jao-notmuch-tree-update-buffer-name))
+
+(defun jao-notmuch-tree-show-or-scroll ()
+ "Show current message, or scroll it if visible."
+ (interactive)
+ (if (window-live-p notmuch-tree-message-window)
+ (scroll-other-window 1)
+ (notmuch-tree-show-message nil)
+ (jao-notmuch-tree-update-buffer-name)))
+
+
+(provide 'jao-notmuch)
+;;; jao-notmuch.el ends here