;;; delight.el --- A dimmer switch for your lighter text -*- lexical-binding:t -*- ;; ;; Copyright (C) 2013-2020 Free Software Foundation, Inc. ;; Author: Phil Sainty ;; Maintainer: Phil Sainty ;; URL: https://savannah.nongnu.org/projects/delight ;; Package-Requires: ((cl-lib "0.5") (nadvice "0.3")) ;; Keywords: convenience ;; Created: 25 Jun 2013 ;; Version: 1.7 ;; This program is free software; you can redistribute it and/or modify ;; it under the terms of the GNU General Public License as published by ;; the Free Software Foundation, either version 3 of the License, or ;; (at your option) any later version. ;; This program is distributed in the hope that it will be useful, ;; but WITHOUT ANY WARRANTY; without even the implied warranty of ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ;; GNU General Public License for more details. ;; You should have received a copy of the GNU General Public License ;; along with this program. If not, see . ;;; Commentary: ;; ;; Enables you to customise the mode names displayed in the mode line. ;; ;; For major modes, the buffer-local `mode-name' variable is modified. ;; For minor modes, the associated value in `minor-mode-alist' is set. ;; ;; Example usage: ;; ;; ;; Delighting a single mode at a time: ;; (require 'delight) ;; (delight 'abbrev-mode " Abv" "abbrev") ;; (delight 'rainbow-mode) ;; ;; ;; Delighting multiple modes together: ;; (require 'delight) ;; (delight '((abbrev-mode " Abv" "abbrev") ;; (smart-tab-mode " \\t" "smart-tab") ;; (eldoc-mode nil "eldoc") ;; (rainbow-mode) ;; (overwrite-mode " Ov" t) ;; (emacs-lisp-mode "Elisp" :major))) ;; ;; The first argument is the mode symbol. ;; ;; The second argument is the replacement name to use in the mode line ;; (or nil to hide it). ;; ;; The third argument is either the keyword :major for major modes or, ;; for minor modes, the library which defines the mode. This is passed ;; to `eval-after-load' and so should be either the name (as a string) ;; of the library file which defines the mode, or the feature (symbol) ;; provided by that library. If this argument is nil, the mode symbol ;; will be passed as the feature. If this argument is either t or 'emacs ;; then it is assumed that the mode is already loaded (you can use this ;; with standard minor modes that are pre-loaded by default when Emacs ;; starts). ;; ;; In the above example, `rainbow-mode' is the symbol for both the minor ;; mode and the feature which provides it, and its lighter text will be ;; hidden from the mode line. ;; ;; To determine which library defines a mode, use e.g.: C-h f eldoc-mode. ;; The name of the library is displayed in the first paragraph, with an ;; ".el" suffix (in this example it displays "eldoc.el", and therefore we ;; could use the value "eldoc" for the library). ;; ;; If you simply cannot figure out which library to specify, an ;; alternative approach is to evaluate (delight 'something-mode nil t) ;; once you know for sure that the mode has already been loaded, perhaps ;; by using the mode hook for that mode. ;; ;; If all else fails, it's worth looking at C-h v minor-mode-alist ;; (after enabling the minor mode in question). There are rare cases ;; where the entry in `minor-mode-alist' has a different symbol to the ;; minor mode with which it is associated, and in these situations you ;; will need to specify the name in the alist, rather than the name of ;; the mode itself. Known examples (and how to delight them) are: ;; ;; `auto-fill-mode': (delight 'auto-fill-function " AF" t) ;; `server-mode': (delight 'server-buffer-clients " SV" 'server) ;; ;; * Important notes: ;; ;; Although strings are common, any mode line construct is permitted as ;; the value (for both minor and major modes); so before you override a ;; value you should check the existing one, as you may want to replicate ;; any structural elements in your replacement if it turns out not to be ;; a simple string. ;; ;; For major modes, M-: mode-name ;; For minor modes, M-: (cadr (assq 'MODE minor-mode-alist)) ;; for the minor MODE in question. ;; ;; Conversely, you may incorporate additional mode line constructs in ;; your replacement values, if you so wish. e.g.: ;; ;; (delight 'emacs-lisp-mode ;; '("Elisp" (lexical-binding ":Lex" ":Dyn")) ;; :major) ;; ;; See `mode-line-format' for information about mode line constructs, and ;; M-: (info "(elisp) Mode Line Format") for further details. ;; ;; Settings for minor modes are held in a global variable and tend to take ;; immediate effect upon calling ‘delight’. Major mode names are held in ;; buffer-local variables, however, so changes to these will not take ;; effect in a given buffer unless the major mode is called again, or the ;; buffer is reverted. Calling M-x normal-mode is sufficient in most ;; cases. ;; ;; Also bear in mind that some modes may dynamically update these values ;; themselves (for instance dired-mode updates mode-name if you change the ;; sorting criteria) in which cases this library may prove inadequate. ;; ;; Some modes also implement direct support for customizing these values; ;; so if delight is not sufficient for a particular mode, be sure to check ;; whether the library in question provides its own way of doing this. ;; ;; * Conflict with `c-mode' and related major modes: ;; ;; Major modes based on cc-mode.el (including ‘c-mode’, ‘c++-mode’, and ;; derivatives such as ‘php-mode’) cannot be delighted, due to Emacs bug ;; #2034: https://debbugs.gnu.org/cgi/bugreport.cgi?bug=2034 ;; ;; cc-mode.el assumes that ‘mode-name’ is always a string (which was true ;; in Emacs 22 and earlier), while delight.el makes use of the fact that ;; ‘mode-name’ can (since Emacs 23) contain any mode line construct. The ;; two are therefore incompatible. ;; ;; The symptom of this conflict is the following error (where the "..." ;; varies): ;; ;; (wrong-type-argument stringp (delight-mode-name-inhibit ...)) ;; ;; The conflicting function is ‘c-update-modeline’ which adds the various ;; suffix characters documented at M-: (info "(ccmode) Minor Modes"). ;; (E.g. In the mode line of a ‘c-mode’ buffer, the name C might be ;; changed to "C/*l" or similar, depending on the minor modes.) ;; ;; If you are willing (or indeed wishing) to eliminate those suffixes ;; entirely for all relevant major modes, then you can work around this ;; conflict between the two libraries by disabling ‘c-update-modeline’ ;; entirely, like so: ;; ;; (advice-add 'c-update-modeline :override #'ignore) ;; ;; * Integration with mode line replacement libraries: ;; ;; Libraries which replace the standard mode line are liable to conflict ;; with delight's treatment of major modes, as such libraries invariably ;; need to call `format-mode-line', which otherwise happens only in ;; circumstances in which delight wishes to show the original mode-name. ;; ;; These libraries (or custom advice) can prevent this by let-binding ;; `delight-mode-name-inhibit' to nil around calls to `format-mode-line' ;; which will ensure that the delighted `mode-name' is displayed. ;; ;; * Configuration via use-package: ;; ;; The popular `use-package' macro supports delight.el so you can also ;; delight modes as part of your package configurations. See its README ;; file for details. ;;; Change Log: ;; ;; 1.7 (2020-07-11) ;; - Add `delight-version'. ;; - Support loading newer versions over the top of older versions. ;; - Support `unload-feature'. ;; - Rename `delighted-modes' to `delight-delighted-modes'. ;; - Rename `delight--inhibit' to `delight-mode-name-inhibit', and ;; document its uses. ;; 1.6 (2019-07-23) ;; - Use cl-lib, nadvice, and lexical-binding. ;; - Rename `inhibit-mode-name-delight' to `delight--inhibit'. ;; 1.5 (2016-03-01) ;; - Support FILE value t, meaning that the minor MODE in question ;; is guaranteed to already be loaded. ;; 1.4 (2016-02-28) ;; - Respect `inhibit-mode-name-delight' when already set. ;; 1.3 (2014-05-30) ;; - Add support for `mode-line-mode-menu'. ;; 1.2 (2014-05-04) ;; - Bug fix for missing 'cl requirement for destructuring-bind macro. ;; 1.1 (2014-05-04) ;; - Allow the keyword :major as the FILE argument for major modes, ;; to avoid also processing them as minor modes. ;; 1.0 (2013-06-25) ;; - Initial release. ;;; Code: (eval-when-compile (require 'cl-lib)) (require 'nadvice) (defconst delight--latest-version "1.7") ;; Check whether a newer version is being loaded over an older one. ;; ;; If `delight-version' has an existing value which is less than ;; `delight--latest-version', then an earlier version was already loaded, ;; and we must perform any necessary updates (see "Live upgrades" below). ;; ;; If `delight-version' is unbound then most likely there was no older ;; version loaded; however, prior to version 1.7 `delight-version' was not ;; defined at all, and so we need to detect that scenario too. (defvar delight-version (if (not (featurep 'delight)) ;; The normal case: delight was not already loaded. delight--latest-version ;; Otherwise delight was loaded. However, as this initial value code is ;; being evaluated, the loaded version had not defined `delight-version'. (cond ;; In 1.5 and earlier, `delight--format-mode-line' didn't exist. ;; (Earlier versions can be treated as 1.5 for upgrade purposes.) ((not (fboundp 'delight--format-mode-line)) "1.5") ;; In 1.6 `delight--inhibit' wasn't an alias. ((eq (indirect-variable 'delight--inhibit) 'delight--inhibit) "1.6") ;; If we get to here, we've probably used `eval-defun' on this defvar. (t delight--latest-version))) "The loaded version of delight.el.") (define-obsolete-variable-alias 'delighted-modes 'delight-delighted-modes "delight-1.7") (defvar delight-delighted-modes nil "List of specs for modifying the display of mode names in the mode line. See `delight'.") ;;;###autoload (defun delight (spec &optional value file) "Modify the lighter value displayed in the mode line for the given mode SPEC if and when the mode is loaded. SPEC can be either a mode symbol, or a list containing multiple elements of the form (MODE VALUE FILE). In the latter case the two optional arguments are omitted, as they are instead specified for each element of the list. For minor modes, VALUE is the replacement lighter value (or nil to disable) to set in the `minor-mode-alist' variable. For major modes VALUE is the replacement buffer-local `mode-name' value to use when a buffer changes to that mode. In both cases VALUE is commonly a string, but may in fact contain any valid mode line construct. For details see the `mode-line-format' variable, and Info node `(elisp) Mode Line Format'. The FILE argument is passed through to `eval-after-load'. If FILE is nil then the mode symbol is passed as the required feature. If FILE is t then it is assumed that the mode is already loaded. (Note that you can also use \\='emacs for this purpose). These FILE options are relevant to minor modes only. For major modes you should specify the keyword :major as the value of FILE, to prevent the mode being treated as a minor mode." (let ((glum (if (consp spec) spec (list (list spec value file))))) (while glum (cl-destructuring-bind (mode &optional value file) (pop glum) (assq-delete-all mode delight-delighted-modes) (add-to-list 'delight-delighted-modes (list mode value file)) ;; Major modes are handled in `after-change-major-mode-hook'. ;; Minor modes are handled at load time: (unless (eq file :major) (eval-after-load (if (eq file t) 'emacs (or file mode)) `(when (featurep 'delight) (let ((minor-delight (assq ',mode minor-mode-alist))) (when minor-delight (setcar (cdr minor-delight) ',value) (delight-mode-line-mode-menu ',mode ',value)))))))))) (defun delight-mode-line-mode-menu (mode value) "Delight `mode-line-mode-menu' (the \"Toggle minor modes\" menu) so that the Lighter text displayed in the menu matches that displayed in the mode line (when such menu items exist). The expected naming scheme for the menu items is: \"Friendly name (Lighter)\" e.g.: \"Highlight changes (Chg)\". We replace the \"Lighter\" portion of that with our delighted VALUE, for the specified MODE, unless VALUE is empty/nil, in which case we remove the text and parentheses altogether. If the delighted VALUE is not a string and not nil, we do nothing." (when (string-or-null-p value) (let* ((menu-keymap mode-line-mode-menu) (menu-item (assq mode (cdr menu-keymap)))) (when menu-item ;; Lighter text is typically prefixed with a space to separate ;; it from the preceding lighter. We need to trim that space. (let* ((trimmed-value (if (and value (string-match "\\`\\s-+" value)) (replace-match "" t t value) value)) (wrapped-value (if (> (length trimmed-value) 0) (concat " (" trimmed-value ")") "")) (menu-def (cdr menu-item)) (label (cadr menu-def)) (new-label (and (stringp label) (or (string-match "\\s-+(.+?)\\s-*\\'" label) (string-match "\\s-*\\'" label)) (replace-match wrapped-value t t label)))) (when new-label ;; Pure storage is used for the default menu items, so we ;; cannot modify those objects directly. (setq menu-def (copy-sequence menu-def)) (setf (cadr menu-def) new-label) (define-key menu-keymap (vector mode) menu-def))))))) ;; Handle major modes at call time. (add-hook 'after-change-major-mode-hook #'delight-major-mode) (defun delight-major-mode () "Delight the 'pretty name' of the current buffer's major mode when displayed in the mode line. When `mode-name' is displayed in other contexts (such as in the `describe-mode' help buffer), its original value will be used, unless `delight-mode-name-inhibit' is bound and nil." (let ((major-delight (assq major-mode delight-delighted-modes))) (when major-delight (setq mode-name `(delight-mode-name-inhibit ,mode-name ;; glum ,(cadr major-delight)))))) ;; delighted (define-obsolete-variable-alias 'inhibit-mode-name-delight 'delight-mode-name-inhibit "delight-1.6") (define-obsolete-variable-alias 'delight--inhibit 'delight-mode-name-inhibit "delight-1.7") (makunbound 'delight-mode-name-inhibit) ;; We explicitly call `makunbound' first because our `delight-unload-function' ;; workaround for dealing with any remaining delighted `mode-name' values is ;; simply to redefine `delight-mode-name-inhibit' with a non-nil default value. (defvar delight-mode-name-inhibit) ;; This variable determines whether the `mode-name' set by `delight-major-mode' ;; will render as the original name or the delighted name. For the purposes of ;; mode line formatting, void and nil are equivalent. It is void by default so ;; that we are able to respect any binding made by external code, and only ;; let-bind it ourselves if no such external binding exists. ;; ;; Note that if this were bound to nil by default, `delight--format-mode-line' ;; would be unable to recognise a nil binding made by some other library; and ;; if it were bound to a non-nil value by default, then we would render the ;; wrong value in the mode line. (put 'delight-mode-name-inhibit 'variable-documentation "Whether to display the original `mode-name' of a delighted major mode. A non-nil value means that the original mode name will be displayed instead of the delighted name. If nil or void, then the delighted mode name will be displayed. With the exception of Emacs' standard mode line rendering, anything rendering a mode line construct (for instance the `describe-mode' help buffer) will call `format-mode-line'. Normally we want to display delighted major mode names only in the mode line itself, and not in other contexts, and so this variable is used to inhibit the delighted names during `format-mode-line' calls. However, certain libraries may call `format-mode-line' for the purpose of replacing the standard mode line entirely, in which case we DO want to see the delighted major mode names during those particular `format-mode-line' calls. This variable is normally void, and bound to t during calls to `format-mode-line'. If, however, it is already bound, then its value will be respected; therefore binding `delight-mode-name-inhibit' to nil around a call to `format-mode-line' will allow the delighted name to be rendered. See also `delight--format-mode-line'.") (defun delight--format-mode-line (orig-fun &rest args) "Advice for `format-mode-line'. Delighted major modes should exhibit their original `mode-name' when `format-mode-line' is called. See `delight-major-mode' as well as `delight-mode-name-inhibit'." (let ((delight-mode-name-inhibit (if (boundp 'delight-mode-name-inhibit) delight-mode-name-inhibit t))) (apply orig-fun args))) (advice-add 'format-mode-line :around #'delight--format-mode-line) (defun delight-unload-function () "Handler for `unload-feature'." (condition-case err (progn (defvar unload-function-defs-list) ;; Remove hook. (remove-hook 'after-change-major-mode-hook #'delight-major-mode) ;; Remove advice. (advice-remove 'format-mode-line #'delight--format-mode-line) ;; Revert the `mode-name' changes (for the normal/expected cases). ;; We're not concerned with reversing ALL changes made, but we make ;; the effort for `mode-name' as it might prevent conflicts with ;; code which wasn't expecting a non-string mode line construct as ;; a value (e.g. Emacs bug 2034). (dolist (buf (buffer-list)) (with-current-buffer buf (when (and (consp mode-name) (symbolp (car mode-name)) (eq (indirect-variable (car mode-name)) 'delight-mode-name-inhibit)) (setq mode-name (cadr mode-name))))) ;; We keep `delight-mode-name-inhibit' around (with delighted values ;; permanently inhibited) for any unexpected cases (e.g. where our ;; modified `mode-name' was further manipulated by something else, ;; and no longer matched the format expected above). (defconst delight-mode-name-inhibit t) (dolist (var '(delight-mode-name-inhibit ;; and its aliases delight--inhibit inhibit-mode-name-delight)) (setq unload-function-defs-list (delq var unload-function-defs-list))) ;; Return nil if unloading was successful. Refer to `unload-feature'. nil) ;; If any error occurred, return non-nil. (error (progn (message "Error unloading delight: %S %S" (car err) (cdr err)) t)))) ;; Live upgrades, for when a newer version is loaded over an older one. (when (version< delight-version delight--latest-version) ;; Perform each update in sequence, as necessary. ;; Update to version 1.6 from earlier versions: (when (version< delight-version "1.6") ;; Old advice was replaced by nadvice. (eval-and-compile (require 'advice)) ;; Both macros and functions. (declare-function ad-find-advice "advice") (declare-function ad-remove-advice "advice") (declare-function ad-activate "advice") (when (ad-find-advice 'format-mode-line 'around 'delighted-modes-are-glum) (ad-remove-advice 'format-mode-line 'around 'delighted-modes-are-glum) (ad-activate 'format-mode-line))) ;; Update to version 1.N: ;; (when (version< delight-version "1.N") ...) ;; ;; All updates completed. (setq delight-version delight--latest-version)) ;; Local Variables: ;; indent-tabs-mode: nil ;; ispell-check-comments: exclusive ;; End: ;;;; ChangeLog: ;; 2020-07-11 Phil Sainty ;; ;; Merge commit '5a0cd5ccb650d7bba1c1ea02cf67b71d7cfa6e9a' from delight ;; ;; 2019-07-23 Stefan Monnier ;; ;; * packages/delight/delight.el: Use cl-lib and nadvice. ;; ;; (delight--inhibit): Rename from inhibit-mode-name-delight to clean up ;; namespace use. ;; (delight--format-mode-line): New function, extracted from the old ;; defadvice. ;; (format-mode-line): Replace defadvice with advice-add. ;; ;; 2016-07-14 Phil Sainty ;; ;; Use GNU ELPA version number formatting ;; ;; 2016-07-13 Stefan Monnier ;; ;; * delight.el: Fix copyright ;; ;; 2016-07-14 Phil Sainty ;; ;; Add 'packages/delight/' from commit ;; 'cd037ed41ae29dda89e36ff2ac8637aea96acded' ;; ;; git-subtree-dir: packages/delight git-subtree-mainline: ;; a1cdea05e8cbfe15ba075c64417db20b814e48e8 git-subtree-split: ;; cd037ed41ae29dda89e36ff2ac8637aea96acded ;; (provide 'delight) ;;; delight.el ends here