(delete-key-deletes-forward): New user-option.
[bpt/emacs.git] / lisp / minibuf-eldef.el
CommitLineData
66d058c4
MB
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.
41When `minibuffer-electric-default-mode' is active, these regexps are
42used to identify the portions of prompts to elide.
43
44Each entry is either a string, which should be a regexp matching the
45default portion of the prompt, or a cons cell, who's car is a regexp
46matching the default part of the prompt, and who's cdr indicates the
47regexp subexpression that matched.")
48
49\f
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\f
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'.
78The 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.
112This is intended to be used as a minibuffer post-command-hook for
113`minibuffer-electric-default-mode'; the minibuffer should have already
114been 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\f
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
137When active, minibuffer prompts that show a default value only show the
138default when it's applicable -- that is, when hitting RET would yield
139the default value. If the user modifies the input such that hitting RET
140would enter a non-default value, the prompt is modified to remove the
141default indication.
142
143With prefix argument ARG, turn on if positive, otherwise off.
144Returns 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