34392
|
1 ;;; minibuf-eldef.el --- Only show defaults in prompts when applicable
|
|
2 ;;
|
|
3 ;; Copyright (C) 2000 Free Software Foundation, Inc.
|
|
4 ;;
|
|
5 ;; Author: Miles Bader <miles@gnu.org>
|
|
6 ;; Keywords: convenience
|
|
7
|
|
8 ;; This file is part of GNU Emacs.
|
|
9
|
|
10 ;; GNU Emacs is free software; you can redistribute it and/or modify
|
|
11 ;; it under the terms of the GNU General Public License as published by
|
|
12 ;; the Free Software Foundation; either version 2, or (at your option)
|
|
13 ;; any later version.
|
|
14
|
|
15 ;; GNU Emacs is distributed in the hope that it will be useful,
|
|
16 ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
17 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
18 ;; GNU General Public License for more details.
|
|
19
|
|
20 ;; You should have received a copy of the GNU General Public License
|
|
21 ;; along with GNU Emacs; see the file COPYING. If not, write to the
|
|
22 ;; Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
23 ;; Boston, MA 02111-1307, USA.
|
|
24
|
|
25 ;;; Commentary:
|
|
26 ;;
|
|
27 ;; Defines the mode `minibuffer-electric-default-mode'.
|
|
28 ;;
|
|
29 ;; When active, minibuffer prompts that show a default value only show
|
|
30 ;; the default when it's applicable -- that is, when hitting RET would
|
|
31 ;; yield the default value. If the user modifies the input such that
|
|
32 ;; hitting RET would enter a non-default value, the prompt is modified
|
|
33 ;; to remove the default indication (which is restored if the input is
|
|
34 ;; ever restore to the match the initial input).
|
|
35
|
|
36 ;;; Code:
|
|
37
|
|
38 (defvar minibuffer-default-in-prompt-regexps
|
|
39 '(("\\( (default\\>.*)\\):? \\'" . 1))
|
|
40 "*A list of regexps matching the parts of minibuffer prompts showing defaults.
|
|
41 When `minibuffer-electric-default-mode' is active, these regexps are
|
|
42 used to identify the portions of prompts to elide.
|
|
43
|
|
44 Each entry is either a string, which should be a regexp matching the
|
|
45 default portion of the prompt, or a cons cell, who's car is a regexp
|
|
46 matching the default part of the prompt, and who's cdr indicates the
|
|
47 regexp subexpression that matched.")
|
|
48
|
|
49
|
|
50 ;;; Internal variables
|
|
51
|
|
52 ;; A list of minibuffers to which we've added a post-command-hook.
|
|
53 (defvar minibuf-eldef-frobbed-minibufs nil)
|
|
54
|
|
55 ;;; The following are all local variables in the minibuffer
|
|
56
|
|
57 ;; Input pre-inserted into the minibuffer before the user can edit it.
|
|
58 (defvar minibuf-eldef-initial-input)
|
|
59 (make-variable-buffer-local 'minibuf-eldef-initial-input)
|
|
60 ;; and the length of the buffer with it inserted.
|
|
61 (defvar minibuf-eldef-initial-buffer-length)
|
|
62 (make-variable-buffer-local 'minibuf-eldef-initial-buffer-length)
|
|
63
|
|
64 ;; True if the current minibuffer prompt contains the default spec.
|
|
65 (defvar minibuf-eldef-showing-default-in-prompt)
|
|
66 (make-variable-buffer-local 'minibuf-eldef-showing-default-in-prompt)
|
|
67
|
|
68 ;; An overlay covering the default portion of the prompt
|
|
69 (defvar minibuf-eldef-overlay)
|
|
70 (make-variable-buffer-local 'minibuf-eldef-overlay)
|
|
71
|
|
72
|
|
73 ;;; Hook functions
|
|
74
|
|
75 ;; This function goes on minibuffer-setup-hook
|
|
76 (defun minibuf-eldef-setup-minibuffer ()
|
|
77 "Set up a minibuffer for `minibuffer-electric-default-mode'.
|
|
78 The prompt and initial input should already have been inserted."
|
|
79 (let ((prompt (field-string-no-properties (point-min)))
|
|
80 (regexps minibuffer-default-in-prompt-regexps)
|
|
81 (match nil)
|
|
82 (inhibit-point-motion-hooks t))
|
|
83 (save-excursion
|
|
84 (save-restriction
|
|
85 ;; Narrow to only the prompt
|
|
86 (goto-char (point-min))
|
|
87 (narrow-to-region (point) (field-end))
|
|
88 ;; See the prompt contains a default input indicator
|
|
89 (while regexps
|
|
90 (setq match (pop regexps))
|
|
91 (if (re-search-forward (if (stringp match) match (car match)) nil t)
|
|
92 (setq regexps nil)
|
|
93 (setq match nil)))))
|
|
94 (if (not match)
|
|
95 ;; Nope, so just make sure our post-command-hook isn't left around.
|
|
96 (remove-hook 'post-command-hook #'minibuf-eldef-update-minibuffer t)
|
|
97 ;; Yup; set things up so we can frob the prompt as the state of
|
|
98 ;; the input string changes.
|
|
99 (setq match (if (consp match) (cdr match) 0))
|
|
100 (setq minibuf-eldef-overlay
|
|
101 (make-overlay (match-beginning match) (match-end match)))
|
|
102 (setq minibuf-eldef-showing-default-in-prompt t)
|
|
103 (setq minibuf-eldef-initial-input
|
|
104 (field-string-no-properties (point-max)))
|
|
105 (setq minibuf-eldef-initial-buffer-length (point-max))
|
|
106 (add-to-list 'minibuf-eldef-frobbed-minibufs (current-buffer))
|
|
107 (add-hook 'post-command-hook #'minibuf-eldef-update-minibuffer nil t))))
|
|
108
|
|
109 ;; post-command-hook to swap prompts when necessary
|
|
110 (defun minibuf-eldef-update-minibuffer ()
|
|
111 "Update a minibuffer's prompt to include a default only when applicable.
|
|
112 This is intended to be used as a minibuffer post-command-hook for
|
|
113 `minibuffer-electric-default-mode'; the minibuffer should have already
|
|
114 been set up by `minibuf-eldef-setup-minibuffer'."
|
|
115 (unless (eq minibuf-eldef-showing-default-in-prompt
|
|
116 (and (= (point-max) minibuf-eldef-initial-buffer-length)
|
|
117 (string-equal (field-string-no-properties (point-max))
|
|
118 minibuf-eldef-initial-input)))
|
|
119 ;; swap state
|
|
120 (setq minibuf-eldef-showing-default-in-prompt
|
|
121 (not minibuf-eldef-showing-default-in-prompt))
|
|
122 (cond (minibuf-eldef-showing-default-in-prompt
|
|
123 (overlay-put minibuf-eldef-overlay 'invisible nil)
|
|
124 (overlay-put minibuf-eldef-overlay 'intangible nil))
|
|
125 (t
|
|
126 (overlay-put minibuf-eldef-overlay 'invisible t)
|
|
127 (overlay-put minibuf-eldef-overlay 'intangible t)))))
|
|
128
|
|
129
|
|
130 ;;; Note this definition must be at the end of the file, because
|
|
131 ;;; `define-minor-mode' actually calls the mode-function if the
|
|
132 ;;; associated variable is non-nil, which requires that all needed
|
|
133 ;;; functions be already defined. [This is arguably a bug in d-m-m]
|
|
134 ;;;###autoload
|
|
135 (define-minor-mode minibuffer-electric-default-mode
|
|
136 "Toggle Minibuffer Electric Default mode
|
|
137 When active, minibuffer prompts that show a default value only show the
|
|
138 default when it's applicable -- that is, when hitting RET would yield
|
|
139 the default value. If the user modifies the input such that hitting RET
|
|
140 would enter a non-default value, the prompt is modified to remove the
|
|
141 default indication.
|
|
142
|
|
143 With prefix argument ARG, turn on if positive, otherwise off.
|
|
144 Returns non-nil if the new state is enabled."
|
|
145 :global t
|
|
146 :group 'minibuffer
|
|
147 (if minibuffer-electric-default-mode
|
|
148 ;; Enable the mode
|
|
149 (add-hook 'minibuffer-setup-hook 'minibuf-eldef-setup-minibuffer)
|
|
150 ;; Disable the mode
|
|
151 (remove-hook 'minibuffer-setup-hook 'minibuf-eldef-setup-minibuffer)
|
|
152 ;; Remove our entry from any post-command-hook variable's it's still in
|
|
153 (dolist (minibuf minibuf-eldef-frobbed-minibufs)
|
|
154 (with-current-buffer minibuf
|
|
155 (remove-hook 'post-command-hook #'minibuf-eldef-update-minibuffer t)))
|
|
156 (setq minibuf-eldef-frobbed-minibufs nil)))
|
|
157
|
|
158
|
|
159 (provide 'minibuf-eldef)
|
|
160
|
|
161 ;;; minibuf-eldef.el ends here
|