forked from legoscia/messages-are-flowing
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmessages-are-flowing.el
89 lines (76 loc) · 4.01 KB
/
messages-are-flowing.el
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
;;; messages-are-flowing.el --- visible indication when composing "flowed" emails -*- lexical-binding: t; -*-
;; Copyright (C) 2017 Magnus Henoch
;; Author: Magnus Henoch <[email protected]>
;; Keywords: mail
;; Version: 0.1
;; 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 <http://www.gnu.org/licenses/>.
;;; Commentary:
;; If you're writing emails to people who might not view them on a
;; display with the same width as yours, you probably want to send the
;; messages as "flowed" (as per RFC 2646) in order to let the
;; recipient's device disregard the line breaks in your message and
;; rewrap the text for readability. In `message-mode', you can do
;; that by turning on the `use-hard-newlines' minor mode.
;;
;; However, you probably want some of your newlines to stay put, for
;; paragraph breaks, and for content where you really do want to break
;; the lines yourself. You can do that with `use-hard-newlines', but
;; it doesn't show you where it's going to put "hard" newlines and
;; where it's going to put "soft" ones.
;;
;; That's where `messages-are-flowing' comes in. It marks all "hard"
;; newlines with a `⏎' symbol, so that you can have an idea about what
;; parts of your message might be reflowed when the recipient reads it.
;;
;; To activate `messages-are-flowing', add the following to your .emacs:
;;
;; (with-eval-after-load "message"
;; (add-hook 'message-mode-hook 'messages-are-flowing-use-and-mark-hard-newlines))
;;; Code:
(defcustom messages-are-flowing-newline-marker "⏎"
"String used to visualise hard newlines."
:type 'string
:group 'message-interface)
(defcustom messages-are-flowing-guess-hard-newlines nil
"Whether to guess automatically which newlines should be marked as hard when running MESSAGES-ARE-FLOWING-USE-AND-MARK-HARD-NEWLINES. GUESS means to do so when in a buffer with newlines but with no hard newlines. NIL means always ask what to do, ALWAYS means to guess even when some hard newlines already exist in the buffer. NEVER means never to guess."
:type 'symbol
:options (list 'always 'guess 'never nil)
:group 'message-interface)
;;;###autoload
(defun messages-are-flowing-use-and-mark-hard-newlines ()
"Turn on `use-hard-newlines', and make hard newlines visible.
The main use of this is to send \"flowed\" email messages, where
line breaks within paragraphs are adjusted by the recipient's
device, such that messages remain readable on narrow displays."
(interactive)
(use-hard-newlines nil messages-are-flowing-guess-hard-newlines)
(messages-are-flowing--mark-hard-newlines (point-min) (point-max))
(add-hook 'after-change-functions 'messages-are-flowing--mark-hard-newlines nil t))
(defun messages-are-flowing--mark-hard-newlines (beg end &rest _ignore)
"Visibly mark hard newlines between BEG and END.
For each hard newline, add a display property that makes it visible.
For each soft newline, remove any display property."
;; Uncomment for debugging:
;;(interactive (list (point-min) (point-max)))
(save-excursion
(goto-char beg)
(while (search-forward "\n" end t)
(let ((pos (1- (point))))
(if (get-text-property pos 'hard)
;; Use `copy-sequence', because display property values must not be `eq'!
(add-text-properties
pos (1+ pos)
(list 'display (copy-sequence
(concat messages-are-flowing-newline-marker "\n"))))
(remove-text-properties pos (1+ pos) '(display nil)))))))
(provide 'messages-are-flowing)
;;; messages-are-flowing.el ends here