;; Abbrev mode commands for Emacs ;; Copyright (C) 1985 Richard M. Stallman. ;; This file is part of GNU Emacs. ;; GNU Emacs is distributed in the hope that it will be useful, ;; but without any warranty. No author or distributor ;; accepts responsibility to anyone for the consequences of using it ;; or for whether it serves any particular purpose or works at all, ;; unless he says so in writing. ;; Everyone is granted permission to copy, modify and redistribute ;; GNU Emacs, but only under the conditions described in the ;; document "GNU Emacs copying permission notice". An exact copy ;; of the document is supposed to have been given to you along with ;; GNU Emacs so that you can know how you may redistribute it all. ;; It should be in a file named COPYING. Among other things, the ;; copyright notice and this notice must be preserved on all copies. (defun abbrev-mode (arg) "Toggle abbrev mode. With arg, turn abbrev mode on iff arg is positive. In abbrev mode, inserting an abbreviation causes it to expand and be replaced by its expansion." (interactive "P") (setq abbrev-mode (if (null arg) (not abbrev-mode) (> (prefix-numeric-value arg) 0)))) (defvar edit-abbrevs-map nil "Keymap used in edit-abbrevs.") (defun kill-all-abbrevs () "Undefine all defined abbrevs." (interactive) (let ((tables abbrev-table-name-list)) (while tables (clear-abbrev-table (symbol-value (car tables))) (setq tables (cdr tables))))) (defun insert-abbrevs () "Insert after dot a description of all defined abbrevs. Mark is set after the inserted text." (interactive) (push-mark (save-excursion (let ((tables abbrev-table-name-list)) (while tables (insert-abbrev-table-description (car tables) t) (setq tables (cdr tables)))) (dot)))) (defun list-abbrevs () "Display a list of all defined abbrevs." (interactive) (with-output-to-temp-buffer "*Abbrevs*" (save-excursion (set-buffer standard-output) (prepare-abbrev-list-buffer)))) (defun prepare-abbrev-list-buffer () (let ((tables abbrev-table-name-list)) (while tables (insert-abbrev-table-description (car tables) t) (setq tables (cdr tables)))) (or edit-abbrevs-map (progn (setq edit-abbrevs-map (make-keymap)) (define-key edit-abbrevs-map "\^x\^s" 'edit-abbrevs-redefine))) (use-local-map edit-abbrevs-map)) (defun edit-abbrevs () "Alter abbrev definitions by editing a list of them. Selects a buffer containing a list of abbrev definitions. You can edit them and type C-X C-S to redefine abbrevs according to your editing. Buffer contains a header line for each abbrev table, which is the abbrev table name in parentheses. This is followed by one line per abbrev in that table: NAME USECOUNT EXPANSION HOOK where NAME and EXPANSION are strings with quotes, USECOUNT is an integer, and HOOK is any valid function or may be omitted (it is usually omitted)." (interactive) (switch-to-buffer (get-buffer-create "*Abbrevs*")) (prepare-abbrev-list-buffer)) (defun edit-abbrevs-redefine () "Redefine abbrevs according to current buffer contents" (interactive) (define-abbrevs t)) (defun define-abbrevs (&optional arg) "Define abbrevs according to current visible buffer contents. See documentation of edit-abbrevs for info on the format of the text you must have in the buffer. With argument, eliminate all abbrev definitions except the ones defined from the buffer now." (interactive "P") (if arg (kill-all-abbrevs)) (save-excursion (goto-char (dot-min)) (while (and (not (eobp)) (re-search-forward "^(" nil t)) (let* ((buf (current-buffer)) (table (read buf)) abbrevs) (forward-line 1) (while (progn (forward-line 1) (not (eolp))) (setq name (read buf) count (read buf) exp (read buf)) (skip-chars-backward " \t\n\f") (setq hook (if (not (eolp)) (read buf))) (skip-chars-backward " \t\n\f") (setq abbrevs (cons (list name exp hook count) abbrevs))) (define-abbrev-table table abbrevs))))) (defun read-abbrev-file (file) "Read abbrev definitions from file written with write-abbrev-file." (interactive "sRead abbrev file: ") (load (if (and file (> (length file) 0)) file abbrev-file-name)) (setq abbrevs-changed nil)) (defun quietly-read-abbrev-file (file) "Read abbrev definitions from file written with write-abbrev-file." (interactive "sRead abbrev file: ") (load (if (and file (> (length file) 0)) file abbrev-file-name) nil t) (setq abbrevs-changed nil)) (defun write-abbrev-file (file) "Write all abbrev definitions to file of Lisp code. The file can be loaded to define the same abbrevs." (interactive "sWrite abbrev file: ") (save-excursion (set-buffer (get-buffer-create " write-abbrev-file")) (erase-buffer) (let ((tables abbrev-table-name-list)) (while tables (insert-abbrev-table-description (car tables) nil) (setq tables (cdr tables)))) (or (and file (> (length file) 0)) (setq file abbrev-file-name)) (write-region 1 (dot-max) file) (erase-buffer))) (defun add-mode-abbrev (arg) "Define mode-specific abbrev for last word(s) before dot. Argument is how many words before dot form the expansion; or zero means the region is the expansion. Reads the abbreviation in the minibuffer." (interactive "p") (add-abbrev (if only-global-abbrevs global-abbrev-table (or local-abbrev-table (error "No per-mode abbrev table."))) "Mode" arg)) (defun add-global-abbrev (arg) "Define global (all modes) abbrev for last word(s) before dot. Argument is how many words before dot form the expansion; or zero means the region is the expansion. Reads the abbreviation in the minibuffer." (interactive "p") (add-abbrev global-abbrev-table "Global" arg)) (defun add-abbrev (table type arg) (let ((exp (and (>= arg 0) (buffer-substring (dot) (if (= arg 0) (mark) (save-excursion (forward-word (- arg)) (dot)))))) name) (setq name (read-string (format "%s abbrev for \"%s\": " type exp))) (if (or (null exp) (not (abbrev-expansion name table)) (y-or-n-p (format "%s expands to \"%s\"; redefine? " name (abbrev-expansion name table)))) (define-abbrev table name exp)))) (defun inverse-add-mode-abbrev (arg) "Define last word before dot as a mode-specific abbrev. With argument N, defines the Nth word before dot. Reads the expansion in the minibuffer. Expands the abbreviation after defining it." (interactive "p") (inverse-add-abbrev (if only-global-abbrevs global-abbrev-table (or local-abbrev-table (error "No per-mode abbrev table."))) "Mode" arg)) (defun inverse-add-global-abbrev (arg) "Define last word before dot as a global (mode-independent) abbrev. With argument N, defines the Nth word before dot. Reads the expansion in the minibuffer. Expands the abbreviation after defining it." (interactive "p") (inverse-add-abbrev global-abbrev-table "Global" arg)) (defun inverse-add-abbrev (table type arg) (let (name nameloc exp) (save-excursion (forward-word (- arg)) (setq name (buffer-substring (dot) (progn (forward-word 1) (setq nameloc (dot)))))) (setq exp (read-string (format "%s expansion for \"%s\": " type name))) (if (or (not (abbrev-expansion name table)) (y-or-n-p (format "%s expands to \"%s\"; redefine? " name (abbrev-expansion name table)))) (progn (define-abbrev table name exp) (save-excursion (goto-char nameloc) (expand-abbrev)))))) (defun abbrev-prefix-mark (&optional arg) "Mark current dot as the beginning of an abbrev. Abbrev to be expanded starts here rather than at beginning of word. This way, you can expand an abbrev with a prefix: insert the prefix, use this command, then insert the abbrev." (interactive "P") (or arg (expand-abbrev)) (setq abbrev-start-location (dot-marker) abbrev-start-location-buffer (current-buffer))) (defun expand-region-abbrevs (start end) "For abbrev occurrence in the region, offer to expand it. The user is asked to type y or n for each occurrence." (interactive "r") (save-excursion (goto-char (min start end)) (let ((lim (max start end))) (while (and (not (eobp)) (progn (forward-word 1) (<= (dot) lim))) (let ((modp (buffer-modified-p))) (if (expand-abbrev) (progn (set-buffer-modified-p modp) (unexpand-abbrev) (if (y-or-n-p "Expand this? ") (expand-abbrev)))))))))