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
|
;;; rainbow-identifiers.el --- Highlight identifiers according to their names -*- lexical-binding: t -*-
;; Author: Fanael Linithien <fanael4@gmail.com>
;; URL: https://github.com/Fanael/rainbow-identifiers
;; Version: 0.2.2
;; Package-Requires: ((emacs "24"))
;; This file is NOT part of GNU Emacs.
;; Copyright (c) 2014, Fanael Linithien
;; All rights reserved.
;;
;; Redistribution and use in source and binary forms, with or without
;; modification, are permitted provided that the following conditions are
;; met:
;;
;; * Redistributions of source code must retain the above copyright
;; notice, this list of conditions and the following disclaimer.
;; * Redistributions in binary form must reproduce the above copyright
;; notice, this list of conditions and the following disclaimer in the
;; documentation and/or other materials provided with the distribution.
;;
;; THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
;; IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
;; TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
;; PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER
;; OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
;; EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
;; PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
;; PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
;; LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
;; NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
;; SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
;;; Commentary:
;; Minor mode providing highlighting of identifiers based on their
;; names. Each identifier gets a color based on a hash of its name.
;;
;; Use `rainbow-identifiers-mode' to enable/disable.
;;
;; Default colors try to be reasonable, but they can be changed by
;; changing the faces `rainbow-identifiers-identifier-<number>'.
;;; Code:
(require 'color)
(defgroup rainbow-identifiers nil
"Highlight identifiers according to their names."
:prefix "rainbow-identifiers-"
:group 'convenience)
(defcustom rainbow-identifiers-choose-face-function
'rainbow-identifiers-predefined-choose-face
"The function used to choose faces used to highlight identifiers.
It should take a single integer, which is the hash of the identifier
currently being highlighting, and return a value suitable to use
as a value of the `face' text property."
:type 'function
:group 'rainbow-identifiers)
(defcustom rainbow-identifiers-filter-functions '(rainbow-identifiers-face-overridable)
"Abnormal hook run to determine whether to rainbow-highlight an identifier.
Two arguments are passed to each function: the position of the beginning and end
of the identifier currently being considered.
Each function should return non-nil if and only if it considers the identifier
to be eligible to rainbow-highlighting. Identifier is rainbow-highlighted only
when all hook functions consider it eligible."
:type '(repeat function)
:group 'rainbow-identifiers)
(defcustom rainbow-identifiers-faces-to-override nil
"List of faces that `rainbow-identifiers' is allowed to override.
It has an effect only when `rainbow-identifiers-face-overridable' is in
`rainbow-identifiers-filter-functions'."
:type '(repeat face)
:group 'rainbow-identifiers)
(defconst rainbow-identifiers--hash-bytes-to-use
(ceiling (/ (log most-positive-fixnum 2) 8.0))
"Number of bytes of returned hash to actually use.")
(defun rainbow-identifiers--hash-function (identifier)
"Hash function used to determine the face of IDENTIFIER."
(let* ((hash (secure-hash 'sha1 identifier nil nil t))
(len (length hash))
(i (- len rainbow-identifiers--hash-bytes-to-use))
(result 0))
(while (< i len)
(setq result (+ (* result 256) (aref hash i)))
(setq i (1+ i)))
result))
;; Predefined face chooser:
(defgroup rainbow-identifiers-faces nil
"Faces for highlighting identifiers."
:group 'rainbow-identifiers
:group 'faces)
(eval-when-compile
(defmacro rainbow-identifiers--define-faces ()
(let ((faces '())
(light-colors ["#78683f" "#43783f" "#3f7178" "#513f78" "#783f5a"
"#707e4f" "#4f7e67" "#4f5c7e" "#7a4f7e" "#7e544f"
"#783778" "#784437" "#5e7837" "#37785e" "#374478"])
(dark-colors ["#9999bb" "#bb99b4" "#bba699" "#a6bb99" "#99bbb4"
"#e0d0a0" "#a3e0a0" "#a0d6e0" "#b6a0e0" "#e0a0bc"
"#a7c0b9" "#a7aac0" "#c0a7bd" "#c0afa7" "#b3c0a7"]))
(dotimes (i 15)
(push `(defface ,(intern (format "rainbow-identifiers-identifier-%d" (1+ i)))
'((((class color) (background dark)) :foreground ,(aref dark-colors i))
(((class color) (background light)) :foreground ,(aref light-colors i)))
,(format "Identifier face #%d" (1+ i))
:group 'rainbow-identifiers-faces)
faces))
`(progn ,@faces))))
(rainbow-identifiers--define-faces)
(defcustom rainbow-identifiers-face-count 15
"Number of faces used for highlighting identifiers.
You can increase this value if you define enough faces named
rainbow-identifiers-identifier-<number>."
:type 'integer
:group 'rainbow-identifiers)
(defun rainbow-identifiers-predefined-choose-face (hash)
"Use HASH to choose one of the `rainbow-identifiers-identifier-N' faces."
(intern-soft
(concat "rainbow-identifiers-identifier-"
(number-to-string (1+ (mod hash rainbow-identifiers-face-count))))))
;; CIE L*a*b* face chooser:
(defcustom rainbow-identifiers-cie-l*a*b*-lightness 50
"The lightness of the generated colors.
Internally, this is the L* color coordinate."
:type 'number
:group 'rainbow-identifiers)
(defcustom rainbow-identifiers-cie-l*a*b*-saturation 15
"The saturation of generated colors.
Internally, this is the radius of a circle where the X and Y
coordinates of a point on that circle are the a* and b* color
coordinates, respectively."
:type 'number
:group 'rainbow-identifiers)
(defcustom rainbow-identifiers-cie-l*a*b*-color-count 65536
"The number of different colors to generate."
:type 'integer
:group 'rainbow-identifiers)
(defun rainbow-identifiers-cie-l*a*b*-choose-face (hash)
"Use HASH to choose a face with a generated foreground color.
The colors are chosen from the CIE L*a*b* color space. If a color not
representable in sRGB is chosen, the components are clamped.
The color generation can be influenced by changing
`rainbow-identifiers-cie-l*a*b*-lightness',
`rainbow-identifiers-cie-l*a*b*-saturation' and
`rainbow-identifiers-cie-l*a*b*-color-count'."
(let* ((bucket (float (mod hash rainbow-identifiers-cie-l*a*b*-color-count)))
(angle (* 2 float-pi (/ bucket rainbow-identifiers-cie-l*a*b*-color-count)))
(a (* rainbow-identifiers-cie-l*a*b*-saturation (cos angle)))
(b (* rainbow-identifiers-cie-l*a*b*-saturation (sin angle))))
(let ((color (color-lab-to-srgb rainbow-identifiers-cie-l*a*b*-lightness a b)))
;; Clamp the color if the result is not representable in sRGB.
(let ((i color))
(while i
(setcar i (max 0.0 (min 1.0 (car i))))
(setq i (cdr i))))
(list (list :foreground (apply #'color-rgb-to-hex color))))))
;; Face filter:
(defun rainbow-identifiers-face-overridable (begin _end)
"Test if the face of the identifier under BEGIN is overridable."
(let ((face (get-text-property begin 'face)))
(cond
((null face)
t)
((listp face)
(catch 'rainbow-identifiers--face-overridable
(dolist (face* face)
(unless (memq face* rainbow-identifiers-faces-to-override)
(throw 'rainbow-identifiers--face-overridable nil)))
t))
(t
(memq face rainbow-identifiers-faces-to-override)))))
(defvar rainbow-identifiers--face nil)
(defun rainbow-identifiers--matcher (end)
"The matcher function to be used by font lock mode."
(catch 'rainbow-identifiers--matcher
(while (re-search-forward (rx symbol-start (*? any) symbol-end) end t)
(let ((beginning (match-beginning 0))
(end (match-end 0)))
(when (run-hook-with-args-until-failure 'rainbow-identifiers-filter-functions beginning end)
(let* ((identifier (buffer-substring-no-properties beginning end))
(hash (rainbow-identifiers--hash-function identifier)))
(setq rainbow-identifiers--face (funcall rainbow-identifiers-choose-face-function hash))
(throw 'rainbow-identifiers--matcher t)))))
nil))
;;;###autoload
(define-minor-mode rainbow-identifiers-mode
"Highlight identifiers according to their names.
Toggle Rainbow Identifiers mode on or off.
With a prefix argument ARG, enable Rainbow Identifiers mode 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 ""
:keymap nil
(let ((keywords '((rainbow-identifiers--matcher 0 rainbow-identifiers--face prepend))))
(font-lock-remove-keywords nil keywords)
(when rainbow-identifiers-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 'rainbow-identifiers)
;;; rainbow-identifiers.el ends here
|