forked from bmag/emacs-purpose
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathwindow-purpose-configuration.el
544 lines (465 loc) · 23.5 KB
/
window-purpose-configuration.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
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
;;; window-purpose-configuration.el --- Configuration handling for Purpose -*- lexical-binding: t -*-
;; Copyright (C) 2015-2021 Bar Magal & contributors
;; Author: Bar Magal
;; Package: purpose
;; This file is not part of GNU Emacs.
;; 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:
;; This file contains the "purpose configuration". The "purpose
;; configuration" is a set of variables that define what is the purpose
;; of each buffer.
;; The configuration is built from 3 layers: the user's config,
;; extensions' config (also called "extended config") and the default
;; config.
;; Each layer of configuration has 3 parameters for determining a
;; buffer's purpose: mode, name and regexp.
;; mode: matches a buffer's major mode
;; name: matches a buffer's name exactly
;; regexp: matches a buffer's name
;;
;; Each layer has 2 sets of variables: non-compiled variables, which are
;; easy to modify, and compiled variables which are used internally when
;; getting a buffer's purpose.
;;
;; Extensions that use Purpose and need to define a configuration,
;; should do so by using `purpose-conf' objects and the functions
;; `purpose-set-extension-configuration' and
;; `purpose-del-extension-configuration'.
;;
;; Users that want to set their own configuration, should do so by
;; customizing `purpose-user-mode-purposes',
;; `purpose-user-name-purposes' and `purpose-user-regexp-purposes'. If a
;; user changes any of these variables outside of customize, the user
;; should call `purpose-compile-user-configuration' for the changes to
;; take effect.
;; It is possible to use or ignore the default configuration by
;; customizing `purpose-use-default-configuration'.
;;; Code:
(require 'cl-lib)
(require 'eieio)
(require 'window-purpose-utils)
;;; Types
(defmacro define-purpose-list-checker (name entry-pred)
"Create a function named NAME to check the content of a list.
The generated function receives parameter OBJ, and checks that it is a
list and each entry in it satisifies ENTRY-PRED."
(declare (indent defun) (debug (&define name function-form)))
`(defun ,name (obj)
,(format "Check that OBJ is a list, and each entry in it satisifies %s." entry-pred)
(and (listp obj)
(cl-loop for entry in obj
always (funcall ,entry-pred entry)))))
(defun purpose-non-nil-symbol-p (obj)
"Check that OBJ is a symbol and not nil."
(and (symbolp obj) obj))
(defun purpose-mode-alist-entry-p (obj)
"Check that OBJ is a pair of mode and purpose.
OBJ should be a cons cell, whose car and cdr are both
`purpose-non-nil-symbol-p'."
(and (consp obj)
(purpose-non-nil-symbol-p (car obj))
(purpose-non-nil-symbol-p (cdr obj))))
(defun purpose-name-alist-entry-p (obj)
"Check that OBJ is a pair of name and purpose.
OBJ should be a cons cell, whose car is a string and cdr is a
`purpose-non-nil-symbol-p'."
(and (consp obj)
(stringp (car obj))
(purpose-non-nil-symbol-p (cdr obj))))
(defalias 'purpose-regexp-alist-entry-p #'purpose-name-alist-entry-p
"Check that OBJ is a pair of regexp and purpose.
OBJ should be a cons cell, whose car is a string and cdr is a
`purpose-non-nil-symbol-p'. Strictly speaking,
`purpose-regexp-alist-entry-p' doesn't actually check that the car is a
valid regexp.")
(define-purpose-list-checker purpose-mode-alist-p
#'purpose-mode-alist-entry-p)
(define-purpose-list-checker purpose-name-alist-p
#'purpose-name-alist-entry-p)
(define-purpose-list-checker purpose-regexp-alist-p
#'purpose-regexp-alist-entry-p)
;;;###autoload
(defclass purpose-conf ()
((mode-purposes :initarg :mode-purposes
:initform '()
:type (satisfies purpose-mode-alist-p))
(name-purposes :initarg :name-purposes
:initform '()
:type (satisfies purpose-name-alist-p))
(regexp-purposes :initarg :regexp-purposes
:initform '()
:type (satisfies purpose-regexp-alist-p))))
;;; Variables
(defcustom purpose-use-default-configuration t
"Determine if the default configuration should be used.
If this is nil, the default configuration is ignored when getting the
purpose of a buffer. The user configuration and extended configuration
are used anyway."
:group 'purpose
:type 'boolean
:package-version '(window-purpose . "1.2"))
(defcustom purpose-user-mode-purposes nil
"User configured alist mapping of modes to purposes.
The alist should match `purpose-mode-alist-p'.
If you set this variable in elisp-code, you should call the function
`purpose-compile-user-configuration' immediately afterwards."
:group 'purpose
:type '(alist :key-type (symbol :tag "major mode")
:value-type (symbol :tag "purpose"))
:set #'(lambda (symbol value)
(prog1 (set-default symbol value)
(purpose-compile-user-configuration)))
:initialize 'custom-initialize-default
:package-version '(window-purpose . "1.2"))
(defcustom purpose-user-name-purposes nil
"User configured alist mapping of names to purposes.
The alist should match `purpose-name-alist-p'.
If you set this variable in elisp-code, you should call the function
`purpose-compile-user-configuration' immediately afterwards."
:group 'purpose
:type '(alist :key-type (string :tag "name")
:value-type (symbol :tag "purpose"))
:set #'(lambda (symbol value)
(prog1 (set-default symbol value)
(purpose-compile-user-configuration)))
:initialize 'custom-initialize-default
:package-version '(window-purpose . "1.2"))
(defcustom purpose-user-regexp-purposes nil
"User configured alist mapping of regexps to purposes.
The alist should match `purpose-regexp-alist-p'.
If you set this variable in elisp-code, you should call the function
`purpose-compile-user-configuration' immediately afterwards."
:group 'purpose
:type '(alist :key-type (string :tag "regexp")
:value-type (symbol :tag "purpose"))
:set #'(lambda (symbol value)
(prog1 (set-default symbol value)
(purpose-compile-user-configuration)))
:initialize 'custom-initialize-default
:package-version '(window-purpose . "1.2"))
(defvar purpose-extended-configuration nil
"A plist containing `purpose-conf' objects.
An example of `purpose-extended-configuration':
(list :python (purpose-conf
:mode-purposes '((python-mode . python)
(python-inferior-mode . interpreter)))
:popups (purpose-conf
:mode-purposes '((help-mode . right)
(occur-mode . bottom)
(grep-mode . bottom))))")
;;; Compiled variables
(defvar purpose--user-mode-purposes (make-hash-table)
"The compiled user mapping of modes to purposes.
The contents of this variable are generated by
`purpose-compile-user-configuration'.")
(defvar purpose--user-name-purposes (make-hash-table :test #'equal)
"The compiled user mapping of names to purposes.
The contents of this variable are generated by
`purpose-compile-user-configuration'.")
(defvar purpose--user-regexp-purposes (make-hash-table :test #'equal)
"The compiled user mapping of regexps to purposes.
The contents of this variable are generated by
`purpose-compile-user-configuration'.")
(defvar purpose--extended-mode-purposes (make-hash-table)
"The combined mapping of modes to purposes, of all extensions.
The contents of this variable are generated by
`purpose-compile-extended-configuration'.")
(defvar purpose--extended-name-purposes (make-hash-table :test #'equal)
"The combined mapping of names to purposes, of all extensions.
The contents of this variable are generated by
`purpose-compile-extended-configuration'.")
(defvar purpose--extended-regexp-purposes (make-hash-table :test #'equal)
"The combined mapping of regexps to purposes, of all extensions.
The contents of this variable are generated by
`purpose-compile-extended-configuration'.")
(defvar purpose--default-mode-purposes (make-hash-table)
"The default mapping of modes to purposes.
The contents of this variable are generated by
`purpose-compile-default-configuration'.")
(defvar purpose--default-name-purposes (make-hash-table :test #'equal)
"The default mapping of names to purposes.
The contents of this variable are generated by
`purpose-compile-default-configuration'.")
(defvar purpose--default-regexp-purposes (make-hash-table :test #'equal)
"The default mapping of regexps to purposes.
The contents of this variable are generated by
`purpose-compile-default-configuration'.")
;;; Configuration compiler functions
(defun purpose--fill-hash (table alist &optional dont-clear)
"Fill hash table TABLE with ALIST's entries.
TABLE is cleared before filling it, unless DONT-CLEAR is non-nil."
(unless dont-clear
(clrhash table))
(mapc #'(lambda (entry)
(puthash (car entry) (cdr entry) table))
alist))
(defun purpose--set-and-compile-configuration (symbol value)
"Set SYMBOL's value to VALUE and recompile user configuration.
Recompilation is done by calling `purpose-compile-user-configuration'."
(prog1
(set-default symbol value)
(purpose-compile-user-configuration)))
(defun purpose-compile-user-configuration ()
"Compile the purpose configuration of the user.
Fill `purpose--user-mode-purposes', `purpose--user-name-purposes' and
`purpose--user-regexp-purposes' according to
`purpose-user-mode-purposes', `purpose-user-name-purposes' and
`purpose-user-regexp-purposes'."
(purpose--fill-hash purpose--user-mode-purposes purpose-user-mode-purposes)
(purpose--fill-hash purpose--user-name-purposes purpose-user-name-purposes)
(purpose--fill-hash purpose--user-regexp-purposes
purpose-user-regexp-purposes))
(defun purpose-compile-extended-configuration ()
"Compile the purpose configuration of extensions.
Fill `purpose--extended-mode-purposes',
`purpose--extended-name-purposes' and
`purpose--extended-regexp-purposes' according to
`purpose-extended-configuration'."
;; clear compiled purposes
(purpose--fill-hash purpose--extended-mode-purposes nil)
(purpose--fill-hash purpose--extended-name-purposes nil)
(purpose--fill-hash purpose--extended-regexp-purposes nil)
;; populate compiled purposes
(mapc #'(lambda (extension-config)
(purpose--fill-hash purpose--extended-mode-purposes
(slot-value extension-config :mode-purposes)
t)
(purpose--fill-hash purpose--extended-name-purposes
(slot-value extension-config :name-purposes)
t)
(purpose--fill-hash purpose--extended-regexp-purposes
(slot-value extension-config :regexp-purposes)
t))
(delq nil (purpose-plist-values purpose-extended-configuration))))
(defun purpose-compile-default-configuration ()
"Compile the default purpose configuraion."
(purpose--fill-hash purpose--default-mode-purposes
'((prog-mode . edit)
(text-mode . edit)
;; in Emacs 24.5-, `css-mode' doesn't derive from `prog-mode'
(css-mode . edit)
(comint-mode . terminal)
(eshell-mode . terminal)
(term-mode . terminal)
(dired-mode . dired)
(ibuffer-mode . buffers)
(Buffer-menu-mode . buffers)
(occur-mode . search)
(grep-mode . search)
(compilation-mode . search)
(image-mode . image)
(package-menu-mode . package)))
(purpose--fill-hash purpose--default-name-purposes
'((".gitignore" . edit)
(".hgignore" . edit)
;; the `shell' command displays its buffer before
;; setting its major-mode, so we must detect it by name
("*shell*" . terminal)))
(purpose--fill-hash purpose--default-regexp-purposes
'(("^ \\*Minibuf-[0-9]*\\*$" . minibuf))))
;;; convenient API functions
(cl-defmethod purpose-conf-add-purposes ((config purpose-conf) modes names regexps)
"Add purposes to a `purpose-conf' object.
MODES, NAMES and REGEXPS must be valid configuration alists as described in
`purpose-mode-alist-p', `purpose-name-alist-p' and `purpose-regexp-alist-p'."
(setf (slot-value config :mode-purposes)
(append modes (slot-value config :mode-purposes)))
(setf (slot-value config :name-purposes)
(append names (slot-value config :name-purposes)))
(setf (slot-value config :regexp-purposes)
(append regexps (slot-value config :regexp-purposes))))
(cl-defmethod purpose-conf-remove-purposes ((config purpose-conf) modes names regexps)
"Remove purposes from a `purpose-conf' object.
MODES must be a list of major modes.
NAMES must be a list names.
REGEXPS must be a list regexps."
;; let-bind before setq-ing, so we don't apply partial changes if one
;; of MODES, NAMES or REGEXPS is malformed
(let ((new-modes (cl-set-difference (slot-value config :mode-purposes) modes
:test (lambda (entry mode)
(eql (car entry) mode))))
(new-names (cl-set-difference (slot-value config :name-purposes) names
:test (lambda (entry name)
(string= (car entry) name))))
(new-regexps (cl-set-difference (slot-value config :regexp-purposes) regexps
:test (lambda (entry regexp)
(string= (car entry) regexp)))))
(setf (slot-value config :mode-purposes) new-modes)
(setf (slot-value config :name-purposes) new-names)
(setf (slot-value config :regexp-purposes) new-regexps)))
;;;###autoload
(defun purpose-set-extension-configuration (ext-keyword config)
"Set an extension's entry in `purpose-extended-configuration'.
EXT-KEYWORD should be a keyword used to identify the extension.
CONFIG is a `purpose-conf' object containing the extension's purpose
configuration.
Example:
(purpose-set-extension-configuration
:python
(purpose-conf \"py\"
:mode-purposes
'((python-mode . python)
(inferior-python-mode . interpreter))))
This function calls `purpose-compile-extended-configuration' when its
done."
(unless (keywordp ext-keyword)
(signal 'wrong-type-argument `(keywordp ,ext-keyword)))
(setq purpose-extended-configuration
(plist-put purpose-extended-configuration ext-keyword config))
(purpose-compile-extended-configuration))
(defun purpose-get-extension-configuration (ext-keyword)
"Get an extension's entry in `purpose-extended-configuration'.
EXT-KEYWORD is the same as in `purpose-set-extension-configuration'."
(unless (keywordp ext-keyword)
(signal 'wrong-type-argument `(keywordp ,ext-keyword)))
(plist-get purpose-extended-configuration ext-keyword))
(defun purpose-del-extension-configuration (ext-keyword)
"Delete an extension's entry in `purpose-extended-configuration'.
EXT-KEYWORD is the same as in `purpose-set-extension-configuration'.
Deletion is actually done by setting the extension's entry to nil.
This function calls `purpose-compile-extended-configuration' when its
done."
(purpose-set-extension-configuration ext-keyword nil))
(cl-defun purpose-add-extension-purposes (ext-keyword &key modes names regexps)
"Add purposes to an extension's purpose configuration.
EXT-KEYWORD is the same as in `purpose-set-extension-configuration'. MODES,
NAMES and REGEXPS must be valid configuration alists as described in
`purpose-mode-alist-p', `purpose-name-alist-p' and `purpose-regexp-alist-p'.
This function calls `purpose-compile-extended-configuration'.
Example:
(purpose-add-extension-purposes :python
:regexps '((\"\\.hy$\" . python)))"
(let ((config (purpose-get-extension-configuration ext-keyword)))
(unless config
(user-error "Missing extension configuration: %s" ext-keyword))
(purpose-conf-add-purposes config modes names regexps)
(purpose-set-extension-configuration ext-keyword config)))
(cl-defun purpose-remove-extension-purposes (ext-keyword &key modes names regexps)
"Remove purposes from an extension's purpose configuration.
EXT-KEYWORD is the same as in `purpose-set-extension-configuration'. MODES,
NAMES and REGEXPS must be valid configuration alists as described in
`purpose-mode-alist-p', `purpose-name-alist-p' and `purpose-regexp-alist-p'.
This function calls `purpose-compile-extended-configuration'.
Example:
(purpose-remove-extension-purposes :python
:modes '(inferior-python-mode)
:regexps '(\"\\.hy$\"))"
(let ((config (purpose-get-extension-configuration ext-keyword)))
(unless config
(user-error "Missing extension configuration: %s" ext-keyword))
(purpose-conf-remove-purposes config modes names regexps)
(purpose-set-extension-configuration ext-keyword config)))
(cl-defun purpose-add-user-purposes (&key modes names regexps)
"Add and compile multiple user purposes.
MODES must be a valid alist mapping major modes to purposes.
NAMES must be a valid alist mapping names to purposes.
REGEXPS must be a valid alist mapping regexps to purposes.
This function calls `purpose-compile-user-configuration' to
update user purposes.
Example:
(purpose-add-user-purposes :modes '((org-mode . org)
(help-mode . popup))
:names '((\"*scratch*\" . popup))
:regexps '((\"^\\*foo\" . terminal)))"
(setq purpose-user-mode-purposes (append modes purpose-user-mode-purposes)
purpose-user-name-purposes (append names purpose-user-name-purposes)
purpose-user-regexp-purposes (append regexps purpose-user-regexp-purposes))
(purpose-compile-user-configuration))
(cl-defun purpose-remove-user-purposes (&key modes names regexps)
"Remove and compile multiple user purposes.
MODES must be a list of major modes.
NAMES must be a list of names.
REGEXPS must be a list of regexps.
This function calls `purpose-compile-user-configuration' to
update user purposes.
Example:
(purpose-remove-user-purposes :modes '(org-mode help-mode)
:names '(\"*scratch*\")
:regexps '(\"^\\*foo\"))"
;; let-bind before setq-ing, so we don't apply partial changes if one
;; of MODES, NAMES or REGEXPS is malformed
(let ((new-modes (cl-set-difference purpose-user-mode-purposes modes
:test (lambda (entry mode)
(eql (car entry) mode))))
(new-names (cl-set-difference purpose-user-name-purposes names
:test (lambda (entry name)
(string= (car entry) name))))
(new-regexps (cl-set-difference purpose-user-regexp-purposes regexps
:test (lambda (entry regexp)
(string= (car entry) regexp)))))
(setq purpose-user-mode-purposes new-modes
purpose-user-name-purposes new-names
purpose-user-regexp-purposes new-regexps)
(purpose-compile-user-configuration)))
;;; change purposes temporarily
(defmacro purpose-save-purpose-config (&rest body)
"Save the purpose configuration, execute BODY, restore the configuration."
(declare (indent defun) (debug body))
`(let ((purpose--user-mode-purposes (copy-hash-table purpose--user-mode-purposes))
(purpose--user-name-purposes (copy-hash-table purpose--user-name-purposes))
(purpose--user-regexp-purposes (copy-hash-table purpose--user-regexp-purposes))
(purpose--extended-mode-purposes (copy-hash-table purpose--extended-mode-purposes))
(purpose--extended-name-purposes (copy-hash-table purpose--extended-name-purposes))
(purpose--extended-regexp-purposes (copy-hash-table purpose--extended-regexp-purposes))
(purpose--default-mode-purposes (copy-hash-table purpose--default-mode-purposes))
(purpose--default-name-purposes (copy-hash-table purpose--default-name-purposes))
(purpose--default-regexp-purposes (copy-hash-table purpose--default-regexp-purposes))
(purpose-use-default-configuration purpose-use-default-configuration)
(purpose-user-mode-purposes purpose-user-mode-purposes)
(purpose-user-name-purposes purpose-user-name-purposes)
(purpose-user-regexp-purposes purpose-user-regexp-purposes)
(purpose-extended-configuration purpose-extended-configuration))
,@body))
(defmacro purpose-with-temp-purposes (names regexps modes &rest body)
"Execute BODY with a temporary purpose configuration.
NAMES, REGEXPS and MODES should be alists as described in
`purpose-user-name-purposes', `purpose-user-regexp-purposes' and
`purpose-user-mode-purposes'.
NAMES, REGEXPS and MODES are used instead of the current purpose configuration
while BODY is executed. The purpose configuration is restored after BODY
is executed."
(declare (indent 3) (debug (sexp sexp sexp body)))
`(purpose-save-purpose-config
(let ((purpose-use-default-configuration nil)
(purpose-extended-configuration nil)
(purpose-user-name-purposes ,names)
(purpose-user-regexp-purposes ,regexps)
(purpose-user-mode-purposes ,modes))
(purpose-compile-user-configuration)
(purpose-compile-extended-configuration)
,@body)))
(defmacro purpose-with-empty-purposes (&rest body)
"Execute BODY with an empty purpose configuration.
The purpose configuration is restored after BODY is executed."
(declare (indent defun) (debug body))
`(purpose-with-temp-purposes nil nil nil ,@body))
(defmacro purpose-with-additional-purposes (names regexps modes &rest body)
"Execute BODY with an additional purpose configuration.
NAMES, REGEXPS and MODES should be alists as described in
`purpose-user-name-purposes', `purpose-user-regexp-purposes' and
`purpose-user-mode-purposes'.
NAMES, REGEXPS and MODES are used to add purposes to the current purpose
configuration while BODY is executed. The purpose configuration is restored
after BODY is executed."
(declare (indent 3) (debug (sexp sexp sexp body)))
`(purpose-save-purpose-config
(let ((purpose-user-name-purposes (append ,names purpose-user-name-purposes))
(purpose-user-regexp-purposes (append ,regexps purpose-user-regexp-purposes))
(purpose-user-mode-purposes (append ,modes purpose-user-mode-purposes)))
(purpose-compile-user-configuration)
,@body)))
;;; Initial compilation
(purpose-compile-user-configuration)
(purpose-compile-extended-configuration)
(purpose-compile-default-configuration)
(provide 'window-purpose-configuration)
;;; window-purpose-configuration.el ends here