emacs-elpa-diffs
[Top][All Lists]
Advanced

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

[elpa] externals/gnorb 70b5534 070/449: Make TODOs from outgoing message


From: Stefan Monnier
Subject: [elpa] externals/gnorb 70b5534 070/449: Make TODOs from outgoing messages
Date: Fri, 27 Nov 2020 23:15:10 -0500 (EST)

branch: externals/gnorb
commit 70b55345a3b93dd25c1aab2a6880bd8e2195524f
Author: Eric Abrahamsen <eric@ericabrahamsen.net>
Commit: Eric Abrahamsen <eric@ericabrahamsen.net>

    Make TODOs from outgoing messages
    
    lisp/gnorb-gnus.el: New function gnorb-gnus-outgoing-make-todo, and its
    helper function gnorb-gnus-outgoing-make-todo-1. Renamed
    `gnorb-gnus-check-org-header' to 'gnorb-gnus-check-outgoing-headers' to
    reflect its more general purpose. New defvar
    `gnorb-gnus-sending-message-info' where outgoing message information is
    stored. New user option `gnorb-gnus-new-todo-capture-key' to specify
    which capture template to use for outgoing messages.
    
    lisp/gnorb-org.el: New user option `gnorb-org-msg-id-key', specifying
    the name of the property where message ids are stored.
    
    README.org: Document the above.
---
 README.org         | 17 ++++++++++
 lisp/gnorb-gnus.el | 94 ++++++++++++++++++++++++++++++++++++++++++++++++++----
 lisp/gnorb-org.el  |  6 ++++
 3 files changed, 111 insertions(+), 6 deletions(-)

diff --git a/README.org b/README.org
index 66603a6..8410f2c 100644
--- a/README.org
+++ b/README.org
@@ -154,6 +154,13 @@ TODO, and the action will depend in part on the value of
 `gnorb-gnus-message-trigger-default', which see. Future versions of
 Gnorb will automagically suggest TODOs relevant to the incoming
 message.
+**** gnorb-gnus-outgoing-make-todo
+Call this while composing a new message (ie in message-mode), or
+immediately after sending a message, to create a new TODO from the
+outgoing message -- for instance, when you need to be sure you get a
+response. New TODOs are created using the capture process, and you'll
+need to specify a capture template to use for outgoing messages: see
+`gnorb-gnus-new-todo-capture-key'.
 *** User Options
 **** gnorb-gnus-mail-search-backend
 Specifies the search backend that you use for searching mails.
@@ -168,6 +175,16 @@ Set to either 'note or 'todo to tell 
`gnorb-gnus-message-trigger-todo'
 what to do by default. You can reach the non-default behavior by
 calling that function with a prefix argument. Alternately, set to
 'prompt to always prompt for the appropriate action.
+**** gnorb-gnus-new-todo-capture-key
+Set this to a single-character string pointing at an Org capture
+template to use when creating TODOs from outgoing messages. The
+template is a regular capture template, with a few exceptions. If Gnus
+helps you archive outgoing messages (ie you have
+`gnus-message-archive-group' set to something, and your outgoing
+messages have a "Fcc" header), a link to that message will be made,
+and you'll be able to use all the escapes related to gnus messages. If
+you don't archive outgoing messages, you'll still be able to use the
+%:subject, %:to, and %:date escapes in the capture template.
 *** Suggested Keybindings
 #+BEGIN_SRC emacs-lisp
   (eval-after-load "gnorb-gnus"
diff --git a/lisp/gnorb-gnus.el b/lisp/gnorb-gnus.el
index f292d68..df6f86e 100644
--- a/lisp/gnorb-gnus.el
+++ b/lisp/gnorb-gnus.el
@@ -72,6 +72,12 @@ Basically behave as if all attachments have 
\":gnus-attachments t\"."
   :group 'gnorb-gnus
   :type 'boolean)
 
+(defcustom gnorb-gnus-new-todo-capture-key nil
+  "Key for the capture template to use when creating a new TODO
+  from an outgoing message."
+  :group 'gnorb-gnus
+  :type 'string)
+
 (defcustom gnorb-gnus-trigger-refile-args '((org-agenda-files :maxlevel . 4))
   "A value to use as an equivalent of `org-refile-targets' (which
   see) when offering trigger targets for
@@ -224,12 +230,31 @@ save them into `gnorb-tmp-dir'."
 
 ;;; Storing, removing, and acting on Org headers in messages.
 
-(defun gnorb-gnus-check-org-header ()
-  "Return the value of the `gnorb-gnus-org-header' for the
-current message; multiple header values returned as a string."
+(defvar gnorb-gnus-sending-message-info nil
+  "Place to store the To, Subject, Date, and Message-ID headers
+  of the currently-sending or last-sent message.")
+
+(defun gnorb-gnus-check-outgoing-headers ()
+  "Save the value of the `gnorb-mail-header' for the current
+message; multiple header values returned as a string. Also save
+information about the outgoing message into
+`gnorb-gnus-sending-message-info'."
   (save-restriction
     (message-narrow-to-headers)
-    (let ((org-ids (mail-fetch-field gnorb-mail-header nil nil t)))
+    (let ((org-ids (mail-fetch-field gnorb-mail-header nil nil t))
+         (msg-id (mail-fetch-field "Message-ID"))
+         (to (mail-fetch-field "To"))
+         (subject (mail-fetch-field "Subject"))
+         (date (mail-fetch-field "Date"))
+         ;; if we can get a link, that's awesome
+         (link (or (and (mail-fetch-field "Gcc")
+                        (org-store-link))
+                   nil)))
+      ;; if we can't, then save some information so we can fake it
+      (setq gnorb-gnus-sending-message-info
+           `(:subject ,subject :msg-id ,msg-id
+                      :to ,to :link ,link
+                      :date ,date))
       (if org-ids
          (progn
            (require 'gnorb-org)
@@ -238,10 +263,67 @@ current message; multiple header values returned as a 
string."
            ;; if we're working from a draft or whatever, it might not
            ;; be there yet
            (add-to-list 'message-exit-actions
-                        'gnorb-org-restore-after-send t))
+                        'gnorb-org-restore-after-send))
        (setq gnorb-message-org-ids nil)))))
 
-(add-hook 'message-send-hook 'gnorb-gnus-check-org-header)
+(add-hook 'message-header-hook 'gnorb-gnus-check-outgoing-headers)
+
+(defun gnorb-gnus-outgoing-make-todo (arg)
+  "Call this function to turn the message currently being
+composed into an email todo action. If it's a new message, or a
+reply to a message that isn't referenced by any TODOs, a new TODO
+will be created. If it is referenced, you'll be prompted to
+trigger a state-change or a note on that TODO, via
+`gnorb-gnus-message-trigger-todo' (this is a lie, I haven't
+implemented this yet -- it will only make a new TODO).
+
+You can call this either in the message buffer, while you're
+composing the message, or after the message is sent: information
+is saved for the most recently-sent email.
+
+If a new todo is made, it needs a capture template: set
+`gnorb-gnus-new-todo-capture-key' to the string key for the
+appropriate capture template. If you're using a gnus-based
+archive method (ie you have `gnus-message-archive-group' set to
+something, and your outgoing messages have a \"Fcc\" header),
+then a real link will be made to the outgoing message, and all
+the gnus-type escapes will be available (see the Info
+manual (org) Template expansion section). If you don't, then the
+%:subject, %:to, and %:date escapes for the outgoing message will
+still be available -- nothing else will work."
+  (interactive "P")
+  (if (not (eq major-mode 'message-mode))
+      (gnorb-gnus-outgoing-make-todo-1)
+    (when (mail-fetch-field gnorb-mail-header)
+      ;; If we're already composing a response to a message that is
+      ;; "trackable" (ie, the In-Reply-To or References headers point
+      ;; to message-ids that are attached to active TODOs), sending
+      ;; the message should run `gnorb-gnus-message-trigger-todo'.
+
+      ;; also, people should be able to add extra TODO ids to the
+      ;; headers, to trigger multiple TODOs, if they're really nuts.
+      (user-error "This message is already being composed in response to a 
TODO."))
+    (add-to-list 'message-exit-actions
+                'gnorb-gnus-outgoing-make-todo-1 t)
+    (message "A TODO will be made from this message after it's sent")))
+
+(defun gnorb-gnus-outgoing-make-todo-1 ()
+  (unless gnorb-gnus-new-todo-capture-key
+    (error "No capture template key set, customize 
gnorb-gnus-new-todo-capture-key"))
+  (let* ((subject (plist-get gnorb-gnus-sending-message-info :subject))
+        (to (plist-get gnorb-gnus-sending-message-info :to))
+        (date (plist-get gnorb-gnus-sending-message-info :date))
+        (msg-id (plist-get gnorb-gnus-sending-message-info :msg-id))
+        (link (plist-get gnorb-gnus-sending-message-info :link))
+        ;; If we actually have a link, then make use of it.
+        ;; Otherwise, fake it.
+        (org-capture-link-is-already-stored
+         (or link t)))
+    (setq org-store-link-plist
+         `(:subject ,subject :to ,to :date ,date))
+    (org-capture nil gnorb-gnus-new-todo-capture-key)
+    (when msg-id
+      (org-entry-put (point) gnorb-org-msg-id-key msg-id))))
 
 ;;; If an incoming message should trigger state-change for a Org todo,
 ;;; call this function on it.
diff --git a/lisp/gnorb-org.el b/lisp/gnorb-org.el
index d75892a..8ff6687 100644
--- a/lisp/gnorb-org.el
+++ b/lisp/gnorb-org.el
@@ -38,6 +38,10 @@ relevant headings with these keywords. Set to nil to call
 org-todo regardless of TODO type."
   :group 'gnorb-org)
 
+(defcustom gnorb-org-msg-id-key "GNORB_MSG_ID"
+  "The name of the org property used to store the Message-IDs
+  from relevant messages.")
+
 (defun gnorb-org-contact-link (rec)
   "Prompt for a BBDB record and insert a link to that record at
 point.
@@ -70,7 +74,9 @@ might have been in the outgoing message's headers and call
     (set-window-configuration gnorb-org-window-conf))
   (dolist (id gnorb-message-org-ids)
     (gnorb-trigger-todo-action nil id))
+  ;; this is a little unnecessary, but it may save grief
   (setq gnorb-org-window-conf nil)
+  (setq gnorb-gnus-sending-message-info nil)
   (setq gnorb-message-org-ids nil))
 
 (defun gnorb-org-extract-mail-stuff ()



reply via email to

[Prev in Thread] Current Thread [Next in Thread]