Fix last change of grow_mini_window.
[bpt/emacs.git] / lisp / hl-line.el
CommitLineData
2230a6e3 1;;; hl-line.el --- highlight the current line -*- lexical-binding:t -*-
798330fe 2
ab422c4d 3;; Copyright (C) 1998, 2000-2013 Free Software Foundation, Inc.
798330fe
DL
4
5;; Author: Dave Love <fx@gnu.org>
e2f9e1b2 6;; Maintainer: FSF
798330fe 7;; Created: 1998-09-13
39f8a48b 8;; Keywords: faces, frames, emulations
798330fe 9
643c911e
DL
10;; This file is part of GNU Emacs.
11
eb3fa2cf 12;; GNU Emacs is free software: you can redistribute it and/or modify
798330fe 13;; it under the terms of the GNU General Public License as published by
eb3fa2cf
GM
14;; the Free Software Foundation, either version 3 of the License, or
15;; (at your option) any later version.
798330fe 16
643c911e 17;; GNU Emacs is distributed in the hope that it will be useful,
798330fe
DL
18;; but WITHOUT ANY WARRANTY; without even the implied warranty of
19;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20;; GNU General Public License for more details.
21
22;; You should have received a copy of the GNU General Public License
eb3fa2cf 23;; along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
798330fe
DL
24
25;;; Commentary:
26
e5f06fce
EZ
27;; Provides a local minor mode (toggled by M-x hl-line-mode) and
28;; a global minor mode (toggled by M-x global-hl-line-mode) to
23db85ff
LK
29;; highlight, on a suitable terminal, the line on which point is. The
30;; global mode highlights the current line in the selected window only
31;; (except when the minibuffer window is selected). This was
32;; implemented to satisfy a request for a feature of Lesser Editors.
33;; The local mode is sticky: it highlights the line about the buffer's
34;; point even if the buffer's window is not selected. Caveat: the
35;; buffer's point might be different from the point of a non-selected
36;; window. Set the variable `hl-line-sticky-flag' to nil to make the
37;; local mode behave like the global mode.
38
39;; You probably don't really want to use the global mode; if the
6772c8e1 40;; cursor is difficult to spot, try changing its color, relying on
23db85ff
LK
41;; `blink-cursor-mode' or both. The hookery used might affect
42;; response noticeably on a slow machine. The local mode may be
43;; useful in non-editing buffers such as Gnus or PCL-CVS though.
44
45;; An overlay is used. In the non-sticky cases, this overlay is
46;; active only on the selected window. A hook is added to
47;; `post-command-hook' to activate the overlay and move it to the line
48;; about point. To get the non-sticky behavior, `hl-line-unhighlight'
49;; is added to `pre-command-hook' as well. This function deactivates
50;; the overlay unconditionally in case the command changes the
51;; selected window. (It does so rather than keeping track of changes
52;; in the selected window).
53
0052fe2a
LK
54;; You could make variable `global-hl-line-mode' buffer-local and set
55;; it to nil to avoid highlighting specific buffers, when the global
56;; mode is used.
57
a4c6ebf9 58;; By default the whole line is highlighted. The range of highlighting
bf247b6e 59;; can be changed by defining an appropriate function as the
9fd76d04
MY
60;; buffer-local value of `hl-line-range-function'.
61
798330fe
DL
62;;; Code:
63
2230a6e3 64(defvar-local hl-line-overlay nil
3775cb5c 65 "Overlay used by Hl-Line mode to highlight the current line.")
3775cb5c
CY
66
67(defvar global-hl-line-overlay nil
68 "Overlay used by Global-Hl-Line mode to highlight the current line.")
69
798330fe 70(defgroup hl-line nil
99f2fb1f 71 "Highlight the current line."
eee06d26 72 :version "21.1"
eba5b4dd 73 :group 'convenience)
798330fe 74
3775cb5c
CY
75(defface hl-line
76 '((t :inherit highlight))
77 "Default face for highlighting the current line in Hl-Line mode."
78 :version "22.1"
798330fe
DL
79 :group 'hl-line)
80
3775cb5c
CY
81(defcustom hl-line-face 'hl-line
82 "Face with which to highlight the current line in Hl-Line mode."
83 :type 'face
84 :group 'hl-line
85 :set (lambda (symbol value)
86 (set symbol value)
87 (dolist (buffer (buffer-list))
88 (with-current-buffer buffer
89 (when hl-line-overlay
90 (overlay-put hl-line-overlay 'face hl-line-face))))
91 (when global-hl-line-overlay
92 (overlay-put global-hl-line-overlay 'face hl-line-face))))
93
23db85ff 94(defcustom hl-line-sticky-flag t
88821ca0 95 "Non-nil means the HL-Line mode highlight appears in all windows.
23db85ff
LK
96Otherwise Hl-Line mode will highlight only in the selected
97window. Setting this variable takes effect the next time you use
88821ca0
CY
98the command `hl-line-mode' to turn Hl-Line mode on.
99
100This variable has no effect in Global Highlight Line mode.
101For that, use `global-hl-line-sticky-flag'."
23db85ff 102 :type 'boolean
bf247b6e 103 :version "22.1"
23db85ff
LK
104 :group 'hl-line)
105
88821ca0
CY
106(defcustom global-hl-line-sticky-flag nil
107 "Non-nil means the Global HL-Line mode highlight appears in all windows.
108Otherwise Global Hl-Line mode will highlight only in the selected
109window. Setting this variable takes effect the next time you use
110the command `global-hl-line-mode' to turn Global Hl-Line mode on."
111 :type 'boolean
112 :version "24.1"
113 :group 'hl-line)
114
9fd76d04
MY
115(defvar hl-line-range-function nil
116 "If non-nil, function to call to return highlight range.
117The function of no args should return a cons cell; its car value
bf247b6e 118is the beginning position of highlight and its cdr value is the
9fd76d04
MY
119end position of highlight in the buffer.
120It should return nil if there's no region to be highlighted.
121
122This variable is expected to be made buffer-local by modes.")
123
ceafc2fa 124;;;###autoload
3a850efa 125(define-minor-mode hl-line-mode
06e21633
CY
126 "Toggle highlighting of the current line (Hl-Line mode).
127With a prefix argument ARG, enable Hl-Line mode if ARG is
128positive, and disable it otherwise. If called from Lisp, enable
129the mode if ARG is omitted or nil.
23db85ff 130
06e21633
CY
131Hl-Line mode is a buffer-local minor mode. If
132`hl-line-sticky-flag' is non-nil, Hl-Line mode highlights the
23db85ff
LK
133line about the buffer's point in all windows. Caveat: the
134buffer's point might be different from the point of a
135non-selected window. Hl-Line mode uses the function
136`hl-line-highlight' on `post-command-hook' in this case.
137
138When `hl-line-sticky-flag' is nil, Hl-Line mode highlights the
139line about point in the selected window only. In this case, it
140uses the function `hl-line-unhighlight' on `pre-command-hook' in
141addition to `hl-line-highlight' on `post-command-hook'."
b862cd15 142 :group 'hl-line
3a850efa
DL
143 (if hl-line-mode
144 (progn
23db85ff
LK
145 ;; In case `kill-all-local-variables' is called.
146 (add-hook 'change-major-mode-hook #'hl-line-unhighlight nil t)
147 (if hl-line-sticky-flag
148 (remove-hook 'pre-command-hook #'hl-line-unhighlight t)
149 (add-hook 'pre-command-hook #'hl-line-unhighlight nil t))
150 (hl-line-highlight)
9df382fe 151 (add-hook 'post-command-hook #'hl-line-highlight nil t))
23db85ff 152 (remove-hook 'post-command-hook #'hl-line-highlight t)
3a850efa 153 (hl-line-unhighlight)
23db85ff
LK
154 (remove-hook 'change-major-mode-hook #'hl-line-unhighlight t)
155 (remove-hook 'pre-command-hook #'hl-line-unhighlight t)))
b184b2dd 156
2230a6e3
SM
157(defun hl-line-make-overlay ()
158 (let ((ol (make-overlay (point) (point))))
159 (overlay-put ol 'priority -50) ;(bug#16192)
160 (overlay-put ol 'face hl-line-face)
161 ol))
162
57e46f94 163(defun hl-line-highlight ()
a4c6ebf9 164 "Activate the Hl-Line overlay on the current line."
23db85ff
LK
165 (if hl-line-mode ; Might be changed outside the mode function.
166 (progn
167 (unless hl-line-overlay
2230a6e3 168 (setq hl-line-overlay (hl-line-make-overlay))) ; To be moved.
23db85ff
LK
169 (overlay-put hl-line-overlay
170 'window (unless hl-line-sticky-flag (selected-window)))
9fd76d04 171 (hl-line-move hl-line-overlay))
23db85ff 172 (hl-line-unhighlight)))
57e46f94
RS
173
174(defun hl-line-unhighlight ()
23db85ff 175 "Deactivate the Hl-Line overlay on the current line."
b1bad9f3
JB
176 (when hl-line-overlay
177 (delete-overlay hl-line-overlay)))
57e46f94 178
23db85ff
LK
179;;;###autoload
180(define-minor-mode global-hl-line-mode
06e21633
CY
181 "Toggle line highlighting in all buffers (Global Hl-Line mode).
182With a prefix argument ARG, enable Global Hl-Line mode if ARG is
183positive, and disable it otherwise. If called from Lisp, enable
184the mode if ARG is omitted or nil.
23db85ff 185
88821ca0
CY
186If `global-hl-line-sticky-flag' is non-nil, Global Hl-Line mode
187highlights the line about the current buffer's point in all
188windows.
189
23db85ff
LK
190Global-Hl-Line mode uses the functions `global-hl-line-unhighlight' and
191`global-hl-line-highlight' on `pre-command-hook' and `post-command-hook'."
192 :global t
193 :group 'hl-line
194 (if global-hl-line-mode
195 (progn
196 (add-hook 'pre-command-hook #'global-hl-line-unhighlight)
197 (add-hook 'post-command-hook #'global-hl-line-highlight))
198 (global-hl-line-unhighlight)
199 (remove-hook 'pre-command-hook #'global-hl-line-unhighlight)
200 (remove-hook 'post-command-hook #'global-hl-line-highlight)))
201
202(defun global-hl-line-highlight ()
503edac9 203 "Highlight the current line in the current window."
23db85ff 204 (when global-hl-line-mode ; Might be changed outside the mode function.
290d5b58 205 (unless (window-minibuffer-p)
23db85ff 206 (unless global-hl-line-overlay
2230a6e3 207 (setq global-hl-line-overlay (hl-line-make-overlay))) ; To be moved.
88821ca0
CY
208 (overlay-put global-hl-line-overlay 'window
209 (unless global-hl-line-sticky-flag
210 (selected-window)))
9fd76d04 211 (hl-line-move global-hl-line-overlay))))
23db85ff
LK
212
213(defun global-hl-line-unhighlight ()
214 "Deactivate the Global-Hl-Line overlay on the current line."
b1bad9f3
JB
215 (when global-hl-line-overlay
216 (delete-overlay global-hl-line-overlay)))
23db85ff 217
9fd76d04 218(defun hl-line-move (overlay)
a4c6ebf9 219 "Move the Hl-Line overlay.
9fd76d04 220If `hl-line-range-function' is non-nil, move the OVERLAY to the position
a4c6ebf9 221where the function returns. If `hl-line-range-function' is nil, fill
9fd76d04
MY
222the line including the point by OVERLAY."
223 (let (tmp b e)
224 (if hl-line-range-function
225 (setq tmp (funcall hl-line-range-function)
226 b (car tmp)
227 e (cdr tmp))
228 (setq tmp t
229 b (line-beginning-position)
230 e (line-beginning-position 2)))
231 (if tmp
232 (move-overlay overlay b e)
233 (move-overlay overlay 1 1))))
234
b1bad9f3
JB
235(defun hl-line-unload-function ()
236 "Unload the Hl-Line library."
237 (global-hl-line-mode -1)
238 (save-current-buffer
239 (dolist (buffer (buffer-list))
240 (set-buffer buffer)
241 (when hl-line-mode (hl-line-mode -1))))
242 ;; continue standard unloading
243 nil)
244
798330fe
DL
245(provide 'hl-line)
246
247;;; hl-line.el ends here