From: John Kitchin <jkitchin@andrew.cmu.edu>
To: "Emacs-orgmode@gnu.org" <emacs-orgmode@gnu.org>
Subject: html-email in org-mode
Date: Sat, 29 Oct 2016 14:37:14 -0400 [thread overview]
Message-ID: <m2pomjvw6d.fsf@Johns-MacBook-Air.local> (raw)
[-- Attachment #1.1: Type: text/plain, Size: 8578 bytes --]
* DONE Sending html emails from org-mode with org-mime
On the org-mode mailing list there was some discussion about sending html mail using orgmode. The support for this in mu4e is deprecated. There is the org-mime library though, and it supports a lot of what is needed for this. I am going to send this post to the orgmode mailing list as an html mail to make sure it works. It should show as a text mail in mu4e, and an html mail in a browser.
As I played around with it though, I came across some limitations:
1. I found equations were not rendered as images in the html, and files (in links) were not attached out of the box. I fixed that [[id:14317E51-C65E-48DD-9B52-B94D6B458E8F][here]].
2. I found it useful to modify the org-mime commands to leave point in the To: field when composing emails from org-buffers.
3. For use with mu4e, I created a function to open a message in org-mu4e-compose-org-mode, and added a C-cC-c hook to allow me to send it easily [[id:D44F059D-180C-41C5-BA0A-873723E0DDFB][(here)]].
This post documents some work I did figuring out how to send html mails. After some testing, some of these should probably be patched in org-mime.
First, you need to require this library.
#+BEGIN_SRC emacs-lisp
(require 'org-mime)
#+END_SRC
You can send a whole org buffer in html like with this command: [[elisp:org-mime-org-buffer-htmlize]]. Not all of the internal links work for me (at least in gmail).
The default behavior leaves you at the end of the buffer, which is not too nice. We lightly modify the function here to leave in the To: field.
#+BEGIN_SRC emacs-lisp
(defun org-mime-org-buffer-htmlize ()
"Create an email buffer containing the current org-mode file
exported to html and encoded in both html and in org formats as
mime alternatives."
(interactive)
(org-mime-send-buffer 'html)
(message-goto-to))
#+END_SRC
** From an org-headline in an org-file
You can compose an email as an org-heading in any org-buffer, and send it as html. In an org-heading, you need to specify a MAIL_FMT property of html, e.g.:
#+BEGIN_EXAMPLE
:PROPERTIES:
:MAIL_FMT: html
:END:
#+END_EXAMPLE
Note the following properties can also be set to modify the composed email.
#+BEGIN_SRC emacs-lisp
(subject (or (funcall mp "MAIL_SUBJECT") (nth 4 (org-heading-components))))
(to (funcall mp "MAIL_TO"))
(cc (funcall mp "MAIL_CC"))
(bcc (funcall mp "MAIL_BCC"))
#+END_SRC
Then, send it with [[elisp:org-mime-subtree]]
Here I modify this function to leave me in the To: field.
#+BEGIN_SRC emacs-lisp
(defun org-mime-subtree ()
"Create an email buffer containing the current org-mode subtree
exported to a org format or to the format specified by the
MAIL_FMT property of the subtree."
(interactive)
(org-mime-send-subtree
(or (org-entry-get nil "MAIL_FMT" org-mime-use-property-inheritance) 'org))
(message-goto-to))
#+END_SRC
Here are some sample elements to see if they convert to html reasonably.
*** Markup
*bold*
_underlined_
/italics/
+strikethrough+
~code~
Subscripts: H_{2}O
Superscripts: H^{+}
An entity: To \infty and beyond
*** Equations
\(x^2\)
\[x^4\]
$e^x$
*** Tables
#+CAPTION: A table for you.
| x | y |
| 1 | 2 |
*** Lists
A nested list.
- one
- Subentry under one.
- two
A definition list:
- def1 :: first definition
A checklist:
- [ ] A checkbox
Here is a numbered list:
1. number 1
2. number 2
*** Code block
#+BEGIN_SRC python
import numpy as np
import matplotlib.pyplot as plt
t = np.linspace(0, 10)
x = np.cos(t) * np.exp(-t)
y = np.sin(t) * np.exp(-t)
plt.plot(x, y)
plt.savefig('spiral.png')
#+END_SRC
#+CAPTION: A spiral
[[file:./spiral.png]]
*** An image from somewhere other than this directory
#+CAPTION: A gold particle
[[file:./images/Au-icosahedron-3.png]]
*** Citations with org-ref
#+NAME: table-1
| a | b | c |
See Table [[ref:table-1]].
[[#Dominik201408][Dominik201408]]
* Bibliography
** -
:PROPERTIES:
:CUSTOM_ID: Dominik201408
:=TYPE=: book
:=KEY=: Dominik201408
:TITLE: {The Org Mode 8 Reference Manual - Organize your life with GNU
Emacs}
:AUTHOR: {Carsten Dominik}
:PUBLISHER: {Samurai Media Limited}
:YEAR: 2014
:MONTH: 8
:ISBN: 9789881327703
:URL: {http://amazon.com/o/ASIN/9881327709/}
:PRICE: {\$19.99}
:TOTALPAGES: 286
:TIMESTAMP: {2015.02.03}
:KEYWORDS: {org-mode}
:END:
** In a mail message
You might prefer to do this directly in an email. Here is how you can do it in mu4e. I use this command to open a message in org-mode. The mode switches if you are in the header, or in the body. If you always do this, you could use a hook instead on message-mode. I do not want default html so I do not do it.
#+BEGIN_SRC emacs-lisp
(defun mu4e-compose-org-mail ()
(interactive)
(mu4e-compose-new)
(org-mu4e-compose-org-mode))
#+END_SRC
For sending, we will use org-mime to htmlize it, and add a C-c C-c hook function to send it. This hook is a little tricky, we want to preserve C-c C-c behavior in org-mode, e.g. in code blocks, but send it if there is no other C-c C-c action that makes sense, so we add it to the end of the hook. Alternatively, you could bind a special key for it, or run the special command. Note the C-c C-c hook only works in the body of the email. From the header, a plain text message is sent.
#+BEGIN_SRC emacs-lisp
(defun htmlize-and-send ()
"When in an org-mu4e-compose-org-mode message, htmlize and send it."
(interactive)
(when (member 'org~mu4e-mime-switch-headers-or-body post-command-hook)
(org-mime-htmlize)
(message-send-and-exit)))
(add-hook 'org-ctrl-c-ctrl-c-hook 'htmlize-and-send t)
#+END_SRC
Here is a way to do this for non-mu4e users. It doesn't have the nice mode switching capability though, so you lose completion in emails, and header specific functions. You can switch back to message-mode to regain those.
#+BEGIN_SRC emacs-lisp
(defun compose-html-org ()
(interactive)
(compose-mail)
(message-goto-body)
(setq *compose-html-org* t)
(org-mode))
(defun org-htmlize-and-send ()
"When in an org-mu4e-compose-org-mode message, htmlize and send it."
(interactive)
(when *compose-html-org*
(setq *compose-html-org* nil)
(message-mode)
(org-mime-htmlize)
(message-send-and-exit)))
(add-hook 'org-ctrl-c-ctrl-c-hook 'org-htmlize-and-send t)
#+END_SRC
** Equations and file attachments do not seem to work out of the box
\(e^{i\pi} - 1 = 0\)
Out of the box, org-mime does not seem to attach file links to emails or make images for equations..
[[file:html-email.org]]
Here is an adaptation of org-mime-compose that does that for html messages.
#+BEGIN_SRC emacs-lisp
(defun org-mime-compose (body fmt file &optional to subject headers)
(require 'message)
(let ((bhook
(lambda (body fmt)
(let ((hook (intern (concat "org-mime-pre-"
(symbol-name fmt)
"-hook"))))
(if (> (eval `(length ,hook)) 0)
(with-temp-buffer
(insert body)
(goto-char (point-min))
(eval `(run-hooks ',hook))
(buffer-string))
body))))
(fmt (if (symbolp fmt) fmt (intern fmt)))
(files (org-element-map (org-element-parse-buffer) 'link
(lambda (link)
(when (string= (org-element-property :type link) "file")
(file-truename (org-element-property :path link)))))))
(compose-mail to subject headers nil)
(message-goto-body)
(cond
((eq fmt 'org)
(require 'ox-org)
(insert (org-export-string-as
(org-babel-trim (funcall bhook body 'org)) 'org t)))
((eq fmt 'ascii)
(require 'ox-ascii)
(insert (org-export-string-as
(concat "#+Title:\n" (funcall bhook body 'ascii)) 'ascii t)))
((or (eq fmt 'html) (eq fmt 'html-ascii))
(require 'ox-ascii)
(require 'ox-org)
(let* ((org-link-file-path-type 'absolute)
;; we probably don't want to export a huge style file
(org-export-htmlize-output-type 'inline-css)
(org-html-with-latex 'dvipng)
(html-and-images
(org-mime-replace-images
(org-export-string-as (funcall bhook body 'html) 'html t)))
(images (cdr html-and-images))
(html (org-mime-apply-html-hook (car html-and-images))))
(insert (org-mime-multipart
(org-export-string-as
(org-babel-trim
(funcall bhook body (if (eq fmt 'html) 'org 'ascii)))
(if (eq fmt 'html) 'org 'ascii) t)
html)
(mapconcat 'identity images "\n")))))
(mapc #'mml-attach-file files)))
#+END_SRC
** Summary
This makes it pretty nice to send rich-formatted html text to people.
[-- Attachment #1.2: Type: text/html, Size: 17906 bytes --]
[-- Attachment #2: latex11551eXz_71dd900d7f17a20875918a89a10eb146fccdd464.png --]
[-- Type: image/png, Size: 369 bytes --]
[-- Attachment #3: Au-icosahedron-3.png --]
[-- Type: image/png, Size: 30407 bytes --]
[-- Attachment #4: spiral.png --]
[-- Type: image/png, Size: 20044 bytes --]
[-- Attachment #5: latex11551EDn_c53faf490456a009247152f9614e65ac64628f0c.png --]
[-- Type: image/png, Size: 231 bytes --]
[-- Attachment #6: latex11551qua_398423584a3c15c2862f1218805c1a0068adb398.png --]
[-- Type: image/png, Size: 248 bytes --]
[-- Attachment #7: latex11551QaO_e39ccbca7983ec6a4ba1d3dcc1d807f16d4dfc22.png --]
[-- Type: image/png, Size: 255 bytes --]
[-- Attachment #8: spiral.png --]
[-- Type: image/png, Size: 20044 bytes --]
[-- Attachment #9: Au-icosahedron-3.png --]
[-- Type: image/png, Size: 30407 bytes --]
[-- Warning: decoded text below may be mangled, UTF-8 assumed --]
[-- Attachment #10: html-email.org --]
[-- Type: text/x-org, Size: 9034 bytes --]
* DONE Sending html emails from org-mode with org-mime
CLOSED: [2016-10-29 Sat 14:33]
:PROPERTIES:
:categories: email,orgmode
:ID: 181982DB-1843-4C3C-8E75-687CFF86A470
:date: 2016/10/29 14:33:16
:updated: 2016/10/29 14:33:16
:END:
On the org-mode mailing list there was some discussion about sending html mail using orgmode. The support for this in mu4e is deprecated. There is the org-mime library though, and it supports a lot of what is needed for this. I am going to send this post to the orgmode mailing list as an html mail to make sure it works. It should show as a text mail in mu4e, and an html mail in a browser.
As I played around with it though, I came across some limitations:
1. I found equations were not rendered as images in the html, and files (in links) were not attached out of the box. I fixed that [[id:14317E51-C65E-48DD-9B52-B94D6B458E8F][here]].
2. I found it useful to modify the org-mime commands to leave point in the To: field when composing emails from org-buffers.
3. For use with mu4e, I created a function to open a message in org-mu4e-compose-org-mode, and added a C-cC-c hook to allow me to send it easily [[id:D44F059D-180C-41C5-BA0A-873723E0DDFB][(here)]].
This post documents some work I did figuring out how to send html mails. After some testing, some of these should probably be patched in org-mime.
First, you need to require this library.
#+BEGIN_SRC emacs-lisp
(require 'org-mime)
#+END_SRC
#+RESULTS:
: org-mime
You can send a whole org buffer in html like with this command: elisp:org-mime-org-buffer-htmlize. Not all of the internal links work for me (at least in gmail).
The default behavior leaves you at the end of the buffer, which is not too nice. We lightly modify the function here to leave in the To: field.
#+BEGIN_SRC emacs-lisp
(defun org-mime-org-buffer-htmlize ()
"Create an email buffer containing the current org-mode file
exported to html and encoded in both html and in org formats as
mime alternatives."
(interactive)
(org-mime-send-buffer 'html)
(message-goto-to))
#+END_SRC
#+RESULTS:
: org-mime-org-buffer-htmlize
** From an org-headline in an org-file
:PROPERTIES:
:MAIL_FMT: html
:END:
You can compose an email as an org-heading in any org-buffer, and send it as html. In an org-heading, you need to specify a MAIL_FMT property of html, e.g.:
#+BEGIN_EXAMPLE
:PROPERTIES:
:MAIL_FMT: html
:END:
#+END_EXAMPLE
Note the following properties can also be set to modify the composed email.
#+BEGIN_SRC emacs-lisp
(subject (or (funcall mp "MAIL_SUBJECT") (nth 4 (org-heading-components))))
(to (funcall mp "MAIL_TO"))
(cc (funcall mp "MAIL_CC"))
(bcc (funcall mp "MAIL_BCC"))
#+END_SRC
Then, send it with elisp:org-mime-subtree
Here I modify this function to leave me in the To: field.
#+BEGIN_SRC emacs-lisp
(defun org-mime-subtree ()
"Create an email buffer containing the current org-mode subtree
exported to a org format or to the format specified by the
MAIL_FMT property of the subtree."
(interactive)
(org-mime-send-subtree
(or (org-entry-get nil "MAIL_FMT" org-mime-use-property-inheritance) 'org))
(message-goto-to))
#+END_SRC
#+RESULTS:
: org-mime-subtree
Here are some sample elements to see if they convert to html reasonably.
*** Markup
*bold*
_underlined_
/italics/
+strikethrough+
~code~
Subscripts: H_{2}O
Superscripts: H^{+}
An entity: To \infty and beyond
*** Equations
:PROPERTIES:
:MAIL_FMT: html
:END:
\(x^2\)
\[x^4\]
$e^x$
*** Tables
#+caption: A table for you.
| x | y |
| 1 | 2 |
*** Lists
A nested list.
- one
- Subentry under one.
- two
A definition list:
- def1 :: first definition
A checklist:
- [ ] A checkbox
Here is a numbered list:
1. number 1
2. number 2
*** Code block
#+BEGIN_SRC python :results output org drawer
import numpy as np
import matplotlib.pyplot as plt
t = np.linspace(0, 10)
x = np.cos(t) * np.exp(-t)
y = np.sin(t) * np.exp(-t)
plt.plot(x, y)
plt.savefig('spiral.png')
#+END_SRC
#+caption: A spiral
[[./spiral.png]]
*** An image from somewhere other than this directory
:PROPERTIES:
:MAIL_FMT: html
:END:
#+caption: A gold particle
[[./images/Au-icosahedron-3.png]]
*** Citations with org-ref
#+name: table-1
| a | b | c |
See Table ref:table-1.
cite:Dominik201408
bibliography:../../../Dropbox/bibliography/references.bib
** In a mail message
:PROPERTIES:
:ID: D44F059D-180C-41C5-BA0A-873723E0DDFB
:END:
You might prefer to do this directly in an email. Here is how you can do it in mu4e. I use this command to open a message in org-mode. The mode switches if you are in the header, or in the body. If you always do this, you could use a hook instead on message-mode. I do not want default html so I do not do it.
#+BEGIN_SRC emacs-lisp
(defun mu4e-compose-org-mail ()
(interactive)
(mu4e-compose-new)
(org-mu4e-compose-org-mode))
#+END_SRC
#+RESULTS:
: mu4e-compose-org-mail
For sending, we will use org-mime to htmlize it, and add a C-c C-c hook function to send it. This hook is a little tricky, we want to preserve C-c C-c behavior in org-mode, e.g. in code blocks, but send it if there is no other C-c C-c action that makes sense, so we add it to the end of the hook. Alternatively, you could bind a special key for it, or run the special command. Note the C-c C-c hook only works in the body of the email. From the header, a plain text message is sent.
#+BEGIN_SRC emacs-lisp
(defun htmlize-and-send ()
"When in an org-mu4e-compose-org-mode message, htmlize and send it."
(interactive)
(when (member 'org~mu4e-mime-switch-headers-or-body post-command-hook)
(org-mime-htmlize)
(message-send-and-exit)))
(add-hook 'org-ctrl-c-ctrl-c-hook 'htmlize-and-send t)
#+END_SRC
#+RESULTS:
| org-babel-hash-at-point | org-babel-execute-safely-maybe | htmlize-and-send |
Here is a way to do this for non-mu4e users. It doesn't have the nice mode switching capability though, so you lose completion in emails, and header specific functions. You can switch back to message-mode to regain those.
#+BEGIN_SRC emacs-lisp
(defun compose-html-org ()
(interactive)
(compose-mail)
(message-goto-body)
(setq *compose-html-org* t)
(org-mode))
(defun org-htmlize-and-send ()
"When in an org-mu4e-compose-org-mode message, htmlize and send it."
(interactive)
(when *compose-html-org*
(setq *compose-html-org* nil)
(message-mode)
(org-mime-htmlize)
(message-send-and-exit)))
(add-hook 'org-ctrl-c-ctrl-c-hook 'org-htmlize-and-send t)
#+END_SRC
#+RESULTS:
| org-babel-hash-at-point | org-babel-execute-safely-maybe | htmlize-and-send | org-htmlize-and-send |
** Equations and file attachments do not seem to work out of the box
:PROPERTIES:
:ID: 14317E51-C65E-48DD-9B52-B94D6B458E8F
:MAIL_FMT: html
:END:
\(e^{i\pi} - 1 = 0\)
Out of the box, org-mime does not seem to attach file links to emails or make images for equations..
file:html-email.org
Here is an adaptation of org-mime-compose that does that for html messages.
#+BEGIN_SRC emacs-lisp
(defun org-mime-compose (body fmt file &optional to subject headers)
(require 'message)
(let ((bhook
(lambda (body fmt)
(let ((hook (intern (concat "org-mime-pre-"
(symbol-name fmt)
"-hook"))))
(if (> (eval `(length ,hook)) 0)
(with-temp-buffer
(insert body)
(goto-char (point-min))
(eval `(run-hooks ',hook))
(buffer-string))
body))))
(fmt (if (symbolp fmt) fmt (intern fmt)))
(files (org-element-map (org-element-parse-buffer) 'link
(lambda (link)
(when (string= (org-element-property :type link) "file")
(file-truename (org-element-property :path link)))))))
(compose-mail to subject headers nil)
(message-goto-body)
(cond
((eq fmt 'org)
(require 'ox-org)
(insert (org-export-string-as
(org-babel-trim (funcall bhook body 'org)) 'org t)))
((eq fmt 'ascii)
(require 'ox-ascii)
(insert (org-export-string-as
(concat "#+Title:\n" (funcall bhook body 'ascii)) 'ascii t)))
((or (eq fmt 'html) (eq fmt 'html-ascii))
(require 'ox-ascii)
(require 'ox-org)
(let* ((org-link-file-path-type 'absolute)
;; we probably don't want to export a huge style file
(org-export-htmlize-output-type 'inline-css)
(org-html-with-latex 'dvipng)
(html-and-images
(org-mime-replace-images
(org-export-string-as (funcall bhook body 'html) 'html t)))
(images (cdr html-and-images))
(html (org-mime-apply-html-hook (car html-and-images))))
(insert (org-mime-multipart
(org-export-string-as
(org-babel-trim
(funcall bhook body (if (eq fmt 'html) 'org 'ascii)))
(if (eq fmt 'html) 'org 'ascii) t)
html)
(mapconcat 'identity images "\n")))))
(mapc #'mml-attach-file files)))
#+END_SRC
#+RESULTS:
: org-mime-compose
** Summary
This makes it pretty nice to send rich-formatted html text to people.
[-- Attachment #11: Type: text/plain, Size: 190 bytes --]
--
Professor John Kitchin
Doherty Hall A207F
Department of Chemical Engineering
Carnegie Mellon University
Pittsburgh, PA 15213
412-268-7803
@johnkitchin
http://kitchingroup.cheme.cmu.edu
next reply other threads:[~2016-10-29 18:37 UTC|newest]
Thread overview: 11+ messages / expand[flat|nested] mbox.gz Atom feed top
2016-10-29 18:37 John Kitchin [this message]
2016-10-30 14:34 ` html-email in org-mode Eric Brown
2016-10-30 15:09 ` Marcin Borkowski
2016-10-30 21:45 ` John Kitchin
2016-11-01 11:58 ` Eric Brown
2016-11-01 19:25 ` Alan Schmitt
2016-11-05 0:13 ` Matt Price
2016-11-07 2:55 ` John Kitchin
2016-11-07 15:37 ` Joseph Vidal-Rosset
-- strict thread matches above, loose matches on Subject: below --
2016-10-30 15:44 tbanelwebmin
2016-11-01 16:10 (no subject) John Kitchin
2016-11-06 0:27 ` html-email in org-mode Eric Brown
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
List information: https://www.orgmode.org/
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=m2pomjvw6d.fsf@Johns-MacBook-Air.local \
--to=jkitchin@andrew.cmu.edu \
--cc=emacs-orgmode@gnu.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
Code repositories for project(s) associated with this public inbox
https://git.savannah.gnu.org/cgit/emacs/org-mode.git
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for read-only IMAP folder(s) and NNTP newsgroup(s).