From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: Received: from localhost (localhost [127.0.0.1]) by olra.theworths.org (Postfix) with ESMTP id 56C90431FAF for ; Thu, 19 Jan 2012 20:30:45 -0800 (PST) X-Virus-Scanned: Debian amavisd-new at olra.theworths.org X-Spam-Flag: NO X-Spam-Score: -0.7 X-Spam-Level: X-Spam-Status: No, score=-0.7 tagged_above=-999 required=5 tests=[RCVD_IN_DNSWL_LOW=-0.7] autolearn=disabled Received: from olra.theworths.org ([127.0.0.1]) by localhost (olra.theworths.org [127.0.0.1]) (amavisd-new, port 10024) with ESMTP id nn4foO6Av-yu for ; Thu, 19 Jan 2012 20:30:44 -0800 (PST) Received: from dmz-mailsec-scanner-3.mit.edu (DMZ-MAILSEC-SCANNER-3.MIT.EDU [18.9.25.14]) by olra.theworths.org (Postfix) with ESMTP id 374FA431FAE for ; Thu, 19 Jan 2012 20:30:44 -0800 (PST) X-AuditID: 1209190e-b7f7c6d0000008c3-f4-4f18edf3a236 Received: from mailhub-auth-4.mit.edu ( [18.7.62.39]) by dmz-mailsec-scanner-3.mit.edu (Symantec Messaging Gateway) with SMTP id C0.BD.02243.3FDE81F4; Thu, 19 Jan 2012 23:30:43 -0500 (EST) Received: from outgoing.mit.edu (OUTGOING-AUTH.MIT.EDU [18.7.22.103]) by mailhub-auth-4.mit.edu (8.13.8/8.9.2) with ESMTP id q0K4UhRi019618; Thu, 19 Jan 2012 23:30:43 -0500 Received: from awakening.csail.mit.edu (awakening.csail.mit.edu [18.26.4.91]) (authenticated bits=0) (User authenticated as amdragon@ATHENA.MIT.EDU) by outgoing.mit.edu (8.13.6/8.12.4) with ESMTP id q0K4Ugj9013396 (version=TLSv1/SSLv3 cipher=AES256-SHA bits=256 verify=NOT); Thu, 19 Jan 2012 23:30:43 -0500 (EST) Received: from amthrax by awakening.csail.mit.edu with local (Exim 4.77) (envelope-from ) id 1Ro67L-0004gV-Vb; Thu, 19 Jan 2012 23:30:24 -0500 Date: Thu, 19 Jan 2012 23:30:23 -0500 From: Austin Clements To: Mark Walters Subject: Re: [PATCH v4 1/1] Make buttons for attachments allow viewing as well as saving Message-ID: <20120120043023.GW16740@mit.edu> References: <1327008185-21194-2-git-send-email-markwalters1009@gmail.com> MIME-Version: 1.0 Content-Type: text/plain; charset=us-ascii Content-Disposition: inline In-Reply-To: <1327008185-21194-2-git-send-email-markwalters1009@gmail.com> User-Agent: Mutt/1.5.21 (2010-09-15) X-Brightmail-Tracker: H4sIAAAAAAAAA+NgFvrPIsWRmVeSWpSXmKPExsUixG6nrvv5rYS/wdK3ahar5/JYXL85k9mB yWPnrLvsHs9W3WIOYIrisklJzcksSy3St0vgyviweyNzwTbbivsbF7A3MC427GLk5JAQMJFY 37ySEcIWk7hwbz1bFyMXh5DAPkaJ/zuPMUE4Gxgldn97zA5SJSRwkkli8/FgCHsJo8Tc2Ykg NouAqsT7UzNZQWw2AQ2JbfuXg00VEdCRuH1oAVgvs4C0xLffzUwgtrBAlMSBd3NYuhg5OHiB aj5uV4cYWSjxb+8isHJeAUGJkzOfsEC0aknc+PeSCaQcZMzyfxwgJqeAl0TPvGqQClEBFYkp J7exTWAUmoWkeRaS5lkIzQsYmVcxyqbkVunmJmbmFKcm6xYnJ+blpRbpGuvlZpbopaaUbmIE hTOnJN8Oxq8HlQ4xCnAwKvHwcrpK+AuxJpYVV+YeYpTkYFIS5VV+DRTiS8pPqcxILM6ILyrN SS0+xCjBwawkwuvwBCjHm5JYWZValA+TkuZgURLnVdN65yckkJ5YkpqdmlqQWgSTleHgUJLg dQXGrZBgUWp6akVaZk4JQpqJgxNkOA/Q8BKQGt7igsTc4sx0iPwpRkUpcV55kIQASCKjNA+u F5ZuXjGKA70izFsLUsUDTFVw3a+ABjMBDfZoEgMZXJKIkJJqYKxPqj2UoHX+9CSRSC2feu// jsIP23lNAtsFY80L5zqrMP3mSZ6bdPPeXW3b8ohUj8Dnm0N0Qk8XrNeavbMxRSOGk7eg0tVE QE9kPteRxS77/u44uE9ka8j/+abdyQu/luVabyp5FM5l/mQSx2vPGR33912WV941bbG0uoH7 jQi/C8m+jpdWKrEUZyQaajEXFScCAGM9q+0SAwAA Cc: notmuch@notmuchmail.org X-BeenThere: notmuch@notmuchmail.org X-Mailman-Version: 2.1.13 Precedence: list List-Id: "Use and development of the notmuch mail system." List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , X-List-Received-Date: Fri, 20 Jan 2012 04:30:45 -0000 One indentation nit and then this LGTM. Quoth Mark Walters on Jan 19 at 9:23 pm: > Define a keymap for attachment buttons to allow multiple actions. > Define 3 possible actions: > save attachment: exactly as currently, > view attachment: uses mailcap entry, > view attachment with user chosen program > > Keymap on a button is: s for save, v for view and o for view with > other program. Default (i.e. enter or mouse button) is save but this > is configurable in notmuch customize. > > One implementation detail: the view attachment function forces all > attachments to be "displayed" using mailcap even if emacs could > display them itself. Thus, for example, text/html appears in a browser > and text/plain asks whether to save (on a standard debian setup) > --- > emacs/notmuch-show.el | 116 ++++++++++++++++++++++++++++++++++++++---------- > 1 files changed, 92 insertions(+), 24 deletions(-) > > diff --git a/emacs/notmuch-show.el b/emacs/notmuch-show.el > index fc13462..1fcd72a 100644 > --- a/emacs/notmuch-show.el > +++ b/emacs/notmuch-show.el > @@ -112,6 +112,16 @@ indentation." > :type 'boolean > :group 'notmuch-show) > > +(defcustom notmuch-show-part-button-default-action 'notmuch-show-save-part > + "Default part header button action (on ENTER or mouse click)." > + :group 'notmuch-show > + :type '(choice (const :tag "Save part" > + notmuch-show-save-part) > + (const :tag "View part" > + notmuch-show-view-part) > + (const :tag "View interactively" > + notmuch-show-interactively-view-part))) > + > (defmacro with-current-notmuch-show-message (&rest body) > "Evaluate body with current buffer set to the text of current message" > `(save-excursion > @@ -283,10 +293,21 @@ message at DEPTH in the current thread." > (run-hooks 'notmuch-show-markup-headers-hook))))) > > (define-button-type 'notmuch-show-part-button-type > - 'action 'notmuch-show-part-button-action > + 'action 'notmuch-show-part-button-default > + 'keymap 'notmuch-show-part-button-map > 'follow-link t > 'face 'message-mml) > > +(defvar notmuch-show-part-button-map > + (let ((map (make-sparse-keymap))) > + (set-keymap-parent map button-map) > + (define-key map "s" 'notmuch-show-part-button-save) > + (define-key map "v" 'notmuch-show-part-button-view) > + (define-key map "o" 'notmuch-show-part-button-interactively-view) > + map) > + "Submap for button commands") > +(fset 'notmuch-show-part-button-map notmuch-show-part-button-map) > + > (defun notmuch-show-insert-part-header (nth content-type declared-type &optional name comment) > (let ((button)) > (setq button > @@ -301,29 +322,58 @@ message at DEPTH in the current thread." > " ]") > :type 'notmuch-show-part-button-type > :notmuch-part nth > - :notmuch-filename name)) > + :notmuch-filename name > + :notmuch-content-type content-type)) > (insert "\n") > ;; return button > button)) > > ;; Functions handling particular MIME parts. > > -(defun notmuch-show-save-part (message-id nth &optional filename) > - (let ((process-crypto notmuch-show-process-crypto)) > - (with-temp-buffer > - (setq notmuch-show-process-crypto process-crypto) > - ;; Always acquires the part via `notmuch part', even if it is > - ;; available in the JSON output. > - (insert (notmuch-show-get-bodypart-internal message-id nth)) > - (let ((file (read-file-name > - "Filename to save as: " > - (or mailcap-download-directory "~/") > - nil nil > - filename))) > - ;; Don't re-compress .gz & al. Arguably we should make > - ;; `file-name-handler-alist' nil, but that would chop > - ;; ange-ftp, which is reasonable to use here. > - (mm-write-region (point-min) (point-max) file nil nil nil 'no-conversion t))))) > +(defmacro notmuch-with-temp-part-buffer (message-id nth &rest body) > + (declare (indent 2)) > + (let ((process-crypto (make-symbol "process-crypto"))) > + `(let ((,process-crypto notmuch-show-process-crypto)) > + (with-temp-buffer > + (setq notmuch-show-process-crypto ,process-crypto) > + ;; Always acquires the part via `notmuch part', even if it is > + ;; available in the JSON output. > + (insert (notmuch-show-get-bodypart-internal ,message-id ,nth)) > + ,@body)))) > + > +(defun notmuch-show-save-part (message-id nth &optional filename content-type) > + (notmuch-with-temp-part-buffer message-id nth > + (let ((file (read-file-name > + "Filename to save as: " > + (or mailcap-download-directory "~/") > + nil nil > + filename))) > + ;; Don't re-compress .gz & al. Arguably we should make > + ;; `file-name-handler-alist' nil, but that would chop > + ;; ange-ftp, which is reasonable to use here. > + (mm-write-region (point-min) (point-max) file nil nil nil 'no-conversion t)))) > + > +(defun notmuch-show-view-part (message-id nth &optional filename content-type ) > + (notmuch-with-temp-part-buffer message-id nth > + ;; set mm-inlined-types to nil to force an external viewer > + (let ((handle (mm-make-handle (current-buffer) (list content-type))) > + (mm-inlined-types nil)) > + ;; We override mm-save-part as notmuch-show-save-part is better > + ;; since it offers the filename. We need to lexically bind > + ;; everything we need for notmuch-show-save-part to prevent > + ;; potential dynamic shadowing. > + (lexical-let ((message-id message-id) > + (nth nth) > + (filename filename) > + (content-type content-type)) > + (flet ((mm-save-part (&rest args) (notmuch-show-save-part This body of lexical-let should be indented like a normal let body. You might have to load cl to get this indentation rule. > + message-id nth filename content-type))) > + (mm-display-part handle)))))) > + > +(defun notmuch-show-interactively-view-part (message-id nth &optional filename content-type) > + (notmuch-with-temp-part-buffer message-id nth > + (let ((handle (mm-make-handle (current-buffer) (list content-type)))) > + (mm-interactively-view-part handle)))) > > (defun notmuch-show-mm-display-part-inline (msg part nth content-type) > "Use the mm-decode/mm-view functions to display a part in the > @@ -1504,12 +1554,30 @@ buffer." > > ;; Commands typically bound to buttons. > > -(defun notmuch-show-part-button-action (button) > - (let ((nth (button-get button :notmuch-part))) > - (if nth > - (notmuch-show-save-part (notmuch-show-get-message-id) nth > - (button-get button :notmuch-filename)) > - (message "Not a valid part (is it a fake part?).")))) > +(defun notmuch-show-part-button-default (&optional button) > + (interactive) > + (notmuch-show-part-button-internal button notmuch-show-part-button-default-action)) > + > +(defun notmuch-show-part-button-save (&optional button) > + (interactive) > + (notmuch-show-part-button-internal button #'notmuch-show-save-part)) > + > +(defun notmuch-show-part-button-view (&optional button) > + (interactive) > + (notmuch-show-part-button-internal button #'notmuch-show-view-part)) > + > +(defun notmuch-show-part-button-interactively-view (&optional button) > + (interactive) > + (notmuch-show-part-button-internal button #'notmuch-show-interactively-view-part)) > + > +(defun notmuch-show-part-button-internal (button handler) > + (let ((button (or button (button-at (point))))) > + (if button > + (let ((nth (button-get button :notmuch-part))) > + (if nth > + (funcall handler (notmuch-show-get-message-id) nth > + (button-get button :notmuch-filename) > + (button-get button :notmuch-content-type))))))) > > ;; >