-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathhighlight-function-calls.el
165 lines (137 loc) · 6.67 KB
/
highlight-function-calls.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
;;; highlight-function-calls.el --- Highlight function/macro calls -*- lexical-binding: t; -*-
;; Author: Adam Porter <adam@alphapapa.net>
;; Url: http://github.com/alphapapa/highlight-function-calls
;; Version: 0.2-pre
;; Package-Requires: ((emacs "24.4"))
;; Keywords: faces, highlighting
;;; Commentary:
;; This package highlights function symbols in function calls. This
;; makes them stand out from other symbols, which makes it easy to see
;; where calls to other functions are. Optionally, macros and special
;; forms can be highlighted as well. Also, a list of symbols can be
;; excluded from highlighting; by default, ones like +/-, </>, error,
;; require, etc. are excluded. Finally, the `not' function can be
;; highlighted specially.
;; Just run `highlight-function-calls-mode' to activate, or you can
;; add that to your `emacs-lisp-mode-hook' to do it automatically.
;;; License:
;; 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/>.
;;; Code:
(defgroup highlight-function-calls nil
"Options for highlighting function/macro calls and special forms."
:group 'faces)
(defface highlight-function-calls-face
'((t (:underline t)))
"Face for highlighting function calls."
:group 'highlight-function-calls)
(defface highlight-function-calls--not-face
'((t (:inherit font-lock-negation-char-face)))
"Face for highlighting `not'."
:group 'highlight-function-calls)
(defcustom highlight-function-calls-exclude-symbols
'( = + - / * < > <= >=
debug ; Not intended as an interactive function
error provide require signal throw user-error)
"List of symbols to not highlight."
:type '(repeat symbol))
(defcustom highlight-function-calls-macro-calls nil
"Whether to highlight macro calls."
:type 'boolean)
(defcustom highlight-function-calls-special-forms nil
"Whether to highlight special-forms calls."
:type 'boolean)
(defcustom highlight-function-calls-not nil
"Whether to highlight `not'."
:type 'boolean)
(defconst highlight-function-calls--keywords
`((
;; First we match an opening paren (which prevents matching
;; function names as arguments) or a function-quote (in which
;; case an argument should be matched). We also avoid matching
;; opening parens immediately after quotes (but this is not
;; perfect, since a list within a list could have its second
;; opening paren on a line by itself, in which case it would
;; appear to us like a normal function call and get
;; highlighted--but trying to fix that doesn't seem worth the
;; trouble, given how much trouble all this has already been).
;; FIXME: This does not avoid matching opening parens in quoted
;; lists. I don't know if we can fix this, because `syntax-ppss'
;; doesn't give any information about this. It might require
;; using semantic, which we probably don't want to mess with.
;; FIXME: It also doesn't avoid matching, e.g. the `map' in "(let
;; ((map". I'm not sure why.
,(rx (or bol (one-or-more space) ",")
(or "(" "#'"))
;; NOTE: The (0 nil) is required, although I don't understand
;; exactly why. This was confusing enough, following the
;; docstring for `font-lock-add-keywords'.
;; FIXME: This doesn't seem valid.
(0 nil)
;; Now we use a HIGHLIGHT MATCH-ANCHORED form to match the symbol
;; after the paren. We call the `highlight-function-calls--matcher'
;; function to test whether the face should be applied. We use a
;; PRE-MATCH-FORM to return a position at the end of the symbol,
;; which prevents matching function name symbols later on the
;; line, but we must not move the point in the process. We do
;; not use a POST-MATCH-FORM. Then we use the MATCH-HIGHLIGHT
;; form to highlight group 0, which is the whole symbol, we apply
;; the `highlight-function-calls-face' face, and we `prepend' it so
;; that it overrides existing faces; this way we even work with,
;; e.g. `rainbow-identifiers-mode', but only if we're activated
;; last.
(highlight-function-calls--matcher
(save-excursion
(forward-symbol 1)
(point))
nil
(0 highlight-function-calls--face-name prepend))))
"Keywords argument for `font-lock-add-keywords'.")
(defvar highlight-function-calls--face-name nil)
(defun highlight-function-calls--matcher (end)
"Match function symbols up to END.
The matcher function to be used by font lock mode."
(catch 'highlight-function-calls--matcher
(when (not (nth 5 (syntax-ppss)))
(while (re-search-forward (rx symbol-start (*? any) symbol-end) end t)
(let ((match (intern-soft (match-string 0))))
(when (and (or (functionp match)
(when highlight-function-calls-macro-calls
(macrop match))
(when highlight-function-calls-special-forms
(special-form-p match)))
(not (member match highlight-function-calls-exclude-symbols)))
(goto-char (match-end 0))
(setq highlight-function-calls--face-name
(pcase match
((and (or 'not 'null) (guard highlight-function-calls-not)) 'highlight-function-calls--not-face)
(_ 'highlight-function-calls-face)))
(throw 'highlight-function-calls--matcher t)))))
nil))
;;;###autoload
(define-minor-mode highlight-function-calls-mode
"Highlight function calls.
Toggle highlighting of function calls on or off.
With a prefix argument ARG, enable if ARG is positive, and
disable it otherwise. If called from Lisp, enable the mode if
ARG is omitted or nil, and toggle it if ARG is `toggle'."
:init-value nil :lighter nil :keymap nil
(let ((keywords highlight-function-calls--keywords))
(font-lock-remove-keywords nil keywords)
(when highlight-function-calls-mode
(font-lock-add-keywords nil keywords 'append)))
;; Refresh font locking.
(when font-lock-mode
(if (fboundp 'font-lock-flush)
(font-lock-flush)
(with-no-warnings (font-lock-fontify-buffer)))))
(provide 'highlight-function-calls)
;;; highlight-function-calls.el ends here