-
Notifications
You must be signed in to change notification settings - Fork 33
/
Copy pathmmm-mode.el
309 lines (249 loc) · 11.3 KB
/
mmm-mode.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
;;; mmm-mode.el --- Allow Multiple Major Modes in a buffer -*- lexical-binding: t; -*-
;; Copyright (C) 1999-2004, 2012-2015, 2018, 2020 Free Software Foundation, Inc.
;; Emacs Lisp Archive Entry
;; Package: mmm-mode
;; Author: Michael Abraham Shulman <[email protected]>
;; Maintainer: Dmitry Gutov <[email protected]>
;; URL: https://github.com/dgutov/mmm-mode
;; Keywords: convenience, faces, languages, tools
;; Version: 0.5.11
;; Package-Requires: ((emacs "25.1") (cl-lib "0.2"))
;;{{{ GPL
;; This file 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 2, or (at your
;; option) any later version.
;; This file 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 GNU Emacs; see the file COPYING. If not, write to the
;; Free Software Foundation, Inc., 59 Temple Place - Suite 330,
;; Boston, MA 02111-1307, USA.
;;}}}
;;; Commentary:
;; MMM Mode is a minor mode that allows multiple major modes to
;; coexist in a single buffer. Refer to the documentation of the
;; function `mmm-mode' for more detailed information. This file
;; contains mode on/off functions and the mode keymap, but mostly
;; just loads all the subsidiary files.
;{{{ Parameter Naming
;; Since version 0.3.7, I've tried to use a uniform scheme for naming
;; parameters. Here's a brief summary.
;; BEG and END refer to the beginning and end of a region.
;; FRONT and BACK refer to the respective delimiters of a region.
;; FRONT- and BACK-OFFSET are the offsets from delimiter matches.
;; FRONT-BEG through BACK-END are the endings of the delimiters.
;; START and STOP bound actions, like searching, fontification, etc.
;}}}
;{{{ CL and Parameters
;; Keyword parameters can be nice because it makes it easier to see
;; what's getting passed as what. But I try not to use them in user
;; functions, because CL doesn't make good documentation strings.
;; Similarly, any hook or callback function can't take keywords,
;; since Emacs as a whole doesn't use them. And for small parameter
;; lists, they are overkill. So I use them only for a large number of
;; optional parameters, such as `mmm-make-region'.
;; An exception is the various submode class application functions,
;; which all take all their arguments as keywords, for consistency
;; and so the classes alist looks nice.
;; When using keyword arguments, defaults should *always* be supplied
;; in all arglists. (This pertains mostly to :start and :stop
;; arguments, usually defaulting to (point-min) and (point-max)
;; respectively.) `mmm-save-keywords' should only be used for lists
;; with more than four arguments, such as in `mmm-ify-by-regexp'.
;; In general, while I have no qualms about using things from CL like
;; `cl-mapl', `cl-loop' and `cl-destructuring-bind', I try not to use `cl-defun'
;; more than I have to. For one, it sometimes makes bad documentation
;; strings. Furthermore, to a `defun'ned function, a nil argument is
;; the same as no argument, so it will use its (manual) default, but
;; to a `cl-defun'ned function, a nil argument *is* the argument, so
;; any default specified in the arglist will be ignored. Confusion of
;; this type should be avoided when at all possible.
;;}}}
;;; Code:
(require 'cl-lib)
;; If we don't load font-lock now, but it is loaded later, the
;; necessary mmm-font-lock-* properties may not be there.
(require 'font-lock)
(require 'mmm-compat)
(require 'mmm-utils)
(require 'mmm-vars)
(require 'mmm-auto)
(require 'mmm-region)
(require 'mmm-class)
;; This file is set up to autoload by `mmm-auto.el'.
;; (require 'mmm-cmds)
(require 'mmm-univ)
;;{{{ Toggle Function
(defun mmm-mode (&optional arg)
"Minor mode to allow multiple major modes in one buffer.
Without ARG, toggle MMM Mode. With ARG, turn MMM Mode on if ARG is
positive and off otherwise.
Commands Available:
\\<mmm-mode-map>
\\{mmm-mode-map}
BASIC CONCEPTS
The idea of MMM Mode is to allow multiple major modes to coexist in
the same buffer. There is one `primary' major mode that controls
most of the buffer, and a number of `submodes' that each hold sway
over certain regions. The submode regions are usually highlighted by
a background color for ease of recognition. While the point is in a
submode region, the following changes \(are supposed to) occur:
1. The local keymap and the syntax table are that of the submode.
2. The mode line changes to show what submode region is active.
3. The major mode menu and mouse popup menu are that of the submode.
4. Some local variables of the submode shadow the default mode's.
5. Font-lock fontifies correctly for the submode.
6. Indentation function dispatches to the appropriate submode.
7. User-specified hooks run when the point enters or exits a submode.
The user may specify a number of hooks which may run when the point
transitions between submodes, or between the primary mode and a
submode. When a major mode is entered, the hook `mmm-x-enter-hook' is
run, where x is the name of the major mode. When a major mode is
left, the hook `mmm-y-exit-hook' is run, where y is the name of the
major mode. Users may also specify hooks with names of the form
`mmm-x-y-hook' which are run when the point leaves a mode named x,
and enters a mode named y.
For further information, including installation and configuration
instructions, see the info node `(mmm)Top' which is included with the
distribution of MMM Mode. Many of MMM's configuration variables are
available in customization group `mmm'."
(interactive "P")
(if (if arg (> (prefix-numeric-value arg) 0) (not mmm-mode))
(mmm-mode-on)
(mmm-mode-off)))
(add-to-list 'minor-mode-alist (list 'mmm-mode mmm-mode-string))
;;}}}
;;{{{ Mode On
(defun mmm-mode-on ()
"Turn on MMM Mode. See `mmm-mode'."
(interactive)
;; This function is called from mode hooks, so we need to make sure
;; we're not in a temporary buffer. We don't need to worry about
;; recursively ending up in ourself, however, since by that time the
;; variable `mmm-mode' will already be set.
(mmm-valid-buffer
(unless mmm-mode
(setq mmm-primary-mode major-mode)
(when (fboundp 'c-make-styles-buffer-local)
(c-make-styles-buffer-local t))
(mmm-update-mode-info major-mode)
(setq mmm-region-saved-locals-for-dominant
;; FIXME: Neither is defined in recent Emacs.
(cl-list* (list 'font-lock-cache-state nil)
(list 'font-lock-cache-position (make-marker))
(copy-tree (cdr (assq major-mode mmm-region-saved-locals-defaults)))))
;; Without the next line, the (make-marker) above gets replaced
;; with the starting value of nil, and all comes to naught.
(mmm-set-local-variables major-mode nil)
(mmm-add-hooks)
(mmm-fixup-skeleton)
(set (make-local-variable 'font-lock-fontify-region-function)
#'mmm-fontify-region)
(when (boundp 'syntax-begin-function)
(set (make-local-variable 'syntax-begin-function) nil))
(set (make-local-variable 'syntax-propertize-function)
#'mmm-syntax-propertize-function)
(set (make-local-variable 'indent-line-function) mmm-indent-line-function)
(setq mmm-mode t)
(condition-case err
(mmm-apply-all)
(mmm-error
;; Complain, but don't die, since we want files to go ahead
;; and be opened anyway, and the mode to go ahead and be
;; turned on. Should we delete all previously made submode
;; regions when we find an invalid one?
(message "%s" (error-message-string err))))
(run-hooks 'mmm-mode-hook)
(mmm-run-major-hook))))
;;}}}
;;{{{ Mode Off
(defun mmm-mode-off ()
"Turn off MMM Mode. See `mmm-mode'."
(interactive)
(when mmm-mode
(mmm-remove-hooks)
(mmm-clear-overlays)
(mmm-clear-history)
(mmm-clear-mode-ext-classes)
(mmm-clear-local-variables)
(mmm-update-submode-region)
(setq font-lock-fontify-region-function
(get mmm-primary-mode 'mmm-fontify-region-function))
(mmm-update-font-lock-buffer)
(mmm-refontify-maybe)
(setq mmm-mode nil)
;; Restore the mode line
(setq mmm-primary-mode-display-name nil
mmm-buffer-mode-display-name nil)
(mmm-set-mode-line)))
;;}}}
;;{{{ Mode Keymap
(defvar mmm-mode-map (make-sparse-keymap)
"Keymap for MMM Minor Mode.")
(defvar mmm-mode-prefix-map (make-sparse-keymap)
"Keymap for MMM Minor Mode after `mmm-mode-prefix-key'.")
(defvar mmm-mode-menu-map (make-sparse-keymap "MMM")
"Keymap for MMM Minor Mode menu.")
(defun mmm-define-key (key binding &optional keymap)
(define-key (or keymap mmm-mode-prefix-map)
(vector (append mmm-command-modifiers (list key)))
binding))
(when mmm-use-old-command-keys
(mmm-use-old-command-keys))
(mmm-define-key ?c 'mmm-ify-by-class)
(mmm-define-key ?x 'mmm-ify-by-regexp)
(mmm-define-key ?r 'mmm-ify-region)
(mmm-define-key ?b 'mmm-parse-buffer)
(mmm-define-key ?g 'mmm-parse-region)
(mmm-define-key ?% 'mmm-parse-block)
(mmm-define-key ?5 'mmm-parse-block)
(mmm-define-key ?k 'mmm-clear-current-region)
(mmm-define-key ?\ 'mmm-reparse-current-region)
(mmm-define-key ?e 'mmm-end-current-region)
(mmm-define-key ?z 'mmm-narrow-to-submode-region)
;; This one is exact, since C-h is (usually) already used for help.
(define-key mmm-mode-prefix-map [?h] 'mmm-insertion-help)
;; Default bindings to do insertion (dynamic)
(mmm-set-keymap-default mmm-mode-prefix-map 'mmm-insert-region)
;; Set up the prefix help command, since otherwise the default binding
;; overrides it.
(define-key mmm-mode-prefix-map (vector help-char) prefix-help-command)
;; And put it all onto the prefix key
(define-key mmm-mode-map mmm-mode-prefix-key mmm-mode-prefix-map)
;; Order matters for the menu bar.
(define-key mmm-mode-menu-map [off]
'("MMM Mode Off" . mmm-mode-off))
(define-key mmm-mode-menu-map [sep0] '(menu-item "----"))
(define-key mmm-mode-menu-map [clhist]
'("Clear History" . mmm-clear-history))
(define-key mmm-mode-menu-map [end]
'("End Current" . mmm-end-current-region))
(define-key mmm-mode-menu-map [clear]
'("Clear Current" . mmm-clear-current-region))
(define-key mmm-mode-menu-map [reparse]
'("Reparse Current" . mmm-reparse-current-region))
(define-key mmm-mode-menu-map [sep10] '(menu-item "----"))
(define-key mmm-mode-menu-map [ins-help]
'("List Insertion Keys" . mmm-insertion-help))
(define-key mmm-mode-menu-map [sep20] '(menu-item "----"))
(define-key mmm-mode-menu-map [region]
'(menu-item "MMM-ify Region" mmm-ify-region :enable mark-active))
(define-key mmm-mode-menu-map [regexp]
'("MMM-ify by Regexp" . mmm-ify-by-regexp))
(define-key mmm-mode-menu-map [class]
'("Apply Submode Class" . mmm-ify-by-class))
(define-key mmm-mode-menu-map [sep30] '(menu-item "----"))
(define-key mmm-mode-menu-map [parse-region]
'(menu-item "Parse Region" mmm-parse-region :enable mark-active))
(define-key mmm-mode-menu-map [parse-buffer]
'("Parse Buffer" . mmm-parse-buffer))
(define-key mmm-mode-menu-map [parse-block]
'("Parse Block" . mmm-parse-block))
(define-key mmm-mode-map [menu-bar mmm] (cons "MMM" mmm-mode-menu-map))
(add-to-list 'minor-mode-map-alist (cons 'mmm-mode mmm-mode-map))
;;}}}
(provide 'mmm-mode)
;;; mmm-mode.el ends here