-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathiexpand.el
290 lines (252 loc) · 10.4 KB
/
iexpand.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
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
;;; iexpand.el --- Expand commands at point --- -*- lexical-binding: t; -*-
;;
;; Copyright (C) 2020 Samuel Barreto
;; License: GPLv3
;;
;; Author: Samuel Barreto <[email protected]>
;; Maintainer: Samuel barreto <[email protected]>
;; Keywords: expansion, convenience
;; Version: 0.1.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:
;;
;; Calls interactive functions from a typed symbol. This is cool
;; because one can just type a symbol, and call `iexpand`, which
;; will trigger the command if there is one associated with the symbol
;; at point; the typed symbol will be deleted, restoring buffer state.
;;
;; Users can define expansion tables in the spirit of `abbrev`
;; tables — that is, in a per-mode fashion, respecting the hierarchy
;; of modes.
;;
;;; Usage:
;;
;; ```emacs-lisp
;; (require 'iexpand)
;; (setq iexpand-default-key "RET") ; default is RET
;; (iexpand-global-mode t)
;; (iexpand-define 'emacs-lisp-mode "eb" #'eval-buffer)
;; (iexpand-define 'prog-mode "compile" #'compile)
;; ```
;;
;; Now in an emacs-lisp buffer, typing `eb` and calling
;; `iexpand` (by default bound to `RET`) will evaluate the buffer.
;; In that same buffer, as `emacs-lisp-mode` inherits from
;; `prog-mode`, typing `compile` and calling `iexpand` will
;; prompt for a compilation command (see `C-h C-f compile`).
;;
;; One can also define multiple expansions in one run:
;;
;; ```emacs-lisp
;; (iexpand-define-table 'text-mode
;; "xs" #'save-buffer
;; "ir" #'indent-region
;; "indent-region" #'indent-region
;; "indent" #'indent-region)
;;
;; ;; define global expansions by adding to the fundamental-mode table
;; (iexpand-define-table 'fundamental-mode
;; "bb" #'switch-to-buffer
;; "file" #'find-file
;; "time" (lambda () (interactive) (message (format-time-string "%FT%T")))
;; "timestamp" (lambda () (interactive) (insert (format-time-string "%FT%T"))))
;;
;; ```
;;
;; See it in action:
;;
;; ![screencast](doc/screencast.gif)
;;
;; > (For those who care, theme is a combination of [modus
;; operandi](https://gitlab.com/protesilaos/modus-themes)
;; and [elegance.el](https://github.com/rougier/elegant-emacs); font
;; is *Roboto mono*.)
;;
;;; Installation
;;
;; ```bash
;; mkdir -p ~/.emacs.d/lib; git clone https://github.com/sam217pa/emacs-iexpand ~/.emacs.d/lib/iexpand
;; ```
;; then in your `init.el` file:
;; ```emacs-lisp
;; (add-to-list 'load-path (expand-file-name "lib/iexpand" user-emacs-directory))
;; (require 'iexpand)
;; ```
;;
;;; Why should you care?
;;
;; As has been remarked to me on [reddit](https://www.reddit.com/r/emacs/comments/hbbqnc/new_package_iexpandel_calling_commands_by/fv8ojfe?utm_source=share&utm_medium=web2x),
;; yasnippet or other packages implements something similar to what iexpand does.
;; However, triggering arbitrary yet simple emacs-lisp commands from yasnippet requires you to 1) write a snippet with the propper (simple) syntax ([yasnippet documentation](https://joaotavora.github.io/yasnippet/snippet-development.html#orgcde188c)), 2) save it to an appropriate place, 3) have yasnippet properly load the snippet (which it does automatically normally).
;; It does work, but I do think that it is not what yasnippet was primarily intended to do, which is to expand plain text, with some assistance from emacs-lisp when need be.
;;
;; Iexpand take it the other way around: it is primarily intended to evaluate emacs-lisp commands, which incidentally allows it to expand some simple text snippets.
;; But I would keep writing snippets for anything more complex than what is displayed in the screencast (the `timestamp` snippet).
;;
;; Another interesting aspect to me is that you can give plain text orders to emacs, say `save`, press return: it saves the buffer.
;; ```emacs-lisp
;; (iexpand-define 'prog-mode "stage" #'magit-stage-file)
;; ```
;; type `stage`, press return, the file is staged.
;; For some reason this “workflow” suits me well, maybe it'll suit you too.
;;
;;; Code:
(eval-when-compile
(require 'subr-x)
(require 'cl-lib))
(require 'thingatpt)
;; CUSTOM ------------------------------------------------------------
(defcustom iexpand-default-key "RET"
"Default keybinding for calling `iexpand'"
:group 'iexpand
:type 'string)
;; CORE --------------------------------------------------------------
(defun iexpand--group (source n)
"Divide SOURCE list in N groups and stack together the last
elements. (from P. Graham's On Lisp.)"
(if (zerop n) (error "Zero length"))
(cl-labels ((rec (source acc)
(let ((rest (nthcdr n source)))
(if (consp rest)
(rec rest (cons (cl-subseq source 0 n) acc))
(nreverse (cons source acc))))))
(if source (rec source nil) nil)))
(defun iexpand--mode-name (mode)
"Return the name of the iexpand-table associated with MODE."
(intern (format "iexpand--%s-table"
(string-trim (symbol-name mode) nil "-mode"))))
(defun iexpand--get-table (mode)
"Get the iexpand-table associated with MODE."
(get (iexpand--mode-name mode) 'table))
(defun iexpand--make-table (mode)
"Create an iexpand-table for MODE if none exists."
(or (iexpand--get-table mode)
(put (iexpand--mode-name mode) 'table (make-hash-table :test #'equal))))
(defmacro iexpand-define-table (mode &rest body)
"Wrapper around `iexpand-define' for defining multiple KEY-COMMAND expansions.
BODY consists of KEY-COMMAND pairs.
Example:
(iexpand-define-table 'emacs-lisp-mode
\"xs\" #'save-buffer
\"hw\" (lambda (interactive) (message \"hello world\")))"
(declare (indent 1))
`(progn
(iexpand--make-table ,mode)
,@(mapcar
(lambda (x) `(iexpand-define ,mode ,(car x) ,(cadr x)))
(iexpand--group body 2))))
(defun iexpand-define (mode key command)
"Define an expansion for COMMAND associated with KEY for MODE.
Calling `iexpand' when point is after KEY in major-mode
MODE triggers calling COMMAND interactively."
(let ((iexpand-table (iexpand--make-table mode)))
(puthash key command iexpand-table)))
(defun iexpand--get-cmd (key)
"Return command associated with KEY in iexpand tables up the
hierarchy of modes from major-mode to fundamental-mode.
Return nil if no key was found.
Major-mode table is searched first, fundamental last."
(cl-labels ((cmd (table) (and (hash-table-p table) (gethash key table)))
(f (mode)
(or (cmd (iexpand--get-table mode))
(when-let ((parent-mode (get mode 'derived-mode-parent)))
(f parent-mode)))))
(or (f major-mode)
(f 'fundamental-mode))))
(defun iexpand--symbol ()
"Call command associated with symbol at point."
(when-let ((be (bounds-of-thing-at-point 'symbol)))
(when-let ((cmd (iexpand--get-cmd (buffer-substring-no-properties (car be) (cdr be)))))
(unwind-protect
(progn
(delete-region (car be) (cdr be))
(undo-boundary)
t)
(call-interactively cmd)))))
(defun try-iexpand (old)
(unless old (iexpand--symbol)))
(defvar-local iexpand--fallback-cmd nil
"Command to call when no expansion was found")
(defun iexpand--get-expanding-key ()
(car (where-is-internal 'iexpand iexpand-mode-map)))
(defun iexpand--set-fallback-behaviour ()
(setq-local iexpand--fallback-cmd
(lookup-key (current-global-map) (iexpand--get-expanding-key))))
(defun iexpand (&optional arg)
"Call command associated with symbol at point."
(interactive "p")
(or (iexpand--symbol)
(call-interactively iexpand--fallback-cmd arg)))
;; DOCUMENTATION -----------------------------------------------------
(defun iexpand--get-derived-modes ()
"Returns the parents of major mode up to fundamental mode."
(cl-labels ((f (mode)
(if-let ((parent-mode (get mode 'derived-mode-parent)))
(cons mode (f parent-mode))
(cons mode (cons 'fundamental-mode nil)))))
(f major-mode)))
(defun iexpand-describe ()
"Describes the expansions associated with current `major-mode'."
(interactive)
(with-help-window (help-buffer)
(princ "[Iexpand Help]
Commands associated with current major-mode in iexpand:
")
(mapc
(lambda (mode) (when-let ((h (iexpand--get-table mode)))
(princ
(concat (format "\n%s\n" (symbol-name mode))
(make-string (length (symbol-name mode)) ?-)
"\n"))
(maphash (lambda (k v) (princ (format "%10s\t%s\n" k v))) h)))
(iexpand--get-derived-modes))))
;; KEYBINDINGS -------------------------------------------------------
(defvar iexpand-mode-map (make-sparse-keymap))
(defun iexpand--define-key (key)
"Correctly bind KEY to `iexpand', erasing previous bindings
made to it."
;; reset for erasing previous definition
(setq iexpand-mode-map (make-sparse-keymap))
(define-key iexpand-mode-map (kbd key) #'iexpand))
;;;###autoload
(define-minor-mode iexpand-minor-mode
"A simple minor mode that leverages typed text to call
interactive commands."
nil "iexp"
:keymap iexpand-mode-map
(when iexpand-minor-mode
(iexpand--set-fallback-behaviour)
(iexpand--define-key iexpand-default-key)))
(defcustom iexpand-except-modes '(help-mode minibuffer-inactive-mode calc-mode dired-mode magit-mode)
"Modes in which `iexpand' should have no effect."
:type (list 'string)
:group 'iexpand)
(defun iexpand--check-current-mode ()
(or (cl-loop for x in iexpand-except-modes
if (provided-mode-derived-p major-mode x)
collect x)
buffer-read-only))
;;;###autoload
(defun turn-on-iexpand-minor-mode ()
"Simple wrapper around `iexpand-minor-mode'"
(interactive)
(unless (iexpand--check-current-mode)
(iexpand-minor-mode t)))
;;;###autoload
(define-globalized-minor-mode iexpand-global-mode iexpand-minor-mode
turn-on-iexpand-minor-mode)
(provide 'iexpand)
;;; iexpand.el ends here