3038 lines
170 KiB
EmacsLisp
3038 lines
170 KiB
EmacsLisp
;;; help-fns+.el --- Extensions to `help-fns.el'.
|
|
;;
|
|
;; Filename: help-fns+.el
|
|
;; Description: Extensions to `help-fns.el'.
|
|
;; Author: Drew Adams
|
|
;; Maintainer: Drew Adams (concat "drew.adams" "@" "oracle" ".com")
|
|
;; Copyright (C) 2007-2018, Drew Adams, all rights reserved.
|
|
;; Created: Sat Sep 01 11:01:42 2007
|
|
;; Version: 0
|
|
;; Package-Requires: ()
|
|
;; Last-Updated: Thu May 10 15:48:56 2018 (-0700)
|
|
;; By: dradams
|
|
;; Update #: 2428
|
|
;; URL: https://www.emacswiki.org/emacs/download/help-fns%2b.el
|
|
;; Doc URL: https://emacswiki.org/emacs/HelpPlus
|
|
;; Keywords: help, faces, characters, packages, description
|
|
;; Compatibility: GNU Emacs: 22.x, 23.x, 24.x, 25.x, 26.x
|
|
;;
|
|
;; Features that might be required by this library:
|
|
;;
|
|
;; `button', `cl', `cl-lib', `gv', `help-fns', `help-mode', `info',
|
|
;; `macroexp', `naked', `radix-tree', `wid-edit', `wid-edit+'.
|
|
;;
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
;;
|
|
;;; Commentary:
|
|
;;
|
|
;; Extensions to `help-fns.el'. Also includes a redefinition of
|
|
;; `describe-face', which is from `faces.el'.
|
|
;;
|
|
;; Note: As of Emacs 24.4, byte-compiling this file in one Emacs
|
|
;; version and using the compiled file in another Emacs version
|
|
;; does not work.
|
|
;;
|
|
;;
|
|
;; Keys bound here:
|
|
;;
|
|
;; `C-h B' `describe-buffer'
|
|
;; `C-h c' `describe-command' (replaces `describe-key-briefly')
|
|
;; `C-h o' `describe-option'
|
|
;; `C-h C-c' `describe-key-briefly' (replaces `C-h c')
|
|
;; `C-h C-o' `describe-option-of-type'
|
|
;; `C-h M-c' `describe-copying' (replaces `C-h C-c')
|
|
;; `C-h M-f' `describe-file'
|
|
;; `C-h M-k' `describe-keymap'
|
|
;; `C-h M-l' `find-function-on-key'
|
|
;;
|
|
;; Commands defined here:
|
|
;;
|
|
;; `describe-buffer', `describe-command', `describe-file',
|
|
;; `describe-keymap', `describe-option', `describe-option-of-type'.
|
|
;;
|
|
;; User options defined here:
|
|
;;
|
|
;; `help-cross-reference-manuals' (Emacs 23.2+).
|
|
;;
|
|
;; Faces defined here:
|
|
;;
|
|
;; `describe-variable-value' (Emacs 24+).
|
|
;;
|
|
;; Non-interactive functions defined here:
|
|
;;
|
|
;; `describe-mode-1', `help-all-exif-data',
|
|
;; `help-commands-to-key-buttons', `help-custom-type',
|
|
;; `help-documentation', `help-documentation-property' (Emacs 23+),
|
|
;; `help-key-button-string', `help-remove-duplicates',
|
|
;; `help-substitute-command-keys', `help-value-satisfies-type-p',
|
|
;; `help-var-inherits-type-p', `help-var-is-of-type-p',
|
|
;; `help-var-matches-type-p', `help-var-val-satisfies-type-p',
|
|
;; `Info-first-index-occurrence' (Emacs 23.2+),
|
|
;; `Info-indexed-find-file' (Emacs 23.2+), `Info-indexed-find-node'
|
|
;; (Emacs 23.2+), `Info-index-entries-across-manuals' (Emacs
|
|
;; 23.2+), `Info-index-occurrences' (Emacs 23.2+),
|
|
;; `Info-make-manuals-xref' (Emacs 23.2+).
|
|
;;
|
|
;; Internal variables defined here:
|
|
;;
|
|
;; `Info-indexed-file' (Emacs 23.2+), `Info-indexed-nodes' (Emacs
|
|
;; 23.2+), `variable-name-history'.
|
|
;;
|
|
;;
|
|
;; ***** NOTE: The following command defined in `faces.el'
|
|
;; has been REDEFINED HERE:
|
|
;;
|
|
;; `describe-face'.
|
|
;;
|
|
;;
|
|
;; ***** NOTE: The following command defined in `help.el'
|
|
;; has been REDEFINED HERE:
|
|
;;
|
|
;; `describe-mode'.
|
|
;;
|
|
;;
|
|
;; ***** NOTE: The following functions defined in `help-fns.el'
|
|
;; have been REDEFINED HERE:
|
|
;;
|
|
;; `describe-function', `describe-function-1', `describe-variable',
|
|
;; `help-fns--key-bindings', `help-fns--signature'.
|
|
;;
|
|
;;
|
|
;; ***** NOTE: The following command defined in `package.el'
|
|
;; has been REDEFINED HERE:
|
|
;;
|
|
;; `describe-package'.
|
|
;;
|
|
;;
|
|
;; Put this in your initialization file (`~/.emacs'):
|
|
;;
|
|
;; (require 'help-fns+)
|
|
;;
|
|
;; Acknowledgement: Passing text properties on doc strings to the
|
|
;; *Help* buffer is an idea from Johan bockgard. He sent it on
|
|
;; 2007-01-24 to emacs-devel@gnu.org, Subject
|
|
;; "display-completion-list should not strip text properties".
|
|
;;
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
;;
|
|
;;; Change Log:
|
|
;;
|
|
;; 2018/05/10 dadams
|
|
;; describe-face: Correct face-at-point call for Emacs 23.
|
|
;; 2018/04/15 dadams
|
|
;; help-fns--signature:
|
|
;; Use substitute-command-keys with BUFFER current.
|
|
;; Corrected order (swap): Use plain cons if _not_ BUFFER.
|
|
;; Made BUFFER arg mandatory - just use nil for older calls.
|
|
;; 2018/04/08 dadams
|
|
;; Updated for Emacs 26-27:
|
|
;; describe-function:
|
|
;; Use (get f 'function-documentation) as alternative for fboundp.
|
|
;; Use help--symbol-completion-table as alternative for obarray.
|
|
;; Use (get function 'reader-construct) as alternative for printing.
|
|
;; help-fns--key-bindings: Use format-message if defined.
|
|
;; help-fns--signature: Do not insert high-usage if (get function 'reader-construct).
|
|
;; describe-function-1, Emacs 24.3-25:
|
|
;; Added void-function error.
|
|
;; Advised & aliased function: Ensure not an autoload.
|
|
;; find-lisp-object-file-name: Pass defun (symbol), if aliased.
|
|
;; Use format-message if defined.
|
|
;; Use substitute-command-keys, for curly-quote handling.
|
|
;; Pass buffer to help-fns--signature. For Emacs 24, pass DOC-RAW and SIG-KEY, like for Emacs 25.
|
|
;; Added pure-function indication at end.
|
|
;; If not straight or grave text-quoting-style then set coding system to utf8.
|
|
;; describe-function-1, Emacs 26-27:
|
|
;; Add help buttons and manual links to doc. Use string-match-p, not string-match.
|
|
;; 2016/09/17 dadams
|
|
;; describe-function: Fix Emacs bug #24221: let FUNCTION be anonymous.
|
|
;; 2015/12/15 dadams
|
|
;; describe-file: Remove `' around file name in title.
|
|
;; 2015/09/09 dadams
|
|
;; describe-variable: Fixed test order for non-"" VARDOC, so it does not become t.
|
|
;; 2015/09/08 dadams
|
|
;; describe-keymap: Added optional arg SEARCH-SYMBOLS-P. Follow alias chain of symbol, and describe last one.
|
|
;; describe-variable: Pick up doc from alias, if help-documentation-property returns "".
|
|
;; 2015/08/30 dadams
|
|
;; describe-function-1: Typo: auto-do-load -> autoload-do-load.
|
|
;; 2015/08/22 dadams
|
|
;; describe-keymap: Allow arg to be a keymap (not a keymap variable), when not interactive. Suggestion by erjoalgo.
|
|
;; 2015/08/13 dadams
|
|
;; describe-variable:
|
|
;; PREDICATE arg to completing-read needs to use original buffer, not minibuffer, when test boundp.
|
|
;; Fixes Emacs BUG #21252.
|
|
;; 2015/08/02 dadams
|
|
;; Updated for Emacs 25
|
|
;; help-fns--signature:
|
|
;; Added arg RAW. Return DOC if FUNCTION is a keymap. Use help--make-usage-docstring.
|
|
;; Use help--docstring-quote. Insert "`X", not "(\` X)", when documenting `X. Use substitute-command-keys
|
|
;; on args to help-highlight-arguments.
|
|
;; describe-function-1:
|
|
;; Use indirect-function if subr (SIG-KEY). Moved autoloads forward). Use help-fns-short-filename.
|
|
;; Use auto-do-load. But do NOT use curly quotes - e.g., no extra substitute-command-name calls.
|
|
;; 2015/04/03 dadams
|
|
;; Use char-before in place of looking-back, for chars before. See Emacs bug #17284.
|
|
;; 2015/03/26 dadams
|
|
;; describe-package: Fix guard to use emacs-minor-version 3, not 24. Thx to Roshan Shariff.
|
|
;; 2015/03/23 dadams
|
|
;; describe-variable (Emacs 24+): Fix terpri's so appearance is better. Fill region for global value.
|
|
;; 2014/11/29 dadams
|
|
;; Info-make-manuals-xref: Control number of newlines before.
|
|
;; describe-function-1: Use same def for Emacs 25.
|
|
;; describe-variable-value: Changed the default colors.
|
|
;; describe-variable: Use face describe-variable-value always. Fill region for value always.
|
|
;; Control number of newlines before and after Value:, and after manuals xref.
|
|
;; 2014/11/12 dadams
|
|
;; describe-package: Added version for Emacs 24.4+ - Use package-alist, package--builtins, or package-archive-contents.
|
|
;; 2014/11/08 dadams
|
|
;; describe-mode-1: Show major-mode and mode-function also, on a separate line (Emacs bug #18992), filling.
|
|
;; 2014/08/10 dadams
|
|
;; describe-command: Bind completion-annotate-function for use with Icicles.
|
|
;; 2014/05/11 dadams
|
|
;; help-substitute-command-keys: Bug: \= was not being removed - C-h f replace-regexp showed \=\N, not \N.
|
|
;; Small loop for \=: changed \\\\=$ to \\\\=.
|
|
;; Main loop, when escaped (\=) and odd: Skip the \=: concat before \= with after \=.
|
|
;; 2014/05/04 dadams
|
|
;; Use called-interactively only for Emacs 23.2+, since we pass it an arg.
|
|
;; 2014/05/02 dadams
|
|
;; describe-package: Updated for Emacs 24.4 - defstruct package-desc.
|
|
;; 2014/04/21 dadams
|
|
;; with-selected-frame: Updated for Emacs 24.4.
|
|
;; describe-face: Updated for Emacs 24.4: Try face-at-point for read-face-name default.
|
|
;; describe-file, describe-keymap, describe-function:
|
|
;; Updated for Emacs 24.4: Use with-help-window, not with-output-to-temp-buffer. See bug #17109.
|
|
;; describe-function-1: Created version for Emacs 24.4+
|
|
;; help-key-button-string: Do not quote :type.
|
|
;; describe-buffer, describe-mode-1, describe-function: Use called-interactively, if available.
|
|
;; Removed autoload cookie for describe-function, describe-keymap (but why?).
|
|
;; 2014/03/06 dadams
|
|
;; describe-variable: Fixed typo in regexp: [n] -> [\n].
|
|
;; 2014/01/04 dadams
|
|
;; Added: describe-variable-value.
|
|
;; describe-variable (Emacs 24+): Highlight the value with face describe-variable-value.
|
|
;; 2013/08/06 dadams
|
|
;; describe-function: Ensure arg is a defined function before calling describe-function-1 (for Emacs 24+).
|
|
;; 2013/07/01 dadams
|
|
;; Revert the filling part of yesterday's update.
|
|
;; 2013/06/30 dadams
|
|
;; describe-variable for Emacs 24+:
|
|
;; Update for vanilla Emacs 24.4. Update for Emacs bug #14754: fill printed value so no long lines.
|
|
;; 2013/06/16 dadams
|
|
;; describe-(variable|option(-of-type)): Fixed for dumb variable-at-point, which returns 0 for no var.
|
|
;; 2013/04/29 dadams
|
|
;; describe-(function|command|variable|option|option-of-type):
|
|
;; Provide default only if symbol is of the right type. Put default in prompt.
|
|
;; 2013/02/08 dadams
|
|
;; describe-variable: Updated wrt Emacs 24 build of 2013-01-30.
|
|
;; 2012/11/18 dadams
|
|
;; describe-(variable|function): Add completion-candidate annotation: (option|comand).
|
|
;; 2012/10/28 dadams
|
|
;; help-fns--key-bindings: Fixed: forgot to mapconcat over keys.
|
|
;; 2012/10/26 dadams
|
|
;; Added: help-fns--key-bindings, help-fns--signature,
|
|
;; Added Emacs 24.3+ version of describe-function-1. Updated version for 23.2-24.2.
|
|
;; help-substitute-command-keys: Fix for \= when no match for \[, \<, \{ past it.
|
|
;; 2012/09/24 dadams
|
|
;; describe-file: Added optional arg NO-ERROR-P.
|
|
;; 2012/09/22 dadams
|
|
;; Info-index-occurrences, Info-first-index-occurrence: Replace Info-directory call by short version. Better Searching msg.
|
|
;; 2012/09/21 dadams
|
|
;; Renamed Info-any-index-occurrences-p to Info-first-index-occurrence.
|
|
;; Info-any-index-occurrences-p: Return the first successful lookup, not t.
|
|
;; Info-index-entries-across-manuals, Info-index-occurrences, Info-any-index-occurrences-p:
|
|
;; Added optional arg INDEX-NODES.
|
|
;; Adjust calls to those fns accordingly, e.g., in define-button-type for help-info-manual-lookup
|
|
;; and help-insert-xref-button in Info-make-manuals-xref.
|
|
;; 2012/07/20 dadams
|
|
;; Added: describe-buffer, describe-mode-1. Bound describe-buffer to C-h B.
|
|
;; describe-mode: Redefined to use describe-mode-1.
|
|
;; 2012/07/03 dadams
|
|
;; Info-make-manuals-xref, Info-index-entries-across-manuals, Info-index-occurrences,
|
|
;; Info-any-index-occurrences-p:
|
|
;; Added optional arg NOMSG.
|
|
;; describe-(function|variable|file|package): No message if not interactive-p.
|
|
;; describe-function-1: pass MSGP to Info-make-manuals-xref (i.e. msg always).
|
|
;; describe-(mode|variable|face|keymap|package): Pass proper NOMSG arg to Info-make-manuals-xref.
|
|
;; 2012/01/11 dadams
|
|
;; describe-variable: Remove * from beginning of doc string.
|
|
;; 2011/11/25 dadams
|
|
;; Reverted yesterday's change and added IMPORTANT note to Commentary.
|
|
;; 2011/11/24 dadams
|
|
;; Added Emacs 24 version of with-help-window. They changed the signature of help-window-setup.
|
|
;; 2011/10/14 dadams
|
|
;; describe-mode: Call help-documentation while in mode's buffer, in case no \\<...>.
|
|
;; 2011/10/08 dadams
|
|
;; Info-make-manuals-xref: Do nothing if OBJECT is not a string or a symbol (e.g. is a keymap).
|
|
;; 2011/10/07 dadams
|
|
;; Added soft require of naked.el.
|
|
;; help-substitute-command-keys, describe-function-1: Use naked-key-description if available.
|
|
;; 2011/08/22 dadams
|
|
;; describe-variable (Emacs 23+): Added terpri after Value: (for multiline value).
|
|
;; 2011/07/25 dadams
|
|
;; describe-mode: Put call to help-documentation inside let for maj: else major-mode gets changed to help-mode.
|
|
;; 2011/06/26 dadams
|
|
;; Added: help-commands-to-key-buttons, help-documentation(-property),
|
|
;; help-key-button-string, help-substitute-command-keys (Emacs 23+).
|
|
;; describe-(mode|variable|function-1|keymap) for Emacs 23+:
|
|
;; Use help-documentation (with insert and button arg), instead of documentation (with princ).
|
|
;; 2011/06/22 dadams
|
|
;; Info-make-manuals-xref: Added optional arg MANUALS.
|
|
;; 2011/06/20 dadams
|
|
;; Info(-any)-index-occurrences(-p): Fix pattern: remove arbitrary prefix [^\n]*.
|
|
;; Added, for Emacs 24+: describe-package.
|
|
;; 2011/06/14 dadams
|
|
;; Added, for Emacs 23.2+: describe-mode.
|
|
;; Info-make-manuals-xref: Added optional arg NO-NEWLINES-AFTER-P.
|
|
;; 2011/06/13 dadams
|
|
;; Added: Info-any-index-occurrences-p.
|
|
;; Info-make-manuals-xref: Use Info-any-index-occurrences-p, not Info-index-occurrences.
|
|
;; 2011/06/11 dadams
|
|
;; Added, for Emacs 23.2+:
|
|
;; describe-face, describe-function-1, help-cross-reference-manuals, Info-indexed-find-file,
|
|
;; Info-indexed-find-node, Info-index-entries-across-manuals, Info-index-occurrences,
|
|
;; Info-make-manuals-xref, Info-indexed-file, Info-indexed-nodes.
|
|
;; describe-keymap: Emacs 23.2+: Added link to manuals.
|
|
;; describe-variable: Updated Emacs 23 version, per vanilla.
|
|
;; Emacs 23.2+: Added link to manuals.
|
|
;; Require info.el for Emacs 23.2+.
|
|
;; 2011/04/25 dadams
|
|
;; describe-file: Incorporate autofile bookmark description. Added optional arg.
|
|
;; 2011/03/31 dadams
|
|
;; help-var-(matches|inherits)-type-p: Wrap string-match with save-match-data.
|
|
;; 2011/03/17 dadams
|
|
;; describe-file: Added clickable thumbnail image to the help for an image file.
|
|
;; 2011/03/02 dadams
|
|
;; Added: help-all-exif-data.
|
|
;; describe-file: Show all EXIF data, using help-all-exif-data.
|
|
;; 2011/02/22 dadams
|
|
;; describe-file: Show also EXIF data for an image file.
|
|
;; 2011/01/04 dadams
|
|
;; Removed autoload cookies from non def* sexps and define-key.
|
|
;; 2010/02/12 dadams
|
|
;; Added variable-name-history.
|
|
;; 2009/08/30 dadams
|
|
;; describe-keymap: Don't print nil if the map has no doc.
|
|
;; 2009/05/26 dadams
|
|
;; describe-variable: Updated wrt latest Emacs 23: Added file-name-non-directory; removed substitute-command-keys.
|
|
;; 2008/09/13 dadams
|
|
;; Updated for latest Emacs 23 CVS.
|
|
;; describe-variable: Create separate version for Emacs 23.
|
|
;; describe-function-1: No longer needed for Emacs 23, since my patch added.
|
|
;; Added: with-selected-frame, with-help-window, at least temporarily.
|
|
;; Require wid-edit.el.
|
|
;; 2008/09/02 dadams
|
|
;; describe-function-1, describe-variable: Emacs 23 uses find-lisp-object-file-name. Thx to Per Nordlow.
|
|
;; 2008/08/19 dadams
|
|
;; describe-keymap: Use insert instead of princ for map part. Thx to Chong Yidong.
|
|
;; 2008/05/20 dadams
|
|
;; describe-function: Different prompt if prefix arg.
|
|
;; 2008/03/02 dadams
|
|
;; Moved describe-file here from misc-cmds.el. Bound to C-h M-f.
|
|
;; Require cl.el at compile time.
|
|
;; 2008/02/01 dadams
|
|
;; Bound M-l to find-function-on-key.
|
|
;; 2008/01/03 dadams
|
|
;; Added: describe-function-1. The redefinition fills overlong lines.
|
|
;; 2007/12/25 dadams
|
|
;; help-var-inherits-type-p:
|
|
;; Recheck var-type match after set var-type to its car.
|
|
;; Handle string (regexp) TYPES elements.
|
|
;; help-value-satisfies-type-p: Skip type check for string type (regexp).
|
|
;; help-var-is-of-type-p: Doc string. Use help-var-matches-type-p.
|
|
;; Added: help-var-matches-type-p.
|
|
;; 2007/12/24 dadams
|
|
;; help-var-inherits-type-p: Recheck type match after set var-type to its car.
|
|
;; Added: help-custom-type.
|
|
;; 2007/12/23 dadams
|
|
;; help-var-is-of-type-p:
|
|
;; Added MODE arg. Use help-var-inherits-type-p, help-var-val-satisfies-type-p.
|
|
;; Redefined as MODE choice, not just a simple or. Treat more cases.
|
|
;; Added: help-var-inherits-type-p, help-var-val-satisfies-type-p,
|
|
;; help-value-satisfies-type-p.
|
|
;; describe-option-of-type: Prefix arg means use mode inherit-or-value.
|
|
;; 2007/12/22 dadams
|
|
;; help-var-is-of-type-p:
|
|
;; Check supertypes also. Use both :validate and :match.
|
|
;; Wrap type check in condition-case. Use widget-put instead of plist-put.
|
|
;; Added soft require of wid-edit+.el.
|
|
;; 2007/12/21 dadams
|
|
;; help-var-is-of-type-p: Use :validate, not :match, for the test.
|
|
;; 2007/12/20 dadams
|
|
;; Moved describe-option-of-type to C-h C-o.
|
|
;; 2007/12/15 dadams
|
|
;; Bound C-h c to describe-command and C-h C-c to describe-key-briefly.
|
|
;; 2007/12/07 dadams
|
|
;; describe-option-of-type:
|
|
;; Call describe-variable with nil buffer. Use "nil" as default value.
|
|
;; 2007/12/06 dadams
|
|
;; describe-option-of-type:
|
|
;; If nil type, all defcustom vars are candidates. Use custom-variable-p.
|
|
;; Specific error if no such custom type.
|
|
;; 2007/12/04 dadams
|
|
;; Added: describe-option-of-type, help-remove-duplicates, help-var-is-of-type-p.
|
|
;; Bound o to describe-option, M-o to describe-option-of-type, C-c to describe-command, M-c to describe-copying.
|
|
;; 2007/11/28 dadams
|
|
;; Renamed describe-bindings-in-map to describe-keymap. Added keymap's doc string.
|
|
;; 2007/11/22 dadams
|
|
;; Added: describe-bindings-in-map. Bound to C-h M-k.
|
|
;; 2007/11/01 dadams
|
|
;; Corrected require typo: help-mode -> help-fns.
|
|
;; 2007/10/18 dadams
|
|
;; Created.
|
|
;;
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
;;
|
|
;; 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 2, 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; see the file COPYING. If not, write to
|
|
;; the Free Software Foundation, Inc., 51 Franklin Street, Fifth
|
|
;; Floor, Boston, MA 02110-1301, USA.
|
|
;;
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
;;
|
|
;;; Code:
|
|
|
|
|
|
(require 'help-fns)
|
|
|
|
(require 'wid-edit+ nil t) ;; (no error if not found):
|
|
;; redefined color widget (for help-var-is-of-type-p)
|
|
(require 'wid-edit) ;; widget-convert
|
|
|
|
(require 'naked nil t) ;; (no error if not found): naked-key-description
|
|
|
|
(when (or (> emacs-major-version 23) (and (= emacs-major-version 23) (> emacs-minor-version 1)))
|
|
(require 'info)) ;; Info-virtual-files
|
|
|
|
(eval-when-compile (require 'cl)) ;; case, gentemp
|
|
|
|
|
|
;; Quiet the byte-compiler.
|
|
(defvar advertised-signature-table)
|
|
(defvar dir-local-variables-alist)
|
|
(defvar dir-locals-file)
|
|
(defvar file-local-variables-alist)
|
|
(defvar icicle-mode) ; In `icicles-mode.el'
|
|
(defvar icicle-pre-minibuffer-buffer) ; In `icicles-var.el'
|
|
(defvar icicle-WYSIWYG-Completions-flag) ; In `icicles-opt.el'
|
|
(defvar Info-indexed-nodes) ; In `info.el'
|
|
(defvar help-cross-reference-manuals) ; For Emacs < 23.2
|
|
(defvar help-enable-auto-load) ; For Emacs < 24.3
|
|
(defvar package-alist)
|
|
(defvar package-archive-contents)
|
|
(defvar package--builtins)
|
|
(defvar package--initialized)
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
(defvar variable-name-history () "Minibuffer history for variable names.")
|
|
|
|
(define-key help-map "B" 'describe-buffer)
|
|
(define-key help-map "c" 'describe-command)
|
|
(define-key help-map "o" 'describe-option)
|
|
(define-key help-map "\C-c" 'describe-key-briefly)
|
|
(define-key help-map "\C-o" 'describe-option-of-type)
|
|
(define-key help-map "\M-c" 'describe-copying)
|
|
(define-key help-map "\M-f" 'describe-file)
|
|
(define-key help-map "\M-k" 'describe-keymap)
|
|
(define-key help-map "\M-l" 'find-function-on-key)
|
|
|
|
|
|
;; Need Emacs 23 for version of `make-text-button' that accepts a string.
|
|
(when (> emacs-major-version 22)
|
|
|
|
(defun help-documentation (function &optional raw add-help-buttons)
|
|
"Same as `documentation', but optionally adds buttons for help.
|
|
Non-nil optional arg ADD-HELP-BUTTONS does that, adding buttons to key
|
|
descriptions, which link to the key's command help."
|
|
(let ((raw-doc (documentation function 'RAW)))
|
|
(if raw raw-doc (help-substitute-command-keys raw-doc add-help-buttons))))
|
|
|
|
(defun help-documentation-property (symbol prop &optional raw add-help-buttons)
|
|
"Same as `documentation-property', but optionally adds buttons for help.
|
|
Non-nil optional arg ADD-HELP-BUTTONS does that, adding buttons to key
|
|
descriptions, which link to the key's command help."
|
|
(let ((raw-doc (documentation-property symbol prop 'RAW)))
|
|
(if raw raw-doc (help-substitute-command-keys raw-doc add-help-buttons))))
|
|
|
|
(defun help-commands-to-key-buttons (string)
|
|
"Like `substitute-command-keys', but adds buttons for help on keys.
|
|
Key descriptions become links to help about their commands."
|
|
(help-substitute-command-keys string 'ADD-HELP-BUTTONS))
|
|
|
|
(defun help-substitute-command-keys (string &optional add-help-buttons)
|
|
"Same as `substitute-command-keys', but optionally adds buttons for help.
|
|
Non-nil optional arg ADD-HELP-BUTTONS does that, adding buttons to key
|
|
descriptions, which link to the key's command help."
|
|
|
|
;; REPEAT:
|
|
;; Search for first occurrence of any of the patterns: \[...], \{...}, or \<...>.
|
|
;; Handle escaping via \=, if present before the pattern or if there is no pattern match.
|
|
;; If pattern is a keymap (\<...>): use it from then on.
|
|
;; If pattern is a command (\[...]): (a) substitute its key description, (b) put a button on it.
|
|
;; If pattern is a bindings spec (\{...}): just substitute the usual text.
|
|
(with-syntax-table emacs-lisp-mode-syntax-table
|
|
(let* ((strg (copy-sequence string))
|
|
(len-strg (length strg))
|
|
(ii 0)
|
|
(jj 0)
|
|
(newstrg "")
|
|
(re-command "\\\\\\[\\(\\(\\sw\\|\\s_\\)+\\)\\]")
|
|
(re-keymap "\\\\<\\(\\(\\sw\\|\\s_\\)+\\)>")
|
|
(re-bindings "\\\\{\\(\\(\\sw\\|\\s_\\)+\\)}")
|
|
(re-any (concat "\\(" re-command "\\|" re-keymap "\\|" re-bindings "\\)"))
|
|
(keymap (or overriding-terminal-local-map overriding-local-map))
|
|
(msg nil)
|
|
key bindings ma mc mk mb)
|
|
(while (< ii len-strg)
|
|
(setq key nil
|
|
bindings ()
|
|
strg (substring strg ii))
|
|
|
|
(save-match-data ; ANY
|
|
(setq ma (string-match re-any strg))
|
|
(cond ((not ma) ; No \[...], \{...}, or \<...>, but we need to handle \=
|
|
(setq jj 0
|
|
newstrg (concat newstrg (replace-regexp-in-string "\\\\=\\(.\\)" "\\1" strg nil nil nil jj)))
|
|
(when (match-beginning 1) (setq jj (match-beginning 1)))
|
|
(setq ii len-strg))
|
|
(t
|
|
(let ((escaped nil)
|
|
(odd nil))
|
|
(save-match-data
|
|
(let ((ma/= ma))
|
|
(setq ii ma)
|
|
(while (string-match "\\\\=" (substring strg 0 ma/=))
|
|
(setq odd (not odd)
|
|
ma/= (match-beginning 0))
|
|
(when odd (setq ii (- ii 2)
|
|
escaped ma/=)))))
|
|
(if (not escaped)
|
|
(setq ii ma
|
|
jj (match-end 0)
|
|
ma (match-string-no-properties 0 strg)
|
|
newstrg (concat newstrg (substring strg 0 ii)))
|
|
(setq jj (match-end 0) ; End of \[...], \{...}, or \<...>
|
|
ma (and (not odd) (match-string-no-properties 0 strg))
|
|
newstrg (if odd
|
|
(concat newstrg
|
|
(substring strg 0 escaped) ; Before \='s
|
|
(substring strg (+ 2 escaped) ii)) ; After \='s
|
|
(concat newstrg (substring strg 0 ii)))))))))
|
|
|
|
(when ma
|
|
|
|
(save-match-data ; KEYMAP
|
|
(setq ma (copy-sequence ma)
|
|
mk (string-match re-keymap ma)
|
|
mk (and mk (match-string-no-properties 0 ma)))
|
|
(when mk
|
|
(setq keymap (intern (match-string-no-properties 1 ma)))
|
|
(if (boundp keymap)
|
|
(setq keymap (symbol-value keymap))
|
|
(setq msg (format "\nUses keymap \"%s\", which is not currently defined.\n" keymap)
|
|
keymap (or overriding-terminal-local-map overriding-local-map)))))
|
|
|
|
(unless mk ; COMMAND
|
|
(save-match-data
|
|
(setq ma (copy-sequence ma)
|
|
mc (string-match re-command ma)
|
|
mc (and mc (match-string-no-properties 0 ma))
|
|
mc (and mc (intern (substring mc 2 -1)))) ; Remove \[...] envelope
|
|
(when mc
|
|
(let ((follow-remap t))
|
|
(while (and (setq key (where-is-internal mc keymap 'FIRSTONLY))
|
|
(vectorp key) (> (length key) 1) (eq 'remap (aref key 0))
|
|
(symbolp (aref key 1)) follow-remap)
|
|
(setq mc (aref key 1)
|
|
follow-remap nil)))
|
|
(setq key (if key
|
|
(if (fboundp 'naked-key-description)
|
|
(naked-key-description key)
|
|
(key-description key))
|
|
(concat "M-x " (symbol-name mc))))
|
|
(when add-help-buttons (setq key (help-key-button-string key mc))))))
|
|
|
|
(unless (or mk mc) ; BINDINGS
|
|
(save-match-data
|
|
(setq ma (copy-sequence ma)
|
|
mb (string-match re-bindings ma)
|
|
mb (and mb (match-string-no-properties 0 ma)))
|
|
(when mb
|
|
(setq bindings (intern (match-string-no-properties 1 ma)))
|
|
(cond ((boundp bindings)
|
|
(setq bindings (substitute-command-keys mb))) ; Use original - no buttons.
|
|
(t
|
|
(setq msg (format "\nUses keymap \"%s\", which is not currently defined.\n" bindings)
|
|
bindings nil))))))
|
|
|
|
(unless mk (setq newstrg (concat newstrg (or key bindings (substring strg ii jj)))))
|
|
(setq ii (or jj len-strg))))
|
|
|
|
(if (string= string newstrg)
|
|
string ; Return original string, not a copy, if no changes.
|
|
newstrg))))
|
|
|
|
(defun help-key-button-string (key-description command)
|
|
"Return a button for KEY-DESCRIPTION that links to the COMMAND description.
|
|
KEY-DESCRIPTION is a key-description string.
|
|
COMMAND is the command (a symbol) associated with the key described.
|
|
Return a copy of string KEY-DESCRIPTION with button properties added.
|
|
Clicking the button shows the help for COMMAND."
|
|
(let ((new-key (copy-sequence key-description)))
|
|
(make-text-button new-key nil 'button (list t) :type 'help-function 'help-args (list command))
|
|
new-key))
|
|
)
|
|
|
|
|
|
(when (boundp 'Info-virtual-files) ; Emacs 23.2+
|
|
|
|
(defcustom help-cross-reference-manuals '(("emacs" "elisp"))
|
|
"*Manuals to search, for a `*Help*' buffer link to the manuals.
|
|
A cons.
|
|
|
|
The car is a list of manuals to search, or the symbol `all', to
|
|
search all. If nil, then do not create a cross-reference link.
|
|
|
|
The cdr is a boolean:
|
|
|
|
Non-`nil' means search the manuals, then create a cross-ref link:
|
|
create it only if some search hits are found.
|
|
|
|
`nil' means create a cross-ref link without searching manuals
|
|
first (but only if there are some manuals to search)."
|
|
:set #'(lambda (sym defs) (custom-set-default sym defs) (setq Info-indexed-nodes ()))
|
|
:type '(cons
|
|
(choice :tag "Which Manuals"
|
|
(repeat :tag "Specific Manuals (files)" string)
|
|
(const :tag "All Manuals" all))
|
|
(boolean :tag "Search Before Creating Button?"))
|
|
:group 'help)
|
|
|
|
(defvar Info-indexed-file "*Indexed*"
|
|
"Info file for virtual manual from `Info-index-entries-across-manuals'.")
|
|
|
|
(defvar Info-indexed-nodes ()
|
|
"Alist of cached nodes with matching index entries.
|
|
Each element is (NODENAME STRING MATCHES), where:
|
|
NODENAME is the name of the node that is indexed,
|
|
STRING is the search string passed to `Info-index-occurrences',
|
|
MATCHES is a list of index matches found by `Info-index-occurrences'.
|
|
|
|
This has the same structure as `Info-apropos-nodes', but the search
|
|
was made by `Info-index-occurrences', not by `Info-apropos-matches',
|
|
so that matches are exact (ignoring case).")
|
|
|
|
(defun Info-indexed-find-file (filename &optional _noerror)
|
|
"Index-search implementation of `Info-find-file'."
|
|
filename)
|
|
|
|
(defun Info-indexed-find-node (_filename nodename &optional _no-going-back)
|
|
"Index-search implementation of `Info-find-node-2'."
|
|
(let* ((nodeinfo (assoc nodename Info-indexed-nodes))
|
|
(matches (nth 2 nodeinfo)))
|
|
(when matches
|
|
(insert (format "\n\^_\nFile: %s, Node: %s, Up: Top\n\n" Info-indexed-file nodename))
|
|
(insert "Index Matches\n")
|
|
(insert "*************\n\n")
|
|
(insert "Index entries that match `" (nth 1 nodeinfo) "':\n\n")
|
|
(insert "\0\b[index\0\b]\n")
|
|
(if (eq matches t)
|
|
(insert "No matches found.\n")
|
|
(insert "* Menu:\n\n")
|
|
(dolist (entry matches)
|
|
(insert (format "* %-38s (%s)%s.%s\n" (format "%s [%s]:" (nth 1 entry) (nth 0 entry))
|
|
(nth 0 entry) (nth 2 entry)
|
|
(if (nth 3 entry) (format " (line %s)" (nth 3 entry)) ""))))))))
|
|
(add-to-list 'Info-virtual-files '("\\`\\*Indexed\\*\\'"
|
|
(find-file . Info-indexed-find-file)
|
|
(find-node . Info-indexed-find-node)
|
|
;; (slow . t) ; $$$$$$ Useless here?
|
|
))
|
|
|
|
(defun Info-make-manuals-xref (object &optional no-newlines-after-p manuals-spec nomsg)
|
|
"Create a cross-ref link for index entries for OBJECT in manuals.
|
|
Non-`nil' optional arg NO-NEWLINES-AFTER-P means do not add two
|
|
newlines after the cross reference.
|
|
|
|
Optional arg MANUALS-SPEC controls which manuals to search. It has
|
|
the same form as option `help-cross-reference-manuals', and it
|
|
defaults to the value of that option.
|
|
|
|
Do nothing if the car of MANUALS-SPEC is nil (no manuals to search).
|
|
If its cdr is `nil' then create the link without first searching any
|
|
manuals. Otherwise, create the link only if there are search hits in
|
|
the manuals."
|
|
(when (or (stringp object) (symbolp object)) ; Exclude, e.g., a keymap as OBJECT.
|
|
(unless manuals-spec (setq manuals-spec help-cross-reference-manuals))
|
|
(when (car manuals-spec) ; Create no link if no manuals to search.
|
|
(let ((books (car manuals-spec))
|
|
(search-now-p (cdr manuals-spec))
|
|
(symb-name (if (stringp object) object (symbol-name object))))
|
|
(when (or (not search-now-p)
|
|
(save-current-buffer (Info-first-index-occurrence symb-name () books nomsg)))
|
|
(let ((buffer-read-only nil)
|
|
(nl-before (cond ((and (eq ?\n (char-before)) ; Quicker than `looking-back', apparently.
|
|
(eq ?\n (char-before (1- (point))))) "")
|
|
((eq ?\n (char-before)) "\n")
|
|
(t "\n\n"))))
|
|
(insert (format "%sFor more information %s the " nl-before (if (cdr manuals-spec) "see" "check")))
|
|
(help-insert-xref-button "manuals" 'help-info-manual-lookup symb-name () books)
|
|
(insert ".")
|
|
(unless no-newlines-after-p (insert "\n\n"))))))))
|
|
|
|
(when (and (> emacs-major-version 21)
|
|
(condition-case nil (require 'help-mode nil t) (error nil))
|
|
(get 'help-xref 'button-category-symbol)) ; In `button.el'
|
|
(define-button-type 'help-info-manual-lookup
|
|
:supertype 'help-xref
|
|
'help-function #'(lambda (string &optional index-nodes books nomsg)
|
|
(Info-index-entries-across-manuals string () books nomsg))
|
|
'help-echo "mouse-2, RET: Look it up in the manuals"))
|
|
|
|
(defun Info-index-entries-across-manuals (string &optional index-nodes manuals nomsg)
|
|
"Look up STRING in Info MANUALS on your system.
|
|
Looks for exact matches (ignoring case): STRING is expected to be an
|
|
index entry. Build an Info menu of the possible matches.
|
|
|
|
Optional arg INDEX-NODES are the index nodes in MANUALS to search.
|
|
By default (nil value), all indexes are searched.
|
|
Optional arg MANUALS is the list of manuals to search, or the symbol
|
|
`all', to search all.
|
|
Optional arg NOMSG non-nil means do not display a progress message."
|
|
(let ((nodes Info-indexed-nodes)
|
|
nodename)
|
|
(while (and nodes (not (equal string (nth 1 (car nodes))))) (setq nodes (cdr nodes)))
|
|
(if nodes
|
|
(Info-find-node Info-indexed-file (car (car nodes)))
|
|
(setq nodename (format "Index for `%s'" string))
|
|
(push (list nodename string (Info-index-occurrences string index-nodes manuals nomsg))
|
|
Info-indexed-nodes)
|
|
(Info-find-node Info-indexed-file nodename))))
|
|
|
|
;; Similar to `Info-apropos-matches', but using exact matches (ignoring case).
|
|
(defun Info-index-occurrences (index-entry &optional index-nodes manuals nomsg)
|
|
"Collect occurrences of INDEX-ENTRY in INDEX-NODES of MANUALS.
|
|
Return a list of the form ((FILE INDEX-ENTRY NODE LINE) ...), where:
|
|
FILE is the name of an Info file,
|
|
NODE is an Info node name,
|
|
LINE is the line number of the INDEX-ENTRY occurrence in that node.
|
|
|
|
Optional arg INDEX-NODES are the index nodes in MANUALS to search.
|
|
By default (nil value), search all indexes of each manual.
|
|
Optional arg MANUALS is the list of manuals to search, or the symbol
|
|
`all', to search all.
|
|
Optional arg NOMSG non-nil means do not display a progress message."
|
|
(unless (string= index-entry "")
|
|
;; Unlike `Info-apropos-matches', we match only the exact string as an index entry.
|
|
(let ((pattern (format "\n\\* +\\(%s\\):[ \t]+\\([^\n]+\\)\
|
|
\\.\\(?:[ \t\n]*(line +\\([0-9]+\\))\\)?"
|
|
(regexp-quote index-entry)))
|
|
matches node)
|
|
(unless nomsg
|
|
(message "Searching indexes of %s..."
|
|
(cond ((eq manuals 'all) "all manuals")
|
|
((null (cadr manuals)) (concat (car manuals) " manual"))
|
|
(t (concat "manuals " (mapconcat #'identity manuals ", "))))))
|
|
(condition-case nil
|
|
(with-temp-buffer
|
|
(when (eq manuals 'all) (setq manuals ()))
|
|
(Info-mode)
|
|
;; Next two lines are essentially `(Info-directory)'.
|
|
(info-initialize)
|
|
(Info-find-node-2 "dir" "top" 'NO-GOING-BACK)
|
|
(unless manuals
|
|
(goto-char (point-min))
|
|
(re-search-forward "\\* Menu: *\n" nil t)
|
|
(let (manual)
|
|
(while (re-search-forward "\\*.*: *(\\([^)]+\\))" nil t)
|
|
;; `add-to-list' ensures no dups in `manuals', so the `dolist' runs faster.
|
|
(setq manual (match-string 1))
|
|
(set-text-properties 0 (length manual) nil manual)
|
|
(add-to-list 'manuals manual))))
|
|
(dolist (manual manuals)
|
|
(unless nomsg (message "Searching indexes of manual `%s'..." manual))
|
|
(when (or index-nodes
|
|
(setq index-nodes (Info-index-nodes (Info-find-file manual))))
|
|
(Info-find-node manual (car index-nodes))
|
|
(while (progn (goto-char (point-min))
|
|
(while (re-search-forward pattern nil t)
|
|
(setq matches (cons (list manual
|
|
(match-string-no-properties 1)
|
|
(match-string-no-properties 2)
|
|
(match-string-no-properties 3))
|
|
matches)))
|
|
(setq index-nodes (cdr index-nodes)
|
|
node (car index-nodes)))
|
|
(Info-goto-node node)))))
|
|
(error nil))
|
|
matches)))
|
|
|
|
;; Like `Info-index-occurrences', but return only the first occurrence found.
|
|
(defun Info-first-index-occurrence (index-entry &optional index-nodes manuals nomsg)
|
|
"Return nil or an occurrence of INDEX-ENTRY in INDEX-NODES of MANUALS.
|
|
Search INDEX-NODES and MANUALS in order.
|
|
A non-nil return value is the first first successful index lookup, in
|
|
the form (FILE INDEX-ENTRY NODE LINE) - see `Info-index-occurrences'.
|
|
|
|
Optional arg INDEX-NODES are the index nodes of MANUALS to search.
|
|
By default (nil value), search all indexes of each manual.
|
|
Optional arg MANUALS is the list of manuals to search, or the symbol
|
|
`all', to search all.
|
|
Optional arg NOMSG non-nil means do not display a progress message."
|
|
(and (not (string= index-entry ""))
|
|
;; Unlike `Info-apropos-matches', we match only the exact string as an index entry.
|
|
(let ((pattern (format "\n\\* +\\(%s\\):[ \t]+\\([^\n]+\\)\
|
|
\\.\\(?:[ \t\n]*(line +\\([0-9]+\\))\\)?"
|
|
(regexp-quote index-entry)))
|
|
(found nil)
|
|
node)
|
|
(unless nomsg
|
|
(message "Searching indexes of %s..."
|
|
(cond ((eq manuals 'all) "all manuals")
|
|
((null (cadr manuals)) (concat (car manuals) " manual"))
|
|
(t (concat "manuals " (mapconcat #'identity manuals ", "))))))
|
|
(condition-case nil
|
|
(with-temp-buffer
|
|
(when (eq manuals 'all) (setq manuals ()))
|
|
(Info-mode)
|
|
;; Next two lines are essentially `(Info-directory)'.
|
|
(info-initialize)
|
|
(Info-find-node-2 "dir" "top" 'NO-GOING-BACK)
|
|
(unless manuals
|
|
(goto-char (point-min))
|
|
(re-search-forward "\\* Menu: *\n" nil t)
|
|
(let (manual)
|
|
(while (re-search-forward "\\*.*: *(\\([^)]+\\))" nil t)
|
|
;; `add-to-list' ensures no dups in `manuals', so the `dolist' runs faster.
|
|
(setq manual (match-string 1))
|
|
(set-text-properties 0 (length manual) nil manual)
|
|
(add-to-list 'manuals manual))))
|
|
(setq found (catch 'Info-first-index-occurrence
|
|
(dolist (manual manuals)
|
|
(unless nomsg
|
|
(message "Searching indexes of manual `%s'..." manual))
|
|
(when (or index-nodes
|
|
(setq index-nodes (Info-index-nodes
|
|
(Info-find-file manual))))
|
|
(Info-find-node manual (car index-nodes))
|
|
(while (progn (goto-char (point-min))
|
|
(when (re-search-forward pattern nil t)
|
|
(throw 'Info-first-index-occurrence
|
|
(list manual
|
|
(match-string-no-properties 1)
|
|
(match-string-no-properties 2)
|
|
(match-string-no-properties 3))))
|
|
(setq index-nodes (cdr index-nodes)
|
|
node (car index-nodes)))
|
|
(Info-goto-node node))))
|
|
nil)))
|
|
(error nil))
|
|
found)))
|
|
|
|
(defun describe-buffer (&optional buffer-name) ; Bound to `C-h B'
|
|
"Describe the existing buffer named BUFFER-NAME.
|
|
The description includes the information provided by `describe-mode'.
|
|
By default, describe the current buffer."
|
|
;; (interactive "bDescribe buffer: ")
|
|
(interactive "@")
|
|
(unless buffer-name (setq buffer-name (buffer-name)))
|
|
(help-setup-xref `(describe-buffer ,buffer-name) (called-interactively-p 'interactive))
|
|
(let ((buf (get-buffer buffer-name)))
|
|
(unless (and buf (buffer-live-p buf)) (error(format "No such live buffer `%s'" buffer-name)))
|
|
(let* ((file (or (buffer-file-name buf)
|
|
(with-current-buffer buf
|
|
(and (eq major-mode 'dired-mode) default-directory))))
|
|
(help-text (concat
|
|
(format "Buffer `%s'\n%s\n\n" buffer-name (make-string
|
|
(+ 9 (length buffer-name)) ?-))
|
|
(and file (format "File/directory:\t%s\n" file))
|
|
(format "Mode:\t\t%s\n"
|
|
(with-current-buffer buf (format-mode-line mode-name)))
|
|
(format "Size in chars:\t%g\n" (buffer-size buf))
|
|
(with-current-buffer buf
|
|
(if (not buffer-display-time)
|
|
"Never displayed\n"
|
|
(format "Last displayed:\t%s\n"
|
|
(format-time-string
|
|
;; Could use this, for short format: "%02H:%02M:%02S"
|
|
;; Or this, for a bit longer: "%_3a %_2l:%02M:%02S %_2p"
|
|
"%a %b %e %T %Y (%z)"
|
|
buffer-display-time))))
|
|
(format "Modified:\t%s\n" (if (buffer-modified-p buf) "yes" "no"))
|
|
(with-current-buffer buf
|
|
(format "Read-only:\t%s\n\n\n" (if buffer-read-only "yes" "no"))))))
|
|
(with-help-window (help-buffer)
|
|
(describe-mode-1 buf))
|
|
(with-current-buffer (help-buffer)
|
|
(let ((inhibit-read-only t))
|
|
(goto-char (point-min))
|
|
(insert help-text))))))
|
|
|
|
|
|
;; REPLACE ORIGINAL
|
|
;;
|
|
;; Use `describe-mode-1', which is different from the original `describe-mode' in these ways:
|
|
;;
|
|
;; 1. Call `Info-make-manuals-xref' to create a cross-ref link to manuals.
|
|
;; 2. Add key-description buttons to command help. Use `insert', not `princ'.
|
|
;;
|
|
(defun describe-mode (&optional buffer)
|
|
"Display documentation of current major mode and minor modes.
|
|
A brief summary of the minor modes comes first, followed by the
|
|
major mode description. This is followed by detailed
|
|
descriptions of the minor modes, each on a separate page.
|
|
|
|
For this to work correctly for a minor mode, the mode's indicator
|
|
variable \(listed in `minor-mode-alist') must also be a function
|
|
whose documentation describes the minor mode."
|
|
(interactive "@")
|
|
(unless buffer (setq buffer (current-buffer)))
|
|
(help-setup-xref (list #'describe-mode buffer) (called-interactively-p 'interactive))
|
|
(with-help-window (help-buffer) (describe-mode-1 buffer))
|
|
nil) ; For the sake of IELM and maybe others
|
|
|
|
(defun describe-mode-1 (buffer)
|
|
"Helper for `describe-mode'.
|
|
Does everything except create the help window and set up the
|
|
back/forward buttons, so you can use this in other help commands that
|
|
have their own back/forward buttons."
|
|
;; For the sake of `help-do-xref' and `help-xref-go-back', do not switch buffers before calling `help-buffer'.
|
|
(with-current-buffer buffer
|
|
(let (minor-modes)
|
|
;; Older packages do not register in minor-mode-list but only in `minor-mode-alist'.
|
|
(dolist (x minor-mode-alist)
|
|
(setq x (car x))
|
|
(unless (memq x minor-mode-list) (push x minor-mode-list)))
|
|
(dolist (mode minor-mode-list) ; Find enabled minor mode we will want to mention.
|
|
;; Document minor mode if listed in `minor-mode-alist', non-nil, and has a function def.
|
|
(let ((fmode (or (get mode :minor-mode-function) mode)))
|
|
(and (boundp mode) (symbol-value mode) (fboundp fmode)
|
|
(let ((pretty-minor-mode (if (string-match "\\(\\(-minor\\)?-mode\\)?\\'"
|
|
(symbol-name fmode))
|
|
(capitalize (substring (symbol-name fmode)
|
|
0 (match-beginning 0)))
|
|
fmode)))
|
|
(push (list fmode pretty-minor-mode
|
|
(format-mode-line (assq mode minor-mode-alist)))
|
|
minor-modes)))))
|
|
(setq minor-modes (sort minor-modes (lambda (a b) (string-lessp (cadr a) (cadr b)))))
|
|
(when minor-modes
|
|
(princ "Enabled minor modes:\n")
|
|
(make-local-variable 'help-button-cache)
|
|
(with-current-buffer standard-output
|
|
(dolist (mode minor-modes)
|
|
(let ((mode-function (nth 0 mode))
|
|
(pretty-minor-mode (nth 1 mode))
|
|
(indicator (nth 2 mode)))
|
|
(add-text-properties 0 (length pretty-minor-mode) '(face bold) pretty-minor-mode)
|
|
(save-excursion
|
|
(goto-char (point-max))
|
|
(princ "\n\f\n")
|
|
(push (point-marker) help-button-cache)
|
|
;; Document the minor modes fully.
|
|
(insert pretty-minor-mode)
|
|
(princ (format " minor mode:\n(`%s'; %s)\n" mode-function (if (zerop (length indicator))
|
|
"no indicator"
|
|
(format "indicator%s" indicator))))
|
|
(save-excursion
|
|
(fill-region-as-paragraph (line-beginning-position 0) (line-end-position 0) nil t t))
|
|
(with-current-buffer standard-output
|
|
(insert (help-documentation mode-function nil 'ADD-HELP-BUTTONS)))
|
|
(Info-make-manuals-xref mode-function
|
|
t nil (not (called-interactively-p 'interactive)))) ; Link manuals.
|
|
(insert-button pretty-minor-mode 'action (car help-button-cache)
|
|
'follow-link t 'help-echo "mouse-2, RET: show full information")
|
|
(newline)))
|
|
(forward-line -1)
|
|
(fill-paragraph nil)
|
|
(forward-line 1))
|
|
(princ "\n(Information about these minor modes follows the major mode info.)\n\n"))
|
|
(let ((mode mode-name)) ; Document the major mode.
|
|
(with-current-buffer standard-output
|
|
(let ((start (point)))
|
|
(insert (format-mode-line mode nil nil buffer))
|
|
(add-text-properties start (point) '(face bold)))))
|
|
(princ " mode")
|
|
(let* ((mode major-mode)
|
|
(file-name (find-lisp-object-file-name mode nil)))
|
|
(when file-name
|
|
(princ (concat " defined in `" (file-name-nondirectory file-name) "'"))
|
|
(with-current-buffer standard-output ; Make a hyperlink to the library.
|
|
(save-excursion (re-search-backward "`\\([^`']+\\)'" nil t)
|
|
(help-xref-button 1 'help-function-def mode file-name))))
|
|
(with-current-buffer standard-output
|
|
(insert (format " (`%s'):\n" mode))
|
|
(save-excursion
|
|
(fill-region-as-paragraph (line-beginning-position 0) (line-end-position 0) nil t t))))
|
|
(let* ((maj major-mode)
|
|
(maj-doc (help-documentation maj nil 'ADD-HELP-BUTTONS)))
|
|
(with-current-buffer standard-output
|
|
(insert maj-doc)
|
|
(Info-make-manuals-xref
|
|
maj t nil (not (called-interactively-p 'interactive)))))))) ; Link to manuals.
|
|
)
|
|
|
|
|
|
|
|
;; REPLACE ORIGINAL in `help-fns.el':
|
|
;;
|
|
;; 1. Preferred candidate is `symbol-nearest-point'.
|
|
;; 2. With a prefix argument, candidates are commands only.
|
|
;; 3. No no-function message if not called interactively.
|
|
;; 4. Works for anonymous functions too: lambda forms and byte-compiled functions. (Fixes Emacs bug #24221.)
|
|
;;
|
|
(defun describe-function (function &optional commandp)
|
|
"Display the full documentation of FUNCTION (a symbol).
|
|
FUNCTION names an Emacs Lisp function, possibly a user command.
|
|
With a prefix argument, candidates are only commands (interactive).
|
|
|
|
Default candidate is: preferably the `symbol-nearest-point', or else
|
|
the innermost function call surrounding point
|
|
\(`function-called-at-point').
|
|
Return the description that was displayed, as a string."
|
|
(interactive
|
|
(let* ((fn (or (and (fboundp 'symbol-nearest-point) (symbol-nearest-point))
|
|
(function-called-at-point)))
|
|
(enable-recursive-minibuffers t)
|
|
(completion-annotate-function (lambda (fn) (and (commandp (intern-soft fn)) " (command)")))
|
|
(type (if current-prefix-arg 'command 'function))
|
|
(prompt (format "Describe %s%s: " type
|
|
(if (if current-prefix-arg (commandp fn) (fboundp fn))
|
|
(format " (default %s)" fn)
|
|
"")))
|
|
(pred (if current-prefix-arg
|
|
#'commandp
|
|
(lambda (fn) (or (fboundp fn) (get fn 'function-documentation)))))
|
|
val)
|
|
(setq val (completing-read prompt (if (fboundp 'help--symbol-completion-table)
|
|
#'help--symbol-completion-table
|
|
obarray)
|
|
pred t nil nil (and (funcall pred fn) (symbol-name fn))))
|
|
(list (if (equal val "") fn (intern val)) current-prefix-arg)))
|
|
(let* ((interactivep (if (or (> emacs-major-version 23) ; Emacs 23.1 `called-interactively' accepts no arg.
|
|
(and (= emacs-major-version 23) (> emacs-minor-version 1)))
|
|
(called-interactively-p 'interactive)
|
|
(interactive-p)))
|
|
(err/msg-fn (if interactivep #'message #'error))
|
|
(fn/cmd-txt (if commandp 'command 'function)))
|
|
(if (and interactivep (not function))
|
|
(funcall err/msg-fn "You did not specify a function symbol") ; Avoid "Not a defined function: `nil'".
|
|
(if (not (if commandp
|
|
(commandp function)
|
|
(or (functionp function) ; Allow anonymous functions (Emacs bug #24221).
|
|
(and function (fboundp (intern-soft function))) ; Allow macros and special forms.
|
|
(and function (get function 'function-documentation)))))
|
|
(funcall err/msg-fn "Not a defined %s: `%S'" fn/cmd-txt function)
|
|
(help-setup-xref (list #'describe-function function)
|
|
(if (or (> emacs-major-version 23) ; Emacs 23.1 `called-interactively' accepts no arg.
|
|
(and (= emacs-major-version 23) (> emacs-minor-version 1)))
|
|
(called-interactively-p 'interactive)
|
|
(interactive-p)))
|
|
(save-excursion
|
|
(if (fboundp 'with-help-window)
|
|
(with-help-window (help-buffer) ; Emacs 24.4 needs this - see Emacs bug #17109.
|
|
(if (get function 'reader-construct)
|
|
(princ function)
|
|
(prin1 function))
|
|
;; Use " is " instead of ": " so it is easier to get the function name using `forward-sexp'.
|
|
(princ " is ")
|
|
(describe-function-1 function)
|
|
(with-current-buffer standard-output (buffer-string))) ; Return help text.
|
|
(with-output-to-temp-buffer (help-buffer)
|
|
(prin1 function)
|
|
;; Use " is " instead of ": " so it is easier to get the function name using `forward-sexp'.
|
|
(princ " is ")
|
|
(describe-function-1 function)
|
|
(print-help-return-message)
|
|
(with-current-buffer standard-output (buffer-string))))))))) ; Return help text.
|
|
|
|
|
|
|
|
;; REPLACE ORIGINAL in `help-fns.el' (`help.el', for Emacs < 22):
|
|
;;
|
|
;; Fill long lines. Add `,' before "which".
|
|
;;
|
|
(when (< emacs-major-version 23)
|
|
|
|
(defun describe-function-1 (function)
|
|
(let* ((def (if (symbolp function) (symbol-function function) function))
|
|
(beg (if (commandp def) "an interactive " "a "))
|
|
(pt1 (with-current-buffer (help-buffer) (point)))
|
|
file-name string)
|
|
(setq string (cond ((or (stringp def) (vectorp def)) "a keyboard macro")
|
|
((subrp def) (if (eq 'unevalled (cdr (subr-arity def)))
|
|
(concat beg "special form")
|
|
(concat beg "built-in function")))
|
|
((byte-code-function-p def) (concat beg "compiled Lisp function"))
|
|
((symbolp def)
|
|
(while (symbolp (symbol-function def))
|
|
(setq def (symbol-function def)))
|
|
(format "an alias for `%s'" def))
|
|
((eq (car-safe def) 'lambda) (concat beg "Lisp function"))
|
|
((eq (car-safe def) 'macro) "a Lisp macro")
|
|
((eq (car-safe def) 'autoload)
|
|
(setq file-name (nth 1 def))
|
|
(format "%s autoloaded %s" (if (commandp def) "an interactive" "an")
|
|
(if (eq (nth 4 def) 'keymap)
|
|
"keymap"
|
|
(if (nth 4 def) "Lisp macro" "Lisp function"))))
|
|
((keymapp def)
|
|
(let ((is-full nil)
|
|
(elts (cdr-safe def)))
|
|
(while elts
|
|
(when (char-table-p (car-safe elts))
|
|
(setq is-full t
|
|
elts ()))
|
|
(setq elts (cdr-safe elts)))
|
|
(if is-full "a full keymap" "a sparse keymap")))
|
|
(t "")))
|
|
(princ string)
|
|
(with-current-buffer standard-output
|
|
(save-excursion (save-match-data (when (re-search-backward "alias for `\\([^`']+\\)'" nil t)
|
|
(help-xref-button 1 'help-function def)))))
|
|
(unless file-name (setq file-name (symbol-file function 'defun)))
|
|
(setq file-name (describe-simplify-lib-file-name file-name))
|
|
(when (equal file-name "loaddefs.el")
|
|
;; Find the real def site of the preloaded function. This is necessary only for defaliases.
|
|
(let ((location (condition-case nil
|
|
(find-function-search-for-symbol function nil "loaddefs.el")
|
|
(error nil))))
|
|
(when location
|
|
(with-current-buffer (car location)
|
|
(goto-char (cdr location))
|
|
(when (re-search-backward "^;;; Generated autoloads from \\(.*\\)" nil t)
|
|
(setq file-name (match-string 1)))))))
|
|
(when (and (null file-name) (subrp def)) ; Find the C source file name.
|
|
(setq file-name (if (get-buffer " *DOC*") (help-C-file-name def 'subr) 'C-source)))
|
|
(when file-name
|
|
(princ " in `")
|
|
;; We used to add `.el' to file name, but that's wrong when the user used `load-file'.
|
|
(princ (if (eq file-name 'C-source) "C source code" file-name))
|
|
(princ "'")
|
|
(with-current-buffer standard-output ; Make a hyperlink to the library.
|
|
(save-excursion (re-search-backward "`\\([^`']+\\)'" nil t)
|
|
(help-xref-button 1 'help-function-def function file-name))))
|
|
(princ ".")
|
|
(with-current-buffer (help-buffer)
|
|
(fill-region-as-paragraph (save-excursion (goto-char pt1) (forward-line 0) (point)) (point)))
|
|
(terpri)(terpri)
|
|
(when (commandp function)
|
|
(let ((pt2 (with-current-buffer (help-buffer) (point))))
|
|
(if (and (eq function 'self-insert-command)
|
|
(eq (key-binding "a") 'self-insert-command)
|
|
(eq (key-binding "b") 'self-insert-command)
|
|
(eq (key-binding "c") 'self-insert-command))
|
|
(princ "It is bound to many ordinary text characters.\n")
|
|
(let* ((remapped (command-remapping function))
|
|
(keys (where-is-internal
|
|
(or remapped function) overriding-local-map nil nil))
|
|
non-modified-keys)
|
|
(dolist (key keys) ; Which non-control non-meta keys run this command?
|
|
(when (member (event-modifiers (aref key 0)) '(nil (shift)))
|
|
(push key non-modified-keys)))
|
|
(when remapped
|
|
(princ "It is remapped to `") (princ (symbol-name remapped)) (princ "'"))
|
|
(when keys
|
|
(princ (if remapped ", which is bound to " "It is bound to "))
|
|
;; If lots of ordinary text characters run this command, don't mention them one by one.
|
|
(if (< (length non-modified-keys) 10)
|
|
(princ (mapconcat (if (fboundp 'naked-key-description)
|
|
#'naked-key-description
|
|
#'key-description)
|
|
keys ", "))
|
|
(dolist (key non-modified-keys) (setq keys (delq key keys)))
|
|
(if keys
|
|
(progn (princ (mapconcat (if (fboundp 'naked-key-description)
|
|
#'naked-key-description
|
|
#'key-description)
|
|
keys ", "))
|
|
(princ ", and many ordinary text characters"))
|
|
(princ "many ordinary text characters"))))
|
|
(when (or remapped keys non-modified-keys) (princ ".") (terpri))))
|
|
(with-current-buffer (help-buffer) (fill-region-as-paragraph pt2 (point)))
|
|
(terpri)))
|
|
(let* ((arglist (help-function-arglist def))
|
|
(doc (documentation function))
|
|
(usage (help-split-fundoc doc function)))
|
|
(with-current-buffer standard-output
|
|
;; If definition is a keymap, skip arglist note.
|
|
(unless (keymapp def)
|
|
(let* ((use (cond (usage (setq doc (cdr usage)) (car usage))
|
|
((listp arglist) (format "%S" (help-make-usage function arglist)))
|
|
((stringp arglist) arglist)
|
|
;; Maybe the arglist is in the docstring of the alias.
|
|
((let ((fun function))
|
|
(while (and (symbolp fun)
|
|
(setq fun (symbol-function fun))
|
|
(not (setq usage (help-split-fundoc (documentation fun)
|
|
function)))))
|
|
usage)
|
|
(car usage))
|
|
((or (stringp def) (vectorp def))
|
|
(format "\nMacro: %s" (format-kbd-macro def)))
|
|
(t "[Missing arglist. Please make a bug report.]")))
|
|
(high (help-highlight-arguments use doc)))
|
|
(let ((fill-begin (point)))
|
|
(insert (car high) "\n")
|
|
(fill-region fill-begin (point)))
|
|
(setq doc (cdr high))))
|
|
(let ((obsolete (and (symbolp function) ; function might be a lambda construct.
|
|
(get function 'byte-obsolete-info))))
|
|
(when obsolete
|
|
(princ "\nThis function is obsolete")
|
|
(when (nth 2 obsolete) (insert (format " since %s" (nth 2 obsolete))))
|
|
(insert ";\n" (if (stringp (car obsolete))
|
|
(car obsolete)
|
|
(format "use `%s' instead." (car obsolete)))
|
|
"\n"))
|
|
(insert "\n" (or doc "Not documented.")))))))
|
|
)
|
|
|
|
|
|
;; REPLACE ORIGINAL in `help-fns.el':
|
|
;;
|
|
;; 1. Call `Info-make-manuals-xref' to create a cross-ref link to manuals.
|
|
;; 2. Add key-description buttons to command help.
|
|
;;
|
|
(when (and (boundp 'Info-virtual-files) ; Emacs 23.2 through 24.2
|
|
(not (fboundp 'help-fns--autoloaded-p)))
|
|
|
|
(defun describe-function-1 (function)
|
|
(let* ((advised (and (symbolp function) (featurep 'advice) (ad-get-advice-info function)))
|
|
;; If the function is advised, use the symbol that has the real def, if already set up.
|
|
(real-function (or (and advised (let ((origname (cdr (assq 'origname advised))))
|
|
(and (fboundp origname) origname)))
|
|
function))
|
|
;; Get the real definition.
|
|
(def (if (symbolp real-function) (symbol-function real-function) function))
|
|
(aliased (symbolp def))
|
|
(real-def (if aliased
|
|
(let ((fn def))
|
|
(while (and (fboundp fn) (symbolp (symbol-function fn)))
|
|
(setq fn (symbol-function fn)))
|
|
fn)
|
|
def))
|
|
(file-name (find-lisp-object-file-name function def))
|
|
(beg (if (commandp def) "an interactive " "a "))
|
|
(pt1 (with-current-buffer (help-buffer) (point)))
|
|
string errtype)
|
|
(setq string (cond ((or (stringp def) (vectorp def)) "a keyboard macro")
|
|
((subrp def) (if (eq 'unevalled (cdr (subr-arity def)))
|
|
(concat beg "special form")
|
|
(concat beg "built-in function")))
|
|
((byte-code-function-p def) (concat beg "compiled Lisp function"))
|
|
((symbolp def)
|
|
(while (and (fboundp def) (symbolp (symbol-function def)))
|
|
(setq def (symbol-function def)))
|
|
;; Handle (defalias 'foo 'bar), where bar is undefined.
|
|
(unless (fboundp def) (setq errtype 'alias))
|
|
(format "an alias for `%s'" def))
|
|
((eq (car-safe def) 'lambda) (concat beg "Lisp function"))
|
|
((eq (car-safe def) 'macro) "a Lisp macro")
|
|
((eq (car-safe def) 'closure) (concat beg "Lisp closure"))
|
|
((eq (car-safe def) 'autoload)
|
|
(format "%s autoloaded %s" (if (commandp def) "an interactive" "an")
|
|
(if (eq (nth 4 def) 'keymap)
|
|
"keymap"
|
|
(if (nth 4 def) "Lisp macro" "Lisp function"))))
|
|
((keymapp def) (let ((is-full nil)
|
|
(elts (cdr-safe def)))
|
|
(while elts
|
|
(when (char-table-p (car-safe elts))
|
|
(setq is-full t
|
|
elts ()))
|
|
(setq elts (cdr-safe elts)))
|
|
(if is-full "a full keymap" "a sparse keymap")))
|
|
(t "")))
|
|
(princ string)
|
|
(if (eq errtype 'alias)
|
|
(princ ",\nwhich is not defined. Please make a bug report.")
|
|
(with-current-buffer standard-output
|
|
(save-excursion (save-match-data (when (re-search-backward "alias for `\\([^`']+\\)'" nil t)
|
|
(help-xref-button 1 'help-function def)))))
|
|
(when file-name
|
|
(princ " in `")
|
|
;; We used to add `.el' to the file name, but that's wrong when the user used `load-file'.
|
|
(princ (if (eq file-name 'C-source) "C source code" (file-name-nondirectory file-name)))
|
|
(princ "'")
|
|
;; Make a hyperlink to the library.
|
|
(with-current-buffer standard-output
|
|
(save-excursion (re-search-backward "`\\([^`']+\\)'" nil t)
|
|
(help-xref-button 1 'help-function-def function file-name))))
|
|
(princ ".")
|
|
(with-current-buffer (help-buffer)
|
|
(fill-region-as-paragraph (save-excursion (goto-char pt1) (forward-line 0) (point)) (point)))
|
|
(terpri) (terpri)
|
|
(when (commandp function)
|
|
(let ((pt2 (with-current-buffer (help-buffer) (point)))
|
|
(remapped (command-remapping function)))
|
|
(unless (memq remapped '(ignore undefined))
|
|
(let ((keys (where-is-internal (or remapped function) overriding-local-map nil nil))
|
|
non-modified-keys)
|
|
(if (and (eq function 'self-insert-command)
|
|
(vectorp (car-safe keys))
|
|
(consp (aref (car keys) 0)))
|
|
(princ "It is bound to many ordinary text characters.\n")
|
|
(dolist (key keys) ; Which non-control non-meta keys run this command?
|
|
(when (member (event-modifiers (aref key 0)) '(nil (shift)))
|
|
(push key non-modified-keys)))
|
|
(when remapped
|
|
(princ "It is remapped to `") (princ (symbol-name remapped)) (princ "'"))
|
|
(when keys
|
|
(princ (if remapped ", which is bound to " "It is bound to "))
|
|
;; If lots of ordinary text chars run this command, don't mention them one by one.
|
|
(if (< (length non-modified-keys) 10)
|
|
(princ (mapconcat (if (fboundp 'naked-key-description)
|
|
#'naked-key-description
|
|
#'key-description)
|
|
keys ", "))
|
|
(dolist (key non-modified-keys) (setq keys (delq key keys)))
|
|
(if keys
|
|
(progn (princ (mapconcat (if (fboundp 'naked-key-description)
|
|
#'naked-key-description
|
|
#'key-description)
|
|
keys ", "))
|
|
(princ ", and many ordinary text characters"))
|
|
(princ "many ordinary text characters"))))
|
|
(when (or remapped keys non-modified-keys) (princ ".") (terpri)))))
|
|
(with-current-buffer (help-buffer)
|
|
(fill-region-as-paragraph pt2 (point))
|
|
(unless (and (eq ?\n (char-before)) ; Quicker than `looking-back', apparently.
|
|
(eq ?\n (char-before (1- (point)))))
|
|
(terpri)))))
|
|
;; `list*' etc. do not get this property until `cl-hack-byte-compiler' runs,
|
|
;; which is after bytecomp is loaded.
|
|
(when (and (symbolp function) (eq (get function 'byte-compile) 'cl-byte-compile-compiler-macro))
|
|
(princ "This function has a compiler macro")
|
|
(let ((lib (get function 'compiler-macro-file)))
|
|
(when (stringp lib)
|
|
(princ (format " in `%s'" lib))
|
|
(with-current-buffer standard-output
|
|
(save-excursion (re-search-backward "`\\([^`']+\\)'" nil t)
|
|
(help-xref-button 1 'help-function-cmacro function lib)))))
|
|
(princ ".\n\n"))
|
|
(let* ((advertised (gethash def advertised-signature-table t))
|
|
(arglist (if (listp advertised) advertised (help-function-arglist def)))
|
|
(doc (condition-case err
|
|
(help-documentation function nil 'ADD-HELP-BUTTONS)
|
|
(error (format "No Doc! %S" err))))
|
|
(usage (help-split-fundoc doc function)))
|
|
(with-current-buffer standard-output
|
|
(unless (keymapp function) ; If definition is a keymap, skip arglist note.
|
|
(when usage (setq doc (cdr usage)))
|
|
(let* ((use (cond ((and usage (not (listp advertised))) (car usage))
|
|
((listp arglist) (format "%S" (help-make-usage function arglist)))
|
|
((stringp arglist) arglist)
|
|
;; Maybe arglist is in doc string of a symbol this one is aliased to.
|
|
((let ((fun real-function))
|
|
(while (and (symbolp fun)
|
|
(setq fun (symbol-function fun))
|
|
(not (setq usage (help-split-fundoc
|
|
(help-documentation
|
|
fun nil 'ADD-HELP-BUTTONS)
|
|
function)))))
|
|
usage)
|
|
(car usage))
|
|
((or (stringp def) (vectorp def))
|
|
(format "\nMacro: %s" (format-kbd-macro def)))
|
|
(t "[Missing arglist. Please submit a bug report.]")))
|
|
(high (help-highlight-arguments use doc)))
|
|
(let ((fill-begin (point)))
|
|
(insert (car high) "\n")
|
|
(fill-region fill-begin (point)))
|
|
(setq doc (cdr high))))
|
|
;; If this is a derived mode, link to the parent.
|
|
(let ((parent-mode (and (symbolp real-function) (get real-function 'derived-mode-parent))))
|
|
(when parent-mode
|
|
(with-current-buffer standard-output
|
|
(insert "\nParent mode: `")
|
|
(let ((beg (point)))
|
|
(insert (format "%s" parent-mode))
|
|
(make-text-button beg (point) 'type 'help-function 'help-args (list parent-mode))))
|
|
(princ "'.\n")))
|
|
(let* ((obsolete (and (symbolp function) ; Function might be a lambda construct.
|
|
(get function 'byte-obsolete-info)))
|
|
(use (car obsolete)))
|
|
(when obsolete
|
|
(princ "\nThis function is obsolete")
|
|
(when (nth 2 obsolete) (insert (format " since %s" (nth 2 obsolete))))
|
|
(insert (cond ((stringp use) (concat ";\n" use))
|
|
(use (format ";\nuse `%s' instead." use))
|
|
(t "."))
|
|
"\n"))
|
|
(insert "\n")
|
|
(when (and doc (boundp 'Info-virtual-files)) ; Emacs 23.2+
|
|
(Info-make-manuals-xref function)) ; Link to manuals. (With progress message.)
|
|
(insert (or doc "Not documented."))))))))
|
|
|
|
)
|
|
|
|
(when (fboundp 'help-fns--autoloaded-p) ; Emacs 24.3+
|
|
|
|
|
|
;; REPLACE ORIGINAL in `help-fns.el':
|
|
;;
|
|
;; Use `naked-key-description' if available, instead of `key-description'.
|
|
;;
|
|
(defun help-fns--key-bindings (function)
|
|
(when (commandp function)
|
|
(let ((pt2 (with-current-buffer standard-output (point)))
|
|
(remapped (command-remapping function)))
|
|
(unless (memq remapped '(ignore undefined))
|
|
(let ((keys (where-is-internal (or remapped function) overriding-local-map nil nil))
|
|
non-modified-keys)
|
|
(if (and (eq function 'self-insert-command)
|
|
(vectorp (car-safe keys))
|
|
(consp (aref (car keys) 0)))
|
|
(princ "It is bound to many ordinary text characters.\n")
|
|
(dolist (key keys) ; Which non-control non-meta keys run this command?
|
|
(when (member (event-modifiers (aref key 0)) '(nil (shift))) (push key non-modified-keys)))
|
|
(when remapped
|
|
(princ "Its keys are remapped to ")
|
|
(princ (if (symbolp remapped)
|
|
(if (fboundp 'format-message)
|
|
(format-message "`%s'" remapped)
|
|
(concat "`" (symbol-name remapped) "'"))
|
|
"an anonymous command"))
|
|
(princ ".\n"))
|
|
(when keys
|
|
(princ (if remapped "Without this remapping, it would be bound to " "It is bound to "))
|
|
;; If lots of ordinary text characters run this command, don't mention them one by one.
|
|
(if (< (length non-modified-keys) 10)
|
|
(princ (mapconcat (if (fboundp 'naked-key-description)
|
|
#'naked-key-description
|
|
#'key-description)
|
|
keys ", "))
|
|
(dolist (key non-modified-keys) (setq keys (delq key keys)))
|
|
(if keys
|
|
(progn (princ (mapconcat (if (fboundp 'naked-key-description)
|
|
#'naked-key-description
|
|
#'key-description)
|
|
keys ", "))
|
|
(princ ", and many ordinary text characters"))
|
|
(princ "many ordinary text characters"))))
|
|
(when (or remapped keys non-modified-keys) (princ ".") (terpri)))))
|
|
(with-current-buffer standard-output
|
|
(fill-region-as-paragraph pt2 (point))
|
|
(unless (and (eq ?\n (char-before)) ; Quicker than `looking-back', apparently.
|
|
(eq ?\n (char-before (1- (point)))))
|
|
(terpri))))))
|
|
|
|
|
|
;; REPLACE ORIGINAL in `help-fns.el'
|
|
;;
|
|
;; Add key-description buttons to command help: Use `help-documentation', not `documentation'.
|
|
;;
|
|
(defun help-fns--signature (function doc real-def real-function buffer)
|
|
"Insert usage at point and return docstring."
|
|
(if (keymapp function)
|
|
doc ; If definition is a keymap, skip arglist note.
|
|
(let* ((advertised (gethash real-def advertised-signature-table t))
|
|
(arglist (if (listp advertised) advertised (help-function-arglist real-def)))
|
|
(usage (help-split-fundoc doc function)))
|
|
(when usage (setq doc (cdr usage)))
|
|
(let* ((use (cond ((and usage (not (listp advertised))) (car usage))
|
|
((listp arglist)
|
|
(if (fboundp 'help--make-usage-docstring)
|
|
(help--make-usage-docstring function arglist) ; Emacs 25+.
|
|
(format "%S" (help-make-usage function arglist))))
|
|
((stringp arglist) arglist)
|
|
;; Maybe the arglist is in the docstring of a symbol this one is aliased to.
|
|
((let ((fun real-function))
|
|
(while (and (symbolp fun)
|
|
(setq fun (symbol-function fun))
|
|
(not (setq usage (help-split-fundoc
|
|
(help-documentation fun nil 'ADD-HELP-BUTTONS)
|
|
function)))))
|
|
usage)
|
|
(car usage))
|
|
((or (stringp real-def) (vectorp real-def))
|
|
(format "\nMacro: %s"
|
|
(if (fboundp 'help--docstring-quote)
|
|
(help--docstring-quote (format-kbd-macro real-def)) ; Emacs 25+.
|
|
(format-kbd-macro real-def))))
|
|
(t "[Missing arglist. Please submit a bug report.]")))
|
|
;; Insert "`X", not "(\` X)", when documenting `X.
|
|
(use1 (replace-regexp-in-string "\\`(\\\\=\\\\\\\\=` \\([^\n ]*\\))\\'" "\\\\=`\\1" use t))
|
|
(high (if buffer
|
|
(let (subst-use1 subst-doc)
|
|
(with-current-buffer buffer
|
|
(setq subst-use1 (substitute-command-keys use1)
|
|
subst-doc (substitute-command-keys doc)))
|
|
(help-highlight-arguments subst-use1 subst-doc))
|
|
(cons use1 doc))))
|
|
(let ((fill-begin (point))
|
|
(high-usage (car high))
|
|
(high-doc (cdr high)))
|
|
(unless (and (symbolp function) (get function 'reader-construct))
|
|
(insert high-usage "\n"))
|
|
(fill-region fill-begin (point))
|
|
high-doc)))))
|
|
)
|
|
|
|
(when (and (= emacs-major-version 24) (= emacs-minor-version 3))
|
|
|
|
(defun describe-function-1 (function)
|
|
(let* ((advised (and (symbolp function) (featurep 'advice) (ad-get-advice-info function)))
|
|
;; If the function is advised, use the symbol that has the real def, if already set up.
|
|
(real-function (or (and advised (let ((origname (cdr (assq 'origname advised))))
|
|
(and (fboundp origname) origname)))
|
|
function))
|
|
;; Get the real definition.
|
|
(def (if (symbolp real-function) (symbol-function real-function) function))
|
|
(aliased (symbolp def))
|
|
(real-def (if aliased
|
|
(let ((fn def))
|
|
(while (and (fboundp fn) (symbolp (symbol-function fn)))
|
|
(setq fn (symbol-function fn)))
|
|
fn)
|
|
def))
|
|
(file-name (find-lisp-object-file-name function def))
|
|
(pt1 (with-current-buffer (help-buffer) (point)))
|
|
(beg (if (and (or (byte-code-function-p def) (keymapp def)
|
|
(memq (car-safe def) '(macro lambda closure)))
|
|
file-name
|
|
(help-fns--autoloaded-p function file-name))
|
|
(if (commandp def) "an interactive autoloaded " "an autoloaded ")
|
|
(if (commandp def) "an interactive " "a "))))
|
|
;; Print what kind of function-like object FUNCTION is.
|
|
(princ (cond ((or (stringp def) (vectorp def)) "a keyboard macro")
|
|
((subrp def) (if (eq 'unevalled (cdr (subr-arity def)))
|
|
(concat beg "special form")
|
|
(concat beg "built-in function")))
|
|
((byte-code-function-p def) (concat beg "compiled Lisp function"))
|
|
(aliased (format "an alias for `%s'" real-def))
|
|
((eq (car-safe def) 'lambda) (concat beg "Lisp function"))
|
|
((eq (car-safe def) 'macro) (concat beg "Lisp macro"))
|
|
((eq (car-safe def) 'closure) (concat beg "Lisp closure"))
|
|
((autoloadp def)
|
|
(format "%s autoloaded %s" (if (commandp def) "an interactive" "an")
|
|
(if (eq (nth 4 def) 'keymap)
|
|
"keymap"
|
|
(if (nth 4 def) "Lisp macro" "Lisp function"))))
|
|
((keymapp def) (let ((is-full nil)
|
|
(elts (cdr-safe def)))
|
|
(while elts
|
|
(when (char-table-p (car-safe elts))
|
|
(setq is-full t
|
|
elts ()))
|
|
(setq elts (cdr-safe elts)))
|
|
(concat beg (if is-full "keymap" "sparse keymap"))))
|
|
(t "")))
|
|
(if (and aliased (not (fboundp real-def)))
|
|
(princ ",\nwhich is not defined. Please submit a bug report.")
|
|
(with-current-buffer standard-output
|
|
(save-excursion (save-match-data (when (re-search-backward "alias for `\\([^`']+\\)'" nil t)
|
|
(help-xref-button 1 'help-function real-def)))))
|
|
(when file-name
|
|
(princ " in `")
|
|
;; We used to add `.el' to the file name, but that's wrong when the user used `load-file'.
|
|
(princ (if (eq file-name 'C-source) "C source code" (file-name-nondirectory file-name)))
|
|
(princ "'")
|
|
;; Make a hyperlink to the library.
|
|
(with-current-buffer standard-output
|
|
(save-excursion (re-search-backward "`\\([^`']+\\)'" nil t)
|
|
(help-xref-button 1 'help-function-def function file-name))))
|
|
(princ ".")
|
|
(with-current-buffer (help-buffer)
|
|
(fill-region-as-paragraph (save-excursion (goto-char pt1) (forward-line 0) (point)) (point)))
|
|
(terpri) (terpri)
|
|
(let* ((doc-raw (documentation function 'RAW))
|
|
;; If the function is autoloaded, and its docstring has key substitution constructs,
|
|
;; load the library. In any case, add help buttons.
|
|
(doc (if (and (autoloadp real-def)
|
|
doc-raw
|
|
help-enable-auto-load
|
|
(string-match "\\([^\\]=\\|[^=]\\|\\`\\)\\\\[[{<]" doc-raw)
|
|
(load (cadr real-def) t))
|
|
(help-substitute-command-keys doc-raw 'ADD-HELP-BUTTONS)
|
|
(condition-case err
|
|
(help-documentation function nil 'ADD-HELP-BUTTONS)
|
|
(error (format "No Doc! %S" err))))))
|
|
(help-fns--key-bindings function)
|
|
(with-current-buffer standard-output
|
|
(setq doc (help-fns--signature function doc real-def real-function nil)) ; No BUFFER arg.
|
|
(help-fns--compiler-macro function)
|
|
(help-fns--parent-mode function)
|
|
(help-fns--obsolete function)
|
|
(insert "\n")
|
|
(when (and doc (boundp 'Info-virtual-files)) ; Emacs 23.2+
|
|
(Info-make-manuals-xref function)) ; Link to manuals. (With progress message.)
|
|
(insert (or doc "Not documented.")))))))
|
|
|
|
)
|
|
|
|
(when (or (= emacs-major-version 25) (and (= emacs-major-version 24) (> emacs-minor-version 3)))
|
|
|
|
(defun describe-function-1 (function)
|
|
(let* ((advised (and (symbolp function)
|
|
(advice--p (advice--symbol-function function))))
|
|
;; If the function is advised, use the symbol that has the real definition, if already set up.
|
|
(real-function (or (and advised (advice--cd*r (advice--symbol-function function)))
|
|
function))
|
|
;; Get the real definition.
|
|
(def (if (symbolp real-function)
|
|
(or (symbol-function real-function) (signal 'void-function (list real-function)))
|
|
real-function))
|
|
(aliased (or (symbolp def)
|
|
(and advised (symbolp real-function) ; Advised & aliased function.
|
|
(not (eq 'autoload (car-safe def))))))
|
|
(real-def (cond (aliased (let ((f real-function))
|
|
(while (and (fboundp f) (symbolp (symbol-function f)))
|
|
(setq f (symbol-function f)))
|
|
f))
|
|
((subrp def) (intern (subr-name def)))
|
|
(t def)))
|
|
(sig-key (if (subrp def) (indirect-function real-def) real-def))
|
|
(file-name (find-lisp-object-file-name function (if aliased 'defun def)))
|
|
(pt1 (with-current-buffer (help-buffer) (point)))
|
|
(beg (if (and (or (byte-code-function-p def) (keymapp def) (memq (car-safe def) '(macro lambda closure)))
|
|
(stringp file-name)
|
|
(help-fns--autoloaded-p function file-name))
|
|
(if (commandp def) "an interactive autoloaded " "an autoloaded ")
|
|
(if (commandp def) "an interactive " "a "))))
|
|
;; Print what kind of function-like object FUNCTION is.
|
|
(princ (cond ((or (stringp def) (vectorp def)) "a keyboard macro")
|
|
((subrp def) (if (eq 'unevalled (cdr (subr-arity def)))
|
|
(concat beg "special form")
|
|
(concat beg "built-in function")))
|
|
;; Aliases are Lisp functions, so we need to check aliases before functions.
|
|
(aliased (funcall (if (fboundp 'format-message) #'format-message #'format)
|
|
"an alias for `%s'"
|
|
real-def))
|
|
((autoloadp def) (format "%s autoloaded %s"
|
|
(if (commandp def) "an interactive" "an")
|
|
(if (eq (nth 4 def) 'keymap)
|
|
"keymap"
|
|
(if (nth 4 def) "Lisp macro" "Lisp function"))))
|
|
((or (eq (car-safe def) 'macro)
|
|
;; For advised macros, DEF is a lambda expression or is `byte-code-function-p',
|
|
;; so check macros before functions.
|
|
(macrop function))
|
|
(concat beg "Lisp macro"))
|
|
((byte-code-function-p def) (concat beg "compiled Lisp function"))
|
|
((eq (car-safe def) 'lambda) (concat beg "Lisp function"))
|
|
((eq (car-safe def) 'closure) (concat beg "Lisp closure"))
|
|
((keymapp def) (let ((is-full nil)
|
|
(elts (cdr-safe def)))
|
|
(while elts
|
|
(when (char-table-p (car-safe elts))
|
|
(setq is-full t
|
|
elts ()))
|
|
(setq elts (cdr-safe elts)))
|
|
(concat beg (if is-full "keymap" "sparse keymap"))))
|
|
(t "")))
|
|
(if (and aliased (not (fboundp real-def)))
|
|
(princ ",\nwhich is not defined. Please submit a bug report.")
|
|
(with-current-buffer standard-output
|
|
(save-excursion
|
|
(save-match-data
|
|
(when (re-search-backward (substitute-command-keys "alias for `\\([^`']+\\)'") nil t)
|
|
(help-xref-button 1 'help-function real-def)))))
|
|
(when file-name
|
|
;; We used to add `.el' to the file name, but that's wrong when the user used `load-file'.
|
|
(princ (funcall (if (fboundp 'format-message) #'format-message #'format)
|
|
" in `%s'"
|
|
(if (eq file-name 'C-source)
|
|
"C source code"
|
|
(if (fboundp 'help-fns-short-filename)
|
|
(help-fns-short-filename file-name) ; Emacs 25+
|
|
(file-name-nondirectory file-name)))))
|
|
;; Make a hyperlink to the library.
|
|
(with-current-buffer standard-output
|
|
(save-excursion (re-search-backward (substitute-command-keys "`\\([^`']+\\)'") nil t)
|
|
(help-xref-button 1 'help-function-def function file-name))))
|
|
(princ ".")
|
|
(with-current-buffer (help-buffer)
|
|
(fill-region-as-paragraph (save-excursion (goto-char pt1) (forward-line 0) (point)) (point)))
|
|
(terpri) (terpri)
|
|
(let* ((doc-raw (documentation function 'RAW))
|
|
(key-bind-buf (and (> emacs-major-version 24) (current-buffer))) ; Not used before Emacs 25.
|
|
;; If the function is autoloaded and its docstring has key substitution constructs, then
|
|
;; load the library. In any case, add help buttons.
|
|
(doc (if (and (autoloadp real-def)
|
|
doc-raw
|
|
help-enable-auto-load
|
|
(string-match "\\([^\\]=\\|[^=]\\|\\`\\)\\\\[[{<]" doc-raw)
|
|
(autoload-do-load real-def))
|
|
(help-substitute-command-keys doc-raw 'ADD-HELP-BUTTONS)
|
|
(condition-case err
|
|
(help-documentation function nil 'ADD-HELP-BUTTONS)
|
|
(error (format "No Doc! %S" err))))))
|
|
(help-fns--key-bindings function)
|
|
(with-current-buffer standard-output
|
|
(setq doc (help-fns--signature function doc-raw sig-key real-function key-bind-buf))
|
|
(run-hook-with-args 'help-fns-describe-function-functions function)
|
|
(insert "\n")
|
|
(when doc (Info-make-manuals-xref function)) ; Link to manuals. (With progress message.)
|
|
(insert (or doc "Not documented."))
|
|
(when (or (function-get function 'pure)
|
|
(function-get function 'side-effect-free))
|
|
(insert "\nThis function does not change global state, including the match data."))
|
|
;; Avoid asking the user annoying questions if she decides
|
|
;; to save the help buffer, when her locale's codeset
|
|
;; isn't UTF-8.
|
|
(unless (and (boundp 'text-quoting-style) ; Emacs 25+
|
|
(memq text-quoting-style '(straight grave)))
|
|
(set-buffer-file-coding-system 'utf-8)))))))
|
|
|
|
)
|
|
|
|
(when (> emacs-major-version 25)
|
|
|
|
(defun describe-function-1 (function)
|
|
(let ((pt1 (with-current-buffer (help-buffer) (point))))
|
|
(help-fns-function-description-header function)
|
|
(with-current-buffer (help-buffer)
|
|
(fill-region-as-paragraph (save-excursion (goto-char pt1) (forward-line 0) (point)) (point))))
|
|
(terpri)(terpri)
|
|
|
|
(pcase-let* ((`(,real-function ,def ,_aliased ,real-def)
|
|
(help-fns--analyze-function function))
|
|
(doc-raw
|
|
(condition-case nil
|
|
;; Maybe `documentation' should return nil for invalid functions, not signal an error.
|
|
(documentation function 'RAW)
|
|
((invalid-function void-function) nil))) ; E.g., an alias for a not yet defined function.
|
|
(key-bind-buf (current-buffer))
|
|
(doc
|
|
;; If the function is autoloaded, and its docstring has key substitution constructs, then load the library.
|
|
;; In any case, add help buttons to doc.
|
|
(if (and (autoloadp real-def)
|
|
doc-raw
|
|
help-enable-auto-load
|
|
(string-match-p "\\([^\\]=\\|[^=]\\|\\`\\)\\\\[[{<]" doc-raw)
|
|
(autoload-do-load real-def))
|
|
(help-substitute-command-keys doc-raw 'ADD-HELP-BUTTONS)
|
|
(condition-case err
|
|
(help-documentation function nil 'ADD-HELP-BUTTONS)
|
|
(error (format "No Doc! %S" err))))))
|
|
(help-fns--key-bindings function)
|
|
(with-current-buffer standard-output
|
|
(setq doc (condition-case nil
|
|
;; Maybe `help-fns--signature' should return `doc' for invalid functions, not signal error.
|
|
(help-fns--signature function doc-raw (if (subrp def) (indirect-function real-def) real-def)
|
|
real-function key-bind-buf)
|
|
((invalid-function void-function) doc-raw))) ; E.g., an alias for a not yet defined function.
|
|
(run-hook-with-args 'help-fns-describe-function-functions function)
|
|
(insert "\n")
|
|
(when (and doc (boundp 'Info-virtual-files))
|
|
(Info-make-manuals-xref function)) ; Link to manuals. (With progress message.)
|
|
(insert (or doc "Not documented."))
|
|
(when (or (function-get function 'pure) (function-get function 'side-effect-free))
|
|
(insert "\nThis function does not change global state, including the match data."))
|
|
;; Avoid asking user questions if she decides to save help buffer, when locale's codeset is not UTF-8.
|
|
(unless (memq text-quoting-style '(straight grave)) (set-buffer-file-coding-system 'utf-8)))))
|
|
|
|
)
|
|
|
|
|
|
;;;###autoload
|
|
(defun describe-command (function) ; Bound to `C-h c'
|
|
"Describe an Emacs command (interactive function).
|
|
Equivalent to using a prefix arg with `describe-function'.
|
|
|
|
If you use Icicles then in Icicle mode keys bound to the commands are
|
|
shown next to them in `*Completions*. You can toggle this keys
|
|
display on/off using `C-x C-a'."
|
|
(interactive
|
|
(let ((fn (or (and (fboundp 'symbol-nearest-point) (symbol-nearest-point))
|
|
(function-called-at-point)))
|
|
(enable-recursive-minibuffers t)
|
|
(completion-annotate-function (and (boundp 'icicle-mode) icicle-mode
|
|
(lambda (cand)
|
|
(with-current-buffer icicle-pre-minibuffer-buffer
|
|
(and (setq cand (intern-soft cand)) (symbolp cand)
|
|
(let ((key (where-is-internal cand nil t)))
|
|
(and key
|
|
(format " %s" (icicle-key-description key)))))))))
|
|
val)
|
|
(setq val (completing-read
|
|
(format "Describe command%s: " (if (commandp fn) (format " (default %s)" fn) ""))
|
|
obarray 'commandp t nil nil (and fn (commandp fn) (symbol-name fn))))
|
|
(list (if (equal val "") fn (intern val)))))
|
|
(describe-function function t))
|
|
|
|
|
|
;; REPLACE ORIGINAL in `help.el':
|
|
;;
|
|
;; 1. With a prefix argument, candidates are user variables (options) only.
|
|
;; 2. Preferred default candidate is `symbol-nearest-point'.
|
|
;; 3. Remove initial `*' from doc string (indicates it is a user variable).
|
|
;; 4. PREDICATE to `completing-read' uses original buffer (not minibuffer), when testing `boundp'. (BUG #21252)
|
|
;; 5. Use `substitute-command-keys' on doc string.
|
|
;; 6. Preserve text properties.
|
|
;; 7. No message if not called interactively.
|
|
;;
|
|
(when (< emacs-major-version 23)
|
|
|
|
(defun describe-variable (variable &optional buffer optionp)
|
|
"Display the full documentation of VARIABLE (a symbol).
|
|
VARIABLE names an Emacs Lisp variable, possibly a user option.
|
|
With a prefix argument, candidates are user variables (options) only.
|
|
Default candidate is the `symbol-nearest-point'.
|
|
Return the documentation, as a string.
|
|
If VARIABLE has a buffer-local value in BUFFER (default to the current buffer),
|
|
it is displayed along with the global value."
|
|
(interactive
|
|
(let ((symb (or (and (fboundp 'symbol-nearest-point) (symbol-nearest-point))
|
|
(variable-at-point)))
|
|
(enable-recursive-minibuffers t)
|
|
(completion-annotate-function (lambda (var) (and (custom-variable-p (intern-soft var)) " (option)")))
|
|
(curbuf (current-buffer))
|
|
val)
|
|
(when (numberp symb) (setq symb nil)) ; `variable-at-point' returns 0 when there is no var.
|
|
(setq val (completing-read
|
|
(format "Describe variable%s: "
|
|
(if (and symb (boundp symb)) (format " (default %s)" symb) ""))
|
|
obarray (if current-prefix-arg
|
|
`(lambda (vv) (with-current-buffer ',curbuf (user-variable-p vv)))
|
|
`(lambda (vv) (with-current-buffer ',curbuf
|
|
(or (boundp vv) (get vv 'variable-documentation)))))
|
|
t nil nil (and (symbolp symb) (boundp symb) (symbol-name symb))))
|
|
(list (if (equal val "") symb (intern val))
|
|
nil
|
|
current-prefix-arg)))
|
|
(unless (buffer-live-p buffer) (setq buffer (current-buffer)))
|
|
(if (not (symbolp variable))
|
|
(when (interactive-p) (message "You did not specify a variable"))
|
|
(unless (or (not optionp) (user-variable-p variable))
|
|
(error "Not a defined Emacs user option: `%s'" variable))
|
|
;;$$ (unless (boundp variable) (error "Not a defined Emacs variable: `%s'" variable))
|
|
(save-excursion
|
|
(let* ((valvoid (not (with-current-buffer buffer (boundp variable))))
|
|
;; Extract the value before setting up the output buffer,
|
|
;; in case `buffer' *is* the output buffer.
|
|
(val (and (not valvoid) (buffer-local-value variable buffer)))
|
|
val-start-pos)
|
|
(help-setup-xref (list #'describe-variable variable buffer) (interactive-p))
|
|
(with-output-to-temp-buffer (help-buffer)
|
|
(with-current-buffer buffer
|
|
(prin1 variable)
|
|
;; Make a hyperlink to the library if appropriate. (Don't change the format of the
|
|
;; buffer's initial line in case anything expects the current format.)
|
|
(let ((file-name (symbol-file variable 'defvar)))
|
|
(setq file-name (describe-simplify-lib-file-name file-name))
|
|
(when (equal file-name "loaddefs.el")
|
|
;; Find the real def site of the preloaded variable.
|
|
(let ((location (condition-case nil
|
|
(find-variable-noselect variable file-name)
|
|
(error nil))))
|
|
(when location
|
|
(with-current-buffer (car location)
|
|
(when (cdr location) (goto-char (cdr location)))
|
|
(when (re-search-backward "^;;; Generated autoloads from \\(.*\\)" nil t)
|
|
(setq file-name (match-string 1)))))))
|
|
(when (and (null file-name) (integerp (get variable 'variable-documentation)))
|
|
;; It's a var not defined in Elisp but in C.
|
|
(setq file-name (if (get-buffer " *DOC*")
|
|
(help-C-file-name variable 'var)
|
|
'C-source)))
|
|
(if file-name
|
|
(progn (princ " is a variable defined in `")
|
|
(princ (if (eq file-name 'C-source) "C source code" file-name))
|
|
(princ "'.\n")
|
|
(with-current-buffer standard-output
|
|
(save-excursion
|
|
(re-search-backward "`\\([^`']+\\)'" nil t)
|
|
(help-xref-button 1 'help-variable-def variable file-name)))
|
|
(if valvoid (princ "It is void as a variable.") (princ "Its ")))
|
|
(if valvoid (princ " is void as a variable.") (princ "'s "))))
|
|
(unless valvoid
|
|
(with-current-buffer standard-output
|
|
(setq val-start-pos (point))
|
|
(princ "value is ") (terpri)
|
|
(let ((from (point)))
|
|
(pp val)
|
|
;; Hyperlinks in variable's value are quite frequently inappropriate
|
|
;; e.g `C-h v <RET> features <RET>'
|
|
;; (help-xref-on-pp from (point))
|
|
(when (< (point) (+ from 20)) (delete-region (1- from) from)))))
|
|
(terpri)
|
|
(when (local-variable-p variable)
|
|
(princ (format "%socal in buffer %s; "
|
|
(if (get variable 'permanent-local) "Permanently l" "L")
|
|
(buffer-name)))
|
|
(if (not (default-boundp variable))
|
|
(princ "globally void")
|
|
(let ((val (default-value variable)))
|
|
(with-current-buffer standard-output
|
|
(princ "global value is ") (terpri)
|
|
;; Fixme: `pp' can take an age if you happen to ask for a very large expression.
|
|
;; We should probably print it raw once and check it's a sensible size before
|
|
;; prettyprinting. -- fx
|
|
(let ((from (point)))
|
|
(pp val)
|
|
;; See previous comment for this function.
|
|
;; (help-xref-on-pp from (point))
|
|
(when (< (point) (+ from 20)) (delete-region (1- from) from)))))))
|
|
;; Add a note for variables that have been `make-var-buffer-local'.
|
|
(when (and (local-variable-if-set-p variable)
|
|
(or (not (local-variable-p variable))
|
|
(with-temp-buffer (local-variable-if-set-p variable))))
|
|
(princ "\nAutomatically becomes buffer-local when set in any fashion.\n"))
|
|
(terpri)
|
|
(with-current-buffer standard-output ; If the value is large, move it to the end.
|
|
(when (> (count-lines (point-min) (point-max)) 10)
|
|
;; Note that setting the syntax table like below makes `forward-sexp' move over a
|
|
;; `'s' at the end of a symbol.
|
|
(set-syntax-table emacs-lisp-mode-syntax-table)
|
|
(goto-char val-start-pos)
|
|
;; The line below previously read as
|
|
;; (delete-region (point) (progn (end-of-line) (point)))
|
|
;; which suppressed display of the buffer local value for large values.
|
|
(when (looking-at "value is") (replace-match ""))
|
|
(save-excursion (insert "\n\nValue:")
|
|
(set (make-local-variable 'help-button-cache) (point-marker)))
|
|
(insert "value is shown ")
|
|
(insert-button "below" 'action help-button-cache 'follow-link t
|
|
'help-echo "mouse-2, RET: show value")
|
|
(insert ".\n")))
|
|
;; Mention if it's an alias
|
|
(let* ((alias (condition-case nil (indirect-variable variable) (error variable)))
|
|
(obsolete (get variable 'byte-obsolete-variable))
|
|
(safe-var (get variable 'safe-local-variable))
|
|
(doc (or (documentation-property variable 'variable-documentation)
|
|
(documentation-property alias 'variable-documentation))))
|
|
(when (and (> (length doc) 1) (eq ?* (elt doc 0)))
|
|
(setq doc (substring doc 1))) ; Remove any user-variable prefix `*'.
|
|
(unless (eq alias variable)
|
|
(princ (format "\nThis variable is an alias for `%s'.\n" alias)))
|
|
(when (or obsolete safe-var) (terpri))
|
|
(when obsolete
|
|
(princ "This variable is obsolete")
|
|
(when (cdr obsolete) (princ (format " since %s" (cdr obsolete))))
|
|
(princ ";") (terpri)
|
|
(princ (if (stringp (car obsolete))
|
|
(car obsolete)
|
|
(format "use `%s' instead." (car obsolete))))
|
|
(terpri))
|
|
(when safe-var
|
|
(princ "This variable is safe as a file local variable ")
|
|
(princ "if its value\nsatisfies the predicate ")
|
|
(princ (if (byte-code-function-p safe-var)
|
|
"which is byte-compiled expression.\n"
|
|
(format "`%s'.\n" safe-var))))
|
|
(princ "\nDocumentation:\n")
|
|
;; Use `insert', not `princ', to keep text properties.
|
|
;; Was: (princ (or doc "Not documented as a variable.")))
|
|
(with-current-buffer standard-output
|
|
(insert (or (substitute-command-keys doc) "Not documented as a variable."))))
|
|
;; Make a link to customize if this variable can be customized.
|
|
(when (custom-variable-p variable)
|
|
(let ((customize-label "customize"))
|
|
(terpri) (terpri) (princ (concat "You can " customize-label " this variable."))
|
|
(with-current-buffer standard-output
|
|
(save-excursion (re-search-backward (concat "\\(" customize-label "\\)") nil t)
|
|
(help-xref-button 1 'help-customize-variable variable)))))
|
|
(print-help-return-message)
|
|
(with-current-buffer standard-output (buffer-string)))))))) ; Return the text displayed.
|
|
)
|
|
|
|
;;; This macro is no different from what is in vanilla Emacs 23+.
|
|
;;; Add it here so this file can be byte-compiled with Emacs 22 and used with Emacs 23+.
|
|
(defmacro with-selected-frame (frame &rest body)
|
|
"Execute the forms in BODY with FRAME as the selected frame.
|
|
Save the selected frame, select FRAME, execute BODY, then restore the
|
|
originally selected frame. Return the value of the last form in BODY.
|
|
|
|
This macro changes the order of neither the recently selected windows
|
|
nor the buffers in the buffer list. See also `with-temp-buffer'."
|
|
(declare (indent 1) (debug t))
|
|
(let ((old-frame (make-symbol "old-frame"))
|
|
(old-buffer (make-symbol "old-buffer")))
|
|
`(let ((,old-frame (selected-frame))
|
|
(,old-buffer (current-buffer)))
|
|
(unwind-protect
|
|
(progn (if (> emacs-major-version 22) (select-frame ,frame 'NORECORD) (select-frame ,frame))
|
|
,@body)
|
|
(when (frame-live-p ,old-frame)
|
|
(if (> emacs-major-version 22) (select-frame ,old-frame 'NORECORD) (select-frame ,old-frame)))
|
|
(when (buffer-live-p ,old-buffer) (set-buffer ,old-buffer))))))
|
|
|
|
|
|
;; REPLACE ORIGINAL in `help.el':
|
|
;;
|
|
;; 1. With a prefix argument, candidates are user variables (options) only.
|
|
;; 2. Preferred default candidate is `symbol-nearest-point'.
|
|
;; 3. PREDICATE to `completing-read' uses original buffer (not minibuffer), when testing `boundp'. (BUG #21252)
|
|
;; 4. Preserve text properties.
|
|
;; 5. Remove initial `*' from doc string (indicates it is a user variable).
|
|
;; 6. Call `Info-make-manuals-xref' to create a cross-ref link to manuals (Emacs 23.3).
|
|
;; 7. Add key-description buttons to command help. Use `insert', not `princ'.
|
|
;; 8. No no-function message if not called interactively.
|
|
;;
|
|
(when (= emacs-major-version 23)
|
|
|
|
(defun describe-variable (variable &optional buffer frame optionp)
|
|
"Display the full documentation of VARIABLE (a symbol).
|
|
VARIABLE names an Emacs Lisp variable, possibly a user option.
|
|
With a prefix argument, candidates are user variables (options) only.
|
|
Default candidate is the `symbol-nearest-point'.
|
|
Return the documentation, as a string.
|
|
If VARIABLE has a buffer-local value in BUFFER or FRAME
|
|
\(default to the current buffer and current frame),
|
|
it is displayed along with the global value."
|
|
(interactive
|
|
(let ((symb (or (and (fboundp 'symbol-nearest-point) (symbol-nearest-point))
|
|
(variable-at-point)))
|
|
(enable-recursive-minibuffers t)
|
|
(completion-annotate-function (lambda (var) (and (custom-variable-p (intern-soft var)) " (option)")))
|
|
(curbuf (current-buffer))
|
|
val)
|
|
(when (numberp symb) (setq symb nil)) ; `variable-at-point' returns 0 when there is no var.
|
|
(setq val (completing-read
|
|
(format "Describe variable%s: "
|
|
(if (and symb (boundp symb)) (format " (default %s)" symb) ""))
|
|
obarray
|
|
(if current-prefix-arg
|
|
`(lambda (vv) (with-current-buffer ',curbuf (user-variable-p vv)))
|
|
`(lambda (vv) (with-current-buffer ',curbuf
|
|
(or (get vv 'variable-documentation) (and (boundp vv) (not (keywordp vv)))))))
|
|
t nil nil (and (symbolp symb) (boundp symb) (symbol-name symb))))
|
|
(list (if (equal val "") symb (intern val))
|
|
nil
|
|
nil
|
|
current-prefix-arg)))
|
|
(let (file-name)
|
|
(unless (buffer-live-p buffer) (setq buffer (current-buffer)))
|
|
(unless (frame-live-p frame) (setq frame (selected-frame)))
|
|
(if (not (symbolp variable))
|
|
(when (if (or (> emacs-major-version 23) ; Emacs 23.1 `called-interactively' accepts no arg.
|
|
(and (= emacs-major-version 23) (> emacs-minor-version 1)))
|
|
(called-interactively-p 'interactive)
|
|
(interactive-p))
|
|
(message "You did not specify a variable"))
|
|
(unless (or (not optionp) (user-variable-p variable))
|
|
(error "Not a defined Emacs user option: `%s'" variable))
|
|
;;$$ (unless (boundp variable) (error "Not a defined Emacs variable: `%s'" variable))
|
|
(save-excursion
|
|
(let ((valvoid (not (with-current-buffer buffer (boundp variable))))
|
|
val val-start-pos locus)
|
|
;; Extract the value before setting up the output buffer, in case BUFFER *is* the
|
|
;; output buffer.
|
|
(unless valvoid
|
|
(with-selected-frame frame
|
|
(with-current-buffer buffer
|
|
(setq val (symbol-value variable)
|
|
locus (variable-binding-locus variable)))))
|
|
(help-setup-xref (list #'describe-variable variable buffer)
|
|
(if (or (> emacs-major-version 23)
|
|
(and (= emacs-major-version 23) (> emacs-minor-version 1)))
|
|
(called-interactively-p 'interactive)
|
|
(interactive-p)))
|
|
(with-help-window (help-buffer)
|
|
(with-current-buffer buffer
|
|
(prin1 variable)
|
|
(setq file-name (find-lisp-object-file-name variable 'defvar))
|
|
(if file-name
|
|
(progn (princ " is a variable defined in `")
|
|
(princ (if (eq file-name 'C-source)
|
|
"C source code"
|
|
(file-name-nondirectory file-name)))
|
|
(princ "'.\n")
|
|
(with-current-buffer standard-output
|
|
(save-excursion
|
|
(re-search-backward "`\\([^`']+\\)'" nil t)
|
|
(help-xref-button 1 'help-variable-def variable file-name)))
|
|
(if valvoid (princ "It is void as a variable.") (princ "Its ")))
|
|
(if valvoid (princ " is void as a variable.") (princ "'s "))))
|
|
(unless valvoid
|
|
(with-current-buffer standard-output
|
|
(setq val-start-pos (point))
|
|
(princ "value is ")
|
|
(let ((from (point)))
|
|
(terpri)
|
|
(pp val)
|
|
(if (< (point) (+ 68 (line-beginning-position 0)))
|
|
(delete-region from (1+ from))
|
|
(delete-region (1- from) from)))))
|
|
(terpri)
|
|
(when locus
|
|
(if (bufferp locus)
|
|
(princ (format "%socal in buffer %s; "
|
|
(if (get variable 'permanent-local) "Permanently l" "L")
|
|
(buffer-name)))
|
|
(princ (format "It is a frame-local variable; ")))
|
|
(if (not (default-boundp variable))
|
|
(princ "globally void")
|
|
(let ((val (default-value variable)))
|
|
(with-current-buffer standard-output
|
|
(princ "global value is ") (terpri)
|
|
;; Fixme: `pp' can take an age if you happen to ask for a very large expression.
|
|
;; We should probably print it raw once and check it's a sensible size before
|
|
;; prettyprinting. -- fx
|
|
(let ((from (point)))
|
|
(pp val)
|
|
;; See previous comment for this function.
|
|
;; (help-xref-on-pp from (point))
|
|
(when (< (point) (+ from 20)) (delete-region (1- from) from))))))
|
|
(terpri))
|
|
(with-current-buffer standard-output ; If the value is large, move it to the end.
|
|
(when (> (count-lines (point-min) (point-max)) 10)
|
|
;; Note that setting the syntax table like below makes `forward-sexp' move over a
|
|
;; `'s' at the end of a symbol.
|
|
(set-syntax-table emacs-lisp-mode-syntax-table)
|
|
(goto-char val-start-pos)
|
|
;; The line below previously read as
|
|
;; (delete-region (point) (progn (end-of-line) (point)))
|
|
;; which suppressed display of the buffer local value for large values.
|
|
(when (looking-at "value is") (replace-match ""))
|
|
(save-excursion (insert "\n\nValue:") (terpri)
|
|
(set (make-local-variable 'help-button-cache) (point-marker)))
|
|
(insert "value is shown ")
|
|
(insert-button "below" 'action help-button-cache 'follow-link t
|
|
'help-echo "mouse-2, RET: show value")
|
|
(insert ".\n")))
|
|
(terpri)
|
|
(let* ((alias (condition-case nil (indirect-variable variable) (error variable)))
|
|
(obsolete (get variable 'byte-obsolete-variable))
|
|
(use (car obsolete))
|
|
(safe-var (get variable 'safe-local-variable))
|
|
(vardoc (help-documentation-property variable 'variable-documentation
|
|
nil 'ADD-HELP-BUTTONS))
|
|
(vardoc (and (not (equal "" vardoc)) vardoc))
|
|
(doc (or vardoc (help-documentation-property alias 'variable-documentation
|
|
nil 'ADD-HELP-BUTTONS)))
|
|
(extra-line nil))
|
|
(when (and (> (length doc) 1) (eq ?* (elt doc 0)))
|
|
(setq doc (substring doc 1))) ; Remove any user-variable prefix `*'.
|
|
;; Add a note for variables that have been `make-var-buffer-local'.
|
|
(when (and (local-variable-if-set-p variable)
|
|
(or (not (local-variable-p variable))
|
|
(with-temp-buffer (local-variable-if-set-p variable))))
|
|
(setq extra-line t)
|
|
(princ " Automatically becomes buffer-local when set in any fashion.\n"))
|
|
;; Mention if it's an alias
|
|
(unless (eq alias variable)
|
|
(setq extra-line t)
|
|
(princ (format " This variable is an alias for `%s'.\n" alias)))
|
|
(when obsolete
|
|
(setq extra-line t)
|
|
(princ " This variable is obsolete")
|
|
(when (cdr obsolete) (princ (format " since %s" (cdr obsolete))))
|
|
(princ (cond ((stringp use) (concat ";\n " use))
|
|
(use (format ";\n use `%s' instead." (car obsolete)))
|
|
(t ".")))
|
|
(terpri))
|
|
(when (member (cons variable val) file-local-variables-alist)
|
|
(setq extra-line t)
|
|
(if (member (cons variable val) dir-local-variables-alist)
|
|
(let ((file (and (buffer-file-name)
|
|
(not (file-remote-p (buffer-file-name)))
|
|
(dir-locals-find-file (buffer-file-name)))))
|
|
(princ " This variable is a directory local variable")
|
|
(when file
|
|
(princ (concat "\n from the file \"" (if (consp file) (car file) file)
|
|
"\"")))
|
|
(princ ".\n"))
|
|
(princ " This variable is a file local variable.\n")))
|
|
(when (memq variable ignored-local-variables)
|
|
(setq extra-line t)
|
|
(princ " This variable is ignored when used as a file local \
|
|
variable.\n"))
|
|
;; Can be both risky and safe, eg `auto-fill-function'.
|
|
(when (risky-local-variable-p variable)
|
|
(setq extra-line t)
|
|
(princ " This variable is potentially risky when used as a \
|
|
file local variable.\n")
|
|
(when (assq variable safe-local-variable-values)
|
|
(princ " However, you have added it to \
|
|
`safe-local-variable-values'.\n")))
|
|
(when safe-var
|
|
(setq extra-line t)
|
|
(princ " This variable is safe as a file local variable ")
|
|
(princ "if its value\n satisfies the predicate ")
|
|
(princ (if (byte-code-function-p safe-var)
|
|
"which is byte-compiled expression.\n"
|
|
(format "`%s'.\n" safe-var))))
|
|
(when extra-line (terpri))
|
|
(princ "Documentation:\n")
|
|
(with-current-buffer standard-output
|
|
(insert (or doc "Not documented as a variable."))))
|
|
;; Make a link to customize if this variable can be customized.
|
|
(when (custom-variable-p variable)
|
|
(let ((customize-label "customize"))
|
|
(terpri) (terpri)
|
|
(princ (concat "You can " customize-label " this variable."))
|
|
(with-current-buffer standard-output
|
|
(save-excursion (re-search-backward (concat "\\(" customize-label "\\)") nil t)
|
|
(help-xref-button 1 'help-customize-variable variable))))
|
|
;; Note variable's version or package version
|
|
(let ((output (describe-variable-custom-version-info variable)))
|
|
(when output (terpri) (terpri) (princ output))))
|
|
(when (boundp 'Info-virtual-files) ; Emacs 23.2+
|
|
(unless valvoid
|
|
(with-current-buffer standard-output ; Link to manuals.
|
|
(Info-make-manuals-xref variable nil nil
|
|
(not (if (or (> emacs-major-version 23)
|
|
(and (= emacs-major-version 23)
|
|
(> emacs-minor-version 1)))
|
|
(called-interactively-p 'interactive)
|
|
(interactive-p)))))))
|
|
(with-current-buffer standard-output (buffer-string)))))))) ; Return the text displayed.
|
|
)
|
|
|
|
|
|
;; REPLACE ORIGINAL in `help-fns.el':
|
|
;;
|
|
;; 1. With a prefix argument, candidates are user variables (options) only.
|
|
;; 2. Preferred default candidate is `symbol-nearest-point'.
|
|
;; 3. PREDICATE to `completing-read' uses original buffer (not minibuffer), when testing `boundp'. (BUG #21252)
|
|
;; 4. Preserve text properties.
|
|
;; 5. Remove initial `*' from doc string (indicates it is a user variable).
|
|
;; 6. Call `Info-make-manuals-xref' to create a cross-ref link to manuals (Emacs 23.3).
|
|
;; 7. Add key-description buttons to command help. Use `insert', not `princ'.
|
|
;; 8. No no-function message if not called interactively.
|
|
;;
|
|
(when (> emacs-major-version 23)
|
|
|
|
(defface describe-variable-value '((((background dark)) (:foreground "#58DFFA4FFFFF")) ; a dark cyan
|
|
(t (:foreground "Firebrick")))
|
|
"*Face used to highlight the variable value, for `describe-variable'."
|
|
:group 'help :group 'faces)
|
|
|
|
(defun describe-variable (variable &optional buffer frame optionp)
|
|
"Display the full documentation of VARIABLE (a symbol).
|
|
With a prefix argument, candidates are user variables (options) only.
|
|
Default candidate is the `symbol-nearest-point'.
|
|
Return the documentation, as a string.
|
|
|
|
VARIABLE names an Emacs Lisp variable, possibly a user option.
|
|
If VARIABLE has a buffer-local value in BUFFER or FRAME (default to
|
|
the current buffer and current frame) then it is displayed, along with
|
|
the global value."
|
|
(interactive
|
|
(let ((symb (or (and (fboundp 'symbol-nearest-point) (symbol-nearest-point))
|
|
(variable-at-point)))
|
|
(enable-recursive-minibuffers t)
|
|
(completion-annotate-function (lambda (vv) (and (custom-variable-p (intern-soft vv)) " (option)")))
|
|
(curbuf (current-buffer))
|
|
val)
|
|
(when (numberp symb) (setq symb nil)) ; `variable-at-point' returns 0 when there is no var.
|
|
(setq val (completing-read
|
|
(format "Describe variable%s: "
|
|
(if (and symb (boundp symb)) (format " (default %s)" symb) ""))
|
|
obarray
|
|
(if current-prefix-arg
|
|
`(lambda (vv) (with-current-buffer ',curbuf (user-variable-p vv)))
|
|
`(lambda (vv) (with-current-buffer ',curbuf
|
|
(or (get vv 'variable-documentation) (and (boundp vv) (not (keywordp vv)))))))
|
|
t nil nil (and (symbolp symb) (boundp symb) (symbol-name symb))))
|
|
(list (if (equal val "") symb (intern val))
|
|
nil
|
|
nil
|
|
current-prefix-arg)))
|
|
(let (file-name)
|
|
(unless (buffer-live-p buffer) (setq buffer (current-buffer)))
|
|
(unless (frame-live-p frame) (setq frame (selected-frame)))
|
|
(if (not (symbolp variable))
|
|
(when (called-interactively-p 'interactive) (message "You did not specify a variable"))
|
|
(unless (or (not optionp) (user-variable-p variable))
|
|
(error "Not a defined Emacs user option: `%s'" variable))
|
|
;;$$ (unless (boundp variable) (error "Not a defined Emacs variable: `%s'" variable))
|
|
(save-excursion
|
|
(let ((valvoid (not (with-current-buffer buffer (boundp variable))))
|
|
(permanent-local (get variable 'permanent-local))
|
|
val val-start-pos locus)
|
|
;; Extract the value before setting up the output buffer, in case BUFFER *is* the output buffer.
|
|
(unless valvoid
|
|
(with-selected-frame frame
|
|
(with-current-buffer buffer
|
|
(setq val (symbol-value variable)
|
|
locus (variable-binding-locus variable)))))
|
|
(help-setup-xref (list #'describe-variable variable buffer)
|
|
(called-interactively-p 'interactive))
|
|
(with-help-window (help-buffer)
|
|
(with-current-buffer buffer
|
|
(prin1 variable)
|
|
(setq file-name (find-lisp-object-file-name variable 'defvar))
|
|
(if file-name
|
|
(progn (princ " is a variable defined in `")
|
|
(princ (if (eq file-name 'C-source)
|
|
"C source code"
|
|
(file-name-nondirectory file-name)))
|
|
(princ "'.\n")
|
|
(with-current-buffer standard-output
|
|
(save-excursion (re-search-backward "`\\([^`']+\\)'" nil t)
|
|
(help-xref-button 1 'help-variable-def variable file-name)))
|
|
(if valvoid (princ "It is void as a variable.") (princ "Its ")))
|
|
(if valvoid (princ " is void as a variable.") (princ "'s "))))
|
|
(unless valvoid
|
|
(with-current-buffer standard-output
|
|
(setq val-start-pos (point))
|
|
(princ "value is ")
|
|
(let ((from (point))
|
|
(line-beg (line-beginning-position))
|
|
(print-rep (let ((print-quoted t)) (prin1-to-string val))))
|
|
(if (< (+ (length print-rep) (point) (- line-beg)) 68)
|
|
(progn (insert print-rep)
|
|
(put-text-property from (point) 'face 'describe-variable-value)
|
|
(terpri))
|
|
(terpri)
|
|
(unless (or (numberp val) (symbolp val) (characterp val)
|
|
(and (stringp val) (string-match-p "[\n]" val)))
|
|
(terpri))
|
|
(let ((opoint (point)))
|
|
(pp val)
|
|
(save-excursion (fill-region-as-paragraph opoint (point) nil t t)))
|
|
(when (stringp val) (terpri))
|
|
(put-text-property from (point) 'face 'describe-variable-value)
|
|
(if (< (point) (+ 68 (line-beginning-position 0)))
|
|
(delete-region from (1+ from))
|
|
(delete-region (1- from) from)))
|
|
(let* ((sv (get variable 'standard-value))
|
|
(origval (and (consp sv)
|
|
(condition-case nil (eval (car sv)) (error :help-eval-error)))))
|
|
(when (and (consp sv)
|
|
(not (equal origval val))
|
|
(not (equal origval :help-eval-error)))
|
|
(princ "\nOriginal value was \n")
|
|
(setq from (point))
|
|
(unless (or (numberp origval) (symbolp origval) (characterp origval)
|
|
(and (stringp origval) (string-match-p "[\n]" origval)))
|
|
(terpri))
|
|
(let ((opoint (point)))
|
|
(pp origval)
|
|
(save-excursion (fill-region-as-paragraph opoint (point) nil t t)))
|
|
(put-text-property from (point) 'face 'describe-variable-value)
|
|
(when (< (point) (+ from 20)) (delete-region (1- from) from) (terpri)))))))
|
|
(terpri)
|
|
(when locus
|
|
(cond ((bufferp locus)
|
|
(terpri)
|
|
(princ (format "%socal in buffer `%s'; "
|
|
(if (get variable 'permanent-local) "Permanently l" "L")
|
|
(buffer-name buffer))))
|
|
((framep locus)
|
|
(princ (format "It is a frame-local variable; ")))
|
|
((terminal-live-p locus)
|
|
(princ (format "It is a terminal-local variable; ")))
|
|
(t (princ (format "It is local to %S" locus))))
|
|
(if (not (default-boundp variable))
|
|
(progn (princ "globally void") (terpri))
|
|
(let ((global-val (default-value variable)))
|
|
(with-current-buffer standard-output
|
|
(princ "global value is")
|
|
(if (eq val global-val)
|
|
(progn (princ " the same.") (terpri))
|
|
(princ ":") (terpri) (terpri)
|
|
;; Fixme: `pp' can take an age if you happen to ask for a very large expression.
|
|
;; We should probably print it raw once and check whether it is a sensible size,
|
|
;; before prettyprinting. -- fx
|
|
(let ((opoint (point)))
|
|
(pp global-val)
|
|
(save-excursion (fill-region-as-paragraph opoint (point) nil t t))
|
|
(put-text-property opoint (point) 'face 'describe-variable-value)
|
|
;; See previous comment for this function. (help-xref-on-pp opoint (point))
|
|
(when (< (point) (+ opoint 20)) (delete-region (1- opoint) opoint))))))))
|
|
(with-current-buffer standard-output ; If the value is large, move it to the end.
|
|
(when (> (count-lines (point-min) (point-max)) 10)
|
|
;; Note that setting the syntax table like below makes `forward-sexp' move over a
|
|
;; `'s' at the end of a symbol.
|
|
(set-syntax-table emacs-lisp-mode-syntax-table)
|
|
(goto-char val-start-pos)
|
|
;; The line below previously read as (delete-region (point) (progn (end-of-line) (point))),
|
|
;; which suppressed display of the buffer local value for large values.
|
|
(when (looking-at "value is") (replace-match ""))
|
|
(save-excursion (let ((nl-before (cond ((and (eq ?\n (char-before)) ; vs `looking-back'.
|
|
(eq ?\n (char-before (1- (point))))) "")
|
|
((eq ?\n (char-before)) "\n")
|
|
(t "\n\n")))
|
|
(nl-after (cond ((looking-at "[\n]") "")
|
|
(t "\n"))))
|
|
(insert (format "%sValue:%s" nl-before nl-after)))
|
|
(set (make-local-variable 'help-button-cache) (point-marker)))
|
|
(insert "value is shown ")
|
|
(insert-button "below" 'action help-button-cache 'follow-link t
|
|
'help-echo "mouse-2, RET: show value")
|
|
(insert ".\n")))
|
|
(terpri)
|
|
(let* ((alias (condition-case nil (indirect-variable variable) (error variable)))
|
|
(obsolete (get variable 'byte-obsolete-variable))
|
|
(use (car obsolete))
|
|
(safe-var (get variable 'safe-local-variable))
|
|
(vardoc (help-documentation-property variable 'variable-documentation
|
|
nil 'ADD-HELP-BUTTONS))
|
|
(vardoc (and (not (equal "" vardoc)) vardoc))
|
|
(doc (or vardoc (help-documentation-property alias 'variable-documentation
|
|
nil 'ADD-HELP-BUTTONS)))
|
|
(extra-line nil))
|
|
(when (and (> (length doc) 1) (eq ?* (elt doc 0)))
|
|
(setq doc (substring doc 1))) ; Remove any user-variable prefix `*'.
|
|
(cond ((and (local-variable-if-set-p variable) ; Mention if it's a local variable.
|
|
(or (not (local-variable-p variable))
|
|
(with-temp-buffer (local-variable-if-set-p variable))))
|
|
(setq extra-line t)
|
|
(princ " Automatically becomes ")
|
|
(when permanent-local (princ "permanently "))
|
|
(princ "buffer-local when set.\n"))
|
|
((not permanent-local))
|
|
((bufferp locus) (princ " This variable's buffer-local value is permanent.\n"))
|
|
(t (princ " This variable's value is permanent when it is given a local binding.\n")))
|
|
(unless (eq alias variable) ; Mention if it's an alias.
|
|
(setq extra-line t)
|
|
(princ (format " This variable is an alias for `%s'.\n" alias)))
|
|
(when obsolete
|
|
(setq extra-line t)
|
|
(princ " This variable is obsolete")
|
|
(when (nth 2 obsolete) (princ (format " since %s" (nth 2 obsolete))))
|
|
(princ (cond ((stringp use) (concat ";\n " use))
|
|
(use (format ";\n use `%s' instead." (car obsolete)))
|
|
(t ".")))
|
|
(terpri))
|
|
(when (member (cons variable val) file-local-variables-alist)
|
|
(setq extra-line t)
|
|
(if (member (cons variable val) dir-local-variables-alist)
|
|
(let ((file (and (buffer-file-name)
|
|
(not (file-remote-p (buffer-file-name)))
|
|
(dir-locals-find-file (buffer-file-name))))
|
|
(dir-file t))
|
|
(princ " This variable's value is directory-local")
|
|
(if (null file)
|
|
(princ ".\n")
|
|
(princ ", set ")
|
|
(when (consp file) ; When result is from cache...
|
|
(if (nth 2 file) ; If cache element has an mtime, assume it came from a file.
|
|
(setq file (expand-file-name dir-locals-file (car file)))
|
|
(setq dir-file nil))) ; Otherwise, assume it was set directly.
|
|
(princ (if dir-file "by the file\n `" "for the directory\n `"))
|
|
(with-current-buffer standard-output
|
|
(insert-text-button file 'type 'help-dir-local-var-def
|
|
'help-args (list variable file)))
|
|
(princ "'.\n")))
|
|
(princ " This variable's value is file-local.\n")))
|
|
(when (memq variable ignored-local-variables)
|
|
(setq extra-line t)
|
|
(princ " This variable is ignored when used as a file-local variable.\n"))
|
|
(when (risky-local-variable-p variable) ; Can be both risky & safe, eg `auto-fill-function'.
|
|
(setq extra-line t)
|
|
(princ " This variable can be risky when used as a file-local variable.\n")
|
|
(when (assq variable safe-local-variable-values)
|
|
(princ " However, it has been added to `safe-local-variable-values'.\n")))
|
|
(when safe-var
|
|
(setq extra-line t)
|
|
(princ " This variable is safe as a file local variable ")
|
|
(princ "if its value\n satisfies the predicate ")
|
|
(princ (if (byte-code-function-p safe-var)
|
|
"which is a byte-compiled expression.\n"
|
|
(format "`%s'.\n" safe-var))))
|
|
(when extra-line (terpri))
|
|
(princ "Documentation:\n")
|
|
(with-current-buffer standard-output (insert (or doc "Not documented as a variable."))))
|
|
;; Make a link to customize if this variable can be customized.
|
|
(when (custom-variable-p variable)
|
|
(let ((customize-label "customize"))
|
|
(terpri) (terpri)
|
|
(princ (concat "You can " customize-label " this variable."))
|
|
(with-current-buffer standard-output
|
|
(save-excursion (re-search-backward (concat "\\(" customize-label "\\)") nil t)
|
|
(help-xref-button 1 'help-customize-variable variable))))
|
|
;; Note variable's version or package version
|
|
(let ((output (describe-variable-custom-version-info variable)))
|
|
(when output (terpri) (terpri) (princ output))))
|
|
(unless valvoid
|
|
(with-current-buffer standard-output ; Link to manuals.
|
|
(Info-make-manuals-xref variable nil nil (not (called-interactively-p 'interactive)))
|
|
(let ((nb-nls (cond ((looking-at "[\n][\n][\n]") 3)
|
|
((looking-at "[\n][\n]") 2)
|
|
((looking-at "[\n]") 1)
|
|
(t 0))))
|
|
(delete-region (- (line-beginning-position) nb-nls) (line-beginning-position)))))
|
|
(with-current-buffer standard-output (buffer-string))))))))) ; Return the text displayed.
|
|
|
|
;;;###autoload
|
|
(defun describe-option (variable &optional buffer) ; Bound to `C-h o'
|
|
"Describe an Emacs user variable (option).
|
|
Same as using a prefix arg with `describe-variable'."
|
|
(interactive (let ((symb (or (and (fboundp 'symbol-nearest-point)
|
|
(symbol-nearest-point))
|
|
(variable-at-point)))
|
|
(enable-recursive-minibuffers t))
|
|
(when (numberp symb) (setq symb nil)) ; `variable-at-point' returns 0 when there is no var.
|
|
(list (intern (completing-read
|
|
(format "Describe user option%s: "
|
|
(if (and symb (user-variable-p symb))
|
|
(format " (default %s)" symb)
|
|
""))
|
|
obarray 'user-variable-p
|
|
t nil nil (and symb (user-variable-p symb) (symbol-name symb)) t)))))
|
|
(describe-variable variable buffer t))
|
|
|
|
;;;###autoload
|
|
(defun describe-option-of-type (type option) ; Bound to `C-h C-o'
|
|
"Describe an Emacs user OPTION (variable) of a given `defcustom' TYPE.
|
|
A prefix argument determines the type-checking behavior:
|
|
- None: OPTION is defined with TYPE or a subtype of TYPE.
|
|
- Plain `C-u': OPTION is defined with TYPE or a subtype of TYPE,
|
|
or its current value is compatible with TYPE.
|
|
- Negative: OPTION is defined with TYPE (exact match).
|
|
- Non-negative: OPTION is defined with TYPE (exact match),
|
|
or its current value is compatible with TYPE.
|
|
|
|
If TYPE is nil (default value) then *all* `defcustom' variables are
|
|
potential candidates. That is different from using `describe-option',
|
|
because `describe-option' includes user-variable candidates not
|
|
defined with `defcustom' (with `*'-prefixed doc strings)."
|
|
(interactive
|
|
(let* ((symb (or (and (fboundp 'symbol-nearest-point) (symbol-nearest-point)) (variable-at-point)))
|
|
(typ (car (condition-case err
|
|
(read-from-string (let ((types ()))
|
|
(mapatoms
|
|
(lambda (cand)
|
|
(when (custom-variable-p cand)
|
|
(push (list
|
|
(format
|
|
"%s"
|
|
(format "%S" (get cand 'custom-type))))
|
|
types))))
|
|
(completing-read "Describe option of type: "
|
|
(help-remove-duplicates types)
|
|
nil nil nil nil "nil")))
|
|
(end-of-file (error "No such custom type")))))
|
|
(pref-arg current-prefix-arg))
|
|
(when (numberp symb) (setq symb nil)) ; `variable-at-point' returns 0 when there is no var.
|
|
(list typ
|
|
(intern
|
|
(completing-read
|
|
(format "Option%s: " (if (and symb (user-variable-p symb)) (format " (default %s)" symb) ""))
|
|
obarray
|
|
(lambda (v)
|
|
(and (custom-variable-p v)
|
|
(or (not typ) ; Allow all vars if requested type = nil.
|
|
(help-var-is-of-type-p v (list typ) (cond ((not pref-arg) 'inherit)
|
|
((consp pref-arg) 'inherit-or-value)
|
|
((wholenump
|
|
(prefix-numeric-value pref-arg))
|
|
'direct-or-value)
|
|
(t 'direct))))))
|
|
t nil nil (and symb (user-variable-p symb) (symbol-name symb)) t)))))
|
|
(describe-variable option nil t))
|
|
|
|
(defun help-var-is-of-type-p (variable types &optional mode)
|
|
"Return non-nil if VARIABLE satisfies one of the custom types in TYPES.
|
|
TYPES is a list of `defcustom' type sexps or a list of regexp strings.
|
|
TYPES are matched, in order, against VARIABLE's type definition or
|
|
VARIABLE's current value, until one is satisfied or all are tried.
|
|
|
|
If TYPES is a list of regexps, then each is regexp-matched against
|
|
VARIABLE's custom type.
|
|
|
|
Otherwise, TYPES is a list of type sexps, each of which is a
|
|
definition acceptable for `defcustom' :type or the first symbol of
|
|
such a definition (e.g. `choice'). In this case, two kinds of type
|
|
comparison are possible:
|
|
|
|
1. VARIABLE's custom type, or its first symbol, is matched using
|
|
`equal' against each type in TYPES.
|
|
|
|
2. VARIABLE's current value is checked against each type in TYPES to
|
|
see if it satisfies one of them. In this case, VARIABLE's own type
|
|
is not used; VARIABLE might not even be typed - it could be a
|
|
variable not defined using `defcustom'.
|
|
|
|
For any of the comparisons against VARIABLE's type, either that type
|
|
can be checked directly or its supertypes (inherited types) can also
|
|
be checked.
|
|
|
|
These different type-checking possibilities depend on the value of
|
|
argument MODE, as follows, and they determine the meaning of the
|
|
returned value:
|
|
|
|
`direct': VARIABLE's type matches a member of list TYPES
|
|
`inherit': VARIABLE's type matches or is a subtype of a TYPES member
|
|
`value': VARIABLE is bound, and its value satisfies a type in TYPES
|
|
`inherit-or-value': `inherit' or `value', tested in that order
|
|
`direct-or-value': `direct' or `value', tested in that order
|
|
anything else (default): `inherit'
|
|
|
|
VARIABLE's current value cannot satisfy a regexp type: it is
|
|
impossible to know which concrete types a value must match."
|
|
(case mode
|
|
((nil inherit) (help-var-inherits-type-p variable types))
|
|
(inherit-or-value (or (help-var-inherits-type-p variable types)
|
|
(help-var-val-satisfies-type-p variable types)))
|
|
(value (help-var-val-satisfies-type-p variable types))
|
|
(direct (help-var-matches-type-p variable types))
|
|
(direct-or-value (or (member (get variable 'custom-type) types)
|
|
(help-var-val-satisfies-type-p variable types)))
|
|
(otherwise (help-var-inherits-type-p variable types))))
|
|
|
|
(defun help-var-matches-type-p (variable types)
|
|
"VARIABLE's type matches a member of TYPES."
|
|
(catch 'help-type-matches
|
|
(let ((var-type (get variable 'custom-type)))
|
|
(dolist (type types)
|
|
(when (if (stringp type)
|
|
(save-match-data (string-match type (format "%s" (format "%S" var-type))))
|
|
(equal var-type type))
|
|
(throw 'help-type-matches t))))
|
|
nil))
|
|
|
|
(defun help-var-inherits-type-p (variable types)
|
|
"VARIABLE's type matches or is a subtype of a member of list TYPES."
|
|
(catch 'help-type-inherits
|
|
(let ((var-type (get variable 'custom-type)))
|
|
(dolist (type types)
|
|
(while var-type
|
|
(when (or (and (stringp type)
|
|
(save-match-data (string-match type (format "%s" (format "%S" var-type)))))
|
|
(equal type var-type))
|
|
(throw 'help-type-inherits t))
|
|
(when (consp var-type) (setq var-type (car var-type)))
|
|
(when (or (and (stringp type)
|
|
(save-match-data (string-match type (format "%s" (format "%S" var-type)))))
|
|
(equal type var-type))
|
|
(throw 'help-type-inherits t))
|
|
(setq var-type (car (get var-type 'widget-type))))
|
|
(setq var-type (get variable 'custom-type))))
|
|
nil))
|
|
|
|
(defun help-var-val-satisfies-type-p (variable types)
|
|
"VARIABLE is bound, and its value satisfies a type in the list TYPES."
|
|
(and (boundp variable)
|
|
(let ((val (symbol-value variable)))
|
|
(and (widget-convert (get variable 'custom-type)) (help-value-satisfies-type-p val types)))))
|
|
|
|
(defun help-value-satisfies-type-p (value types)
|
|
"Return non-nil if VALUE satisfies a type in the list TYPES."
|
|
(catch 'help-type-value-satisfies
|
|
(dolist (type types)
|
|
(unless (stringp type) ; Skip, for regexp type.
|
|
(setq type (widget-convert type))
|
|
(when (condition-case nil ; Satisfies if either :match or :validate.
|
|
(progn (when (and (widget-get type :match) (widget-apply type :match value))
|
|
(throw 'help-type-value-satisfies t))
|
|
(when (and (widget-get type :validate)
|
|
(progn (widget-put type :value value)
|
|
(not (widget-apply type :validate))))
|
|
(throw 'help-type-value-satisfies t)))
|
|
(error nil))
|
|
(throw 'help-type-value-satisfies t))))
|
|
nil))
|
|
|
|
(defun help-custom-type (variable)
|
|
"Returns the `defcustom' type of VARIABLE.
|
|
Returns nil if VARIABLE is not a user option.
|
|
|
|
Note: If the library that defines VARIABLE has not yet been loaded,
|
|
then `help-custom-type' loads it. Be sure you want to do that
|
|
before you call this function."
|
|
(and (custom-variable-p variable)
|
|
(or (get variable 'custom-type) (progn (custom-load-symbol variable)
|
|
(get variable 'custom-type)))))
|
|
|
|
;; Borrowed from `ps-print.el'
|
|
(defun help-remove-duplicates (list)
|
|
"Copy of LIST with duplicate elements removed. Tested with `equal'."
|
|
(let ((tail list)
|
|
new)
|
|
(while tail
|
|
(unless (member (car tail) new) (push (car tail) new))
|
|
(pop tail))
|
|
(nreverse new)))
|
|
|
|
|
|
;; REPLACE ORIGINAL in `faces.el':
|
|
;;
|
|
;; Call `Info-make-manuals-xref' to create a cross-ref link to manuals.
|
|
;;
|
|
(when (or (> emacs-major-version 23) (and (= emacs-major-version 23) (> emacs-minor-version 1)))
|
|
|
|
(defun describe-face (face &optional frame)
|
|
"Display the properties of face FACE on FRAME.
|
|
Interactively, FACE defaults to the faces of the character after point
|
|
and FRAME defaults to the selected frame.
|
|
|
|
If the optional argument FRAME is given, report on face FACE in that frame.
|
|
If FRAME is t, report on the defaults for face FACE (for new frames).
|
|
If FRAME is omitted or nil, use the selected frame."
|
|
(interactive
|
|
(list (read-face-name "Describe face" (if (> emacs-major-version 23)
|
|
(or (if (and (= emacs-major-version 24) (< emacs-minor-version 4))
|
|
(face-at-point)
|
|
(face-at-point t))
|
|
'default)
|
|
"= `default' face")
|
|
(or (not (boundp 'icicle-WYSIWYG-Completions-flag))
|
|
(not icicle-WYSIWYG-Completions-flag)))))
|
|
(let* ((attrs '((:family . "Family")
|
|
(:foundry . "Foundry")
|
|
(:width . "Width")
|
|
(:height . "Height")
|
|
(:weight . "Weight")
|
|
(:slant . "Slant")
|
|
(:foreground . "Foreground")
|
|
(:background . "Background")
|
|
(:underline . "Underline")
|
|
(:overline . "Overline")
|
|
(:strike-through . "Strike-through")
|
|
(:box . "Box")
|
|
(:inverse-video . "Inverse")
|
|
(:stipple . "Stipple")
|
|
(:font . "Font")
|
|
(:fontset . "Fontset")
|
|
(:inherit . "Inherit")))
|
|
(max-width (apply #'max (mapcar #'(lambda (x) (length (cdr x))) attrs))))
|
|
(help-setup-xref (list #'describe-face face) (called-interactively-p 'interactive))
|
|
(unless face (setq face 'default))
|
|
(unless (listp face) (setq face (list face)))
|
|
(with-help-window (help-buffer)
|
|
(with-current-buffer standard-output
|
|
(dolist (f face)
|
|
(when (stringp f) (setq f (intern f)))
|
|
;; We may get called for anonymous faces (i.e., faces expressed using prop-value plists).
|
|
;; Those can't be usefully customized, so ignore them.
|
|
(when (symbolp f)
|
|
(insert "Face: " (symbol-name f))
|
|
(if (not (facep f))
|
|
(insert " undefined face.\n")
|
|
(let ((customize-label "customize this face")
|
|
file-name)
|
|
(insert (concat " (" (propertize "sample" 'font-lock-face f) ")"))
|
|
(princ (concat " (" customize-label ")\n"))
|
|
;; FIXME not sure how much of this belongs here, how much in `face-documentation'.
|
|
;; The latter is not used much, but needs to return nil for undocumented faces.
|
|
(let ((alias (get f 'face-alias))
|
|
(face f)
|
|
obsolete)
|
|
(when alias
|
|
(setq face alias)
|
|
(insert (format "\n %s is an alias for the face `%s'.\n%s" f alias
|
|
(if (setq obsolete (get f 'obsolete-face))
|
|
(format " This face is obsolete%s; use `%s' instead.\n"
|
|
(if (stringp obsolete) (format " since %s" obsolete) "")
|
|
alias)
|
|
""))))
|
|
(insert "\nDocumentation:\n" (or (face-documentation face)
|
|
"Not documented as a face.")
|
|
"\n\n"))
|
|
(with-current-buffer standard-output
|
|
(save-excursion (re-search-backward (concat "\\(" customize-label "\\)") nil t)
|
|
(help-xref-button 1 'help-customize-face f)))
|
|
(setq file-name (find-lisp-object-file-name f 'defface))
|
|
(when file-name
|
|
(princ "Defined in `") (princ (file-name-nondirectory file-name)) (princ "'")
|
|
(save-excursion ; Make a hyperlink to the library.
|
|
(re-search-backward "`\\([^`']+\\)'" nil t)
|
|
(help-xref-button 1 'help-face-def f file-name))
|
|
(princ ".") (terpri) (terpri))
|
|
(dolist (a attrs)
|
|
(let ((attr (face-attribute f (car a) frame)))
|
|
(insert (make-string (- max-width (length (cdr a))) ?\s)
|
|
(cdr a) ": " (format "%s" attr))
|
|
(when (and (eq (car a) :inherit) (not (eq attr 'unspecified)))
|
|
(save-excursion ; Make a hyperlink to the parent face.
|
|
(re-search-backward ": \\([^:]+\\)" nil t)
|
|
(help-xref-button 1 'help-face attr)))
|
|
(insert "\n")))
|
|
(when (boundp 'Info-virtual-files) ; Emacs 23.2+
|
|
(with-current-buffer standard-output ; Link to manuals.
|
|
(Info-make-manuals-xref f nil nil (not (called-interactively-p 'interactive)))))))
|
|
(terpri)))))))
|
|
)
|
|
|
|
;;;###autoload
|
|
(defun describe-file (filename &optional internal-form-p no-error-p) ; Bound to `C-h M-f'
|
|
"Describe the file named FILENAME.
|
|
If FILENAME is nil, describe current directory (`default-directory').
|
|
|
|
Starting with Emacs 22, if the file is an image file then:
|
|
* Show a thumbnail of the image as well.
|
|
* If you have command-line tool `exiftool' installed and in your
|
|
`$PATH' or `exec-path', then show EXIF data (metadata) about the
|
|
image. See standard Emacs library `image-dired.el' for more
|
|
information about `exiftool'.
|
|
|
|
If FILENAME is the name of an autofile bookmark and you use library
|
|
`Bookmark+', then show also the bookmark information (tags etc.). In
|
|
this case, a prefix arg shows the internal form of the bookmark.
|
|
|
|
In Lisp code:
|
|
|
|
Non-nil optional arg INTERNAL-FORM-P shows the internal form.
|
|
Non-nil optional arg NO-ERROR-P prints an error message but does not
|
|
raise an error."
|
|
(interactive "FDescribe file: \nP")
|
|
(unless filename (setq filename default-directory))
|
|
(help-setup-xref `(describe-file ,filename ,internal-form-p ,no-error-p)
|
|
(if (or (> emacs-major-version 23) ; Emacs 23.1 `called-interactively' accepts no arg.
|
|
(and (= emacs-major-version 23) (> emacs-minor-version 1)))
|
|
(called-interactively-p 'interactive)
|
|
(interactive-p)))
|
|
(let ((attrs (file-attributes filename))
|
|
;; Functions `bmkp-*' are defined in `bookmark+.el'.
|
|
(bmk (and (fboundp 'bmkp-get-autofile-bookmark) (bmkp-get-autofile-bookmark filename))))
|
|
(if (not attrs)
|
|
(if no-error-p (message "Cannot open file `%s'" filename) (error "Cannot open file `%s'" filename))
|
|
(let* ((type (nth 0 attrs))
|
|
(numlinks (nth 1 attrs))
|
|
(uid (nth 2 attrs))
|
|
(gid (nth 3 attrs))
|
|
(last-access (nth 4 attrs))
|
|
(last-mod (nth 5 attrs))
|
|
(last-status-chg (nth 6 attrs))
|
|
(size (nth 7 attrs))
|
|
(permissions (nth 8 attrs))
|
|
;; Skip 9: t iff file's gid would change if file were deleted and recreated.
|
|
(inode (nth 10 attrs))
|
|
(device (nth 11 attrs))
|
|
(thumb-string (and (fboundp 'image-file-name-regexp) ; In `image-file.el' (Emacs 22+).
|
|
(if (fboundp 'string-match-p)
|
|
(string-match-p (image-file-name-regexp) filename)
|
|
(save-match-data
|
|
(string-match (image-file-name-regexp) filename)))
|
|
(if (fboundp 'display-graphic-p) (display-graphic-p) window-system)
|
|
(require 'image-dired nil t)
|
|
(image-dired-get-thumbnail-image filename)
|
|
(apply #'propertize "XXXX"
|
|
`(display ,(append (image-dired-get-thumbnail-image filename)
|
|
'(:margin 10))
|
|
rear-nonsticky (display)
|
|
mouse-face highlight
|
|
follow-link t
|
|
help-echo "`mouse-2' or `RET': Show full image"
|
|
keymap (keymap
|
|
(mouse-2 . (lambda (e) (interactive "e")
|
|
(find-file ,filename)))
|
|
(13 . (lambda () (interactive)
|
|
(find-file ,filename))))))))
|
|
(image-info (and (require 'image-dired nil t)
|
|
(fboundp 'image-file-name-regexp)
|
|
(if (fboundp 'string-match-p)
|
|
(string-match-p (image-file-name-regexp) filename)
|
|
(save-match-data
|
|
(string-match (image-file-name-regexp) filename)))
|
|
(progn (when (if (or (> emacs-major-version 23)
|
|
(and (= emacs-major-version 23)
|
|
(> emacs-minor-version 1)))
|
|
(called-interactively-p 'interactive)
|
|
(interactive-p))
|
|
(message "Gathering image data...")) t)
|
|
(condition-case nil
|
|
(let ((all (help-all-exif-data (expand-file-name filename))))
|
|
(concat
|
|
(and all
|
|
(not (zerop (length all)))
|
|
(format "\nImage Data (EXIF)\n-----------------\n%s" all))))
|
|
(error nil))))
|
|
(help-text (concat
|
|
(format "%s\n%s\n\n" filename (make-string (length filename) ?-))
|
|
(format "File Type: %s\n"
|
|
(cond ((eq t type) "Directory")
|
|
((stringp type) (format "Symbolic link to `%s'" type))
|
|
(t "Normal file")))
|
|
(format "Permissions: %s\n" permissions)
|
|
(and (not (eq t type)) (format "Size in bytes: %g\n" size))
|
|
(format-time-string
|
|
"Time of last access: %a %b %e %T %Y (%Z)\n" last-access)
|
|
(format-time-string
|
|
"Time of last modification: %a %b %e %T %Y (%Z)\n" last-mod)
|
|
(format-time-string
|
|
"Time of last status change: %a %b %e %T %Y (%Z)\n" last-status-chg)
|
|
(format "Number of links: %d\n" numlinks)
|
|
(format "User ID (UID): %s\n" uid)
|
|
(format "Group ID (GID): %s\n" gid)
|
|
(format "Inode: %S\n" inode)
|
|
(format "Device number: %s\n" device)
|
|
image-info)))
|
|
(if (fboundp 'with-help-window)
|
|
(with-help-window (help-buffer)
|
|
(when bmk
|
|
(if internal-form-p
|
|
(let* ((bname (bookmark-name-from-full-record bmk))
|
|
(bmk-defn (format "Bookmark `%s'\n%s\n\n%s" bname
|
|
(make-string (+ 11 (length bname)) ?-)
|
|
(pp-to-string bmk))))
|
|
(princ bmk-defn) (terpri) (terpri))
|
|
(princ (bmkp-bookmark-description bmk 'NO-IMAGE)) (terpri) (terpri)))
|
|
(princ help-text))
|
|
(with-output-to-temp-buffer "*Help*"
|
|
(when bmk
|
|
(if internal-form-p
|
|
(let* ((bname (bookmark-name-from-full-record bmk))
|
|
(bmk-defn (format "Bookmark `%s'\n%s\n\n%s" bname
|
|
(make-string (+ 11 (length bname)) ?-)
|
|
(pp-to-string bmk))))
|
|
(princ bmk-defn) (terpri) (terpri))
|
|
(princ (bmkp-bookmark-description bmk 'NO-IMAGE)) (terpri) (terpri)))
|
|
(princ help-text)))
|
|
(when thumb-string
|
|
(with-current-buffer "*Help*"
|
|
(save-excursion
|
|
(goto-char (point-min))
|
|
(let ((buffer-read-only nil))
|
|
(when (re-search-forward "Device number:.+\n" nil t) (insert thumb-string))))))
|
|
help-text)))) ; Return displayed text.
|
|
|
|
(defun help-all-exif-data (file)
|
|
"Return all EXIF data from FILE, using command-line tool `exiftool'."
|
|
(with-temp-buffer
|
|
(delete-region (point-min) (point-max))
|
|
(unless (eq 0 (call-process shell-file-name nil t nil shell-command-switch
|
|
(format "exiftool -All \"%s\"" file)))
|
|
(error "Could not get EXIF data"))
|
|
(buffer-substring (point-min) (point-max))))
|
|
|
|
(defun describe-keymap (keymap &optional search-symbols-p) ; Bound to `C-h M-k'
|
|
"Describe key bindings in KEYMAP.
|
|
Interactively, prompt for a variable that has a keymap value.
|
|
Completion is available for the variable name.
|
|
|
|
Non-interactively:
|
|
* KEYMAP can be such a keymap variable or a keymap.
|
|
* Non-nil optional arg SEARCH-SYMBOLS-P means that if KEYMAP is not a
|
|
symbol then search all variables for one whose value is KEYMAP."
|
|
(interactive (list (intern (completing-read "Keymap: " obarray
|
|
(lambda (m) (and (boundp m) (keymapp (symbol-value m))))
|
|
t nil 'variable-name-history))))
|
|
(unless (and (symbolp keymap) (boundp keymap) (keymapp (symbol-value keymap)))
|
|
(if (not (keymapp keymap))
|
|
(error "%sot a keymap%s"
|
|
(if (symbolp keymap) (format "`%S' is n" keymap) "N")
|
|
(if (symbolp keymap) " variable" ""))
|
|
(let ((sym nil))
|
|
(when search-symbols-p
|
|
(setq sym (catch 'describe-keymap
|
|
(mapatoms (lambda (symb) (when (and (boundp symb)
|
|
(eq (symbol-value symb) keymap)
|
|
(not (eq symb 'keymap))
|
|
(throw 'describe-keymap symb)))))
|
|
nil)))
|
|
(unless sym
|
|
(setq sym (gentemp "KEYMAP OBJECT (no variable) "))
|
|
(set sym keymap))
|
|
(setq keymap sym))))
|
|
(setq keymap (or (condition-case nil (indirect-variable keymap) (error nil)) keymap)) ; Follow aliasing.
|
|
(let* ((name (symbol-name keymap))
|
|
(doc (if (fboundp 'help-documentation-property) ; Emacs 23+
|
|
(help-documentation-property keymap 'variable-documentation nil 'ADD-HELP-BUTTONS)
|
|
(documentation-property keymap 'variable-documentation)))
|
|
(doc (and (not (equal "" doc)) doc)))
|
|
(help-setup-xref (list #'describe-keymap keymap)
|
|
(if (or (> emacs-major-version 23) ; Emacs 23.1 `called-interactively' accepts no arg.
|
|
(and (= emacs-major-version 23) (> emacs-minor-version 1)))
|
|
(called-interactively-p 'interactive)
|
|
(interactive-p)))
|
|
(if (fboundp 'with-help-window)
|
|
(with-help-window (help-buffer)
|
|
(princ name) (terpri) (princ (make-string (length name) ?-)) (terpri) (terpri)
|
|
(when doc
|
|
(when (boundp 'Info-virtual-files) ; Emacs 23.2+
|
|
(with-current-buffer "*Help*" ; Link to manuals.
|
|
(Info-make-manuals-xref name nil nil (not (if (or (> emacs-major-version 23)
|
|
(and (= emacs-major-version 23)
|
|
(> emacs-minor-version 1)))
|
|
(called-interactively-p 'interactive)
|
|
(interactive-p))))))
|
|
(princ doc) (terpri) (terpri))
|
|
;; Use `insert' instead of `princ', so control chars (e.g. \377) insert correctly.
|
|
(with-current-buffer "*Help*" (insert (substitute-command-keys (concat "\\{" name "}")))))
|
|
(with-output-to-temp-buffer "*Help*"
|
|
(princ name) (terpri) (princ (make-string (length name) ?-)) (terpri) (terpri)
|
|
(when doc
|
|
(when (boundp 'Info-virtual-files) ; Emacs 23.2+
|
|
(with-current-buffer "*Help*" ; Link to manuals.
|
|
(Info-make-manuals-xref name nil nil (not (if (or (> emacs-major-version 23)
|
|
(and (= emacs-major-version 23)
|
|
(> emacs-minor-version 1)))
|
|
(called-interactively-p 'interactive)
|
|
(interactive-p))))))
|
|
(princ doc) (terpri) (terpri))
|
|
;; Use `insert' instead of `princ', so control chars (e.g. \377) insert correctly.
|
|
(with-current-buffer "*Help*" (insert (substitute-command-keys (concat "\\{" name "}"))))))))
|
|
|
|
|
|
;; REPLACE ORIGINAL in `package.el':
|
|
;;
|
|
;; Call `Info-make-manuals-xref' to create a cross-ref link to manuals.
|
|
;;
|
|
(when (fboundp 'describe-package) ; Emacs 24+
|
|
|
|
(when (or (> emacs-major-version 24) (and (= emacs-major-version 24) (> emacs-minor-version 3)))
|
|
(defun describe-package (package)
|
|
"Display the full documentation of PACKAGE (a symbol)."
|
|
(interactive
|
|
(let* ((guess (function-called-at-point)))
|
|
(require 'finder-inf nil t)
|
|
;; Load the package list if necessary (but don't activate them).
|
|
(unless package--initialized
|
|
(package-initialize t))
|
|
(let ((packages (append (mapcar 'car package-alist)
|
|
(mapcar 'car package-archive-contents)
|
|
(mapcar 'car package--builtins))))
|
|
(unless (memq guess packages)
|
|
(setq guess nil))
|
|
(setq packages (mapcar 'symbol-name packages))
|
|
(let ((val
|
|
(completing-read (if guess
|
|
(format "Describe package (default %s): "
|
|
guess)
|
|
"Describe package: ")
|
|
packages nil t nil nil guess)))
|
|
(list (intern val))))))
|
|
(if (not (or (package-desc-p package) (and package (symbolp package))))
|
|
(message "No package specified")
|
|
(help-setup-xref (list #'describe-package package)
|
|
(called-interactively-p 'interactive))
|
|
(with-help-window (help-buffer)
|
|
(with-current-buffer standard-output
|
|
(describe-package-1 package)
|
|
(let* ((desc (or (and (package-desc-p package) package)
|
|
(cadr (assq package package-alist))
|
|
(let ((built-in (assq package package--builtins)))
|
|
(if built-in
|
|
(package--from-builtin built-in)
|
|
(cadr (assq package package-archive-contents))))))
|
|
(name (if desc (package-desc-name desc) package)))
|
|
(setq package name)
|
|
(Info-make-manuals-xref (concat (symbol-name package) " package")
|
|
nil nil (not (called-interactively-p 'interactive))))))))) ; Link to manuals
|
|
|
|
(unless (or (> emacs-major-version 24) (and (= emacs-major-version 24) (> emacs-minor-version 3)))
|
|
(defun describe-package (package)
|
|
"Display the full documentation of PACKAGE (a symbol)."
|
|
(interactive
|
|
(let* ((guess (function-called-at-point)))
|
|
(require 'finder-inf nil t)
|
|
;; Load the package list if necessary (but don't activate them).
|
|
(unless package--initialized (package-initialize t))
|
|
(let ((packages (append (mapcar 'car package-alist) (mapcar 'car package-archive-contents)
|
|
(mapcar 'car package--builtins))))
|
|
(unless (memq guess packages) (setq guess nil))
|
|
(setq packages (mapcar 'symbol-name packages))
|
|
(let ((val (completing-read (if guess
|
|
(format "Describe package (default %s): " guess)
|
|
"Describe package: ")
|
|
packages nil t nil nil guess)))
|
|
(list (if (equal val "") guess (intern val)))))))
|
|
(if (not (or (and (fboundp 'package-desc-p) (package-desc-p package))
|
|
(and package (symbolp package))))
|
|
(when (called-interactively-p 'interactive) (message "No package specified"))
|
|
(help-setup-xref (list #'describe-package package) (called-interactively-p 'interactive))
|
|
(with-help-window (help-buffer)
|
|
(with-current-buffer standard-output
|
|
(describe-package-1 package)
|
|
(when (fboundp 'package-desc-name) (setq package (package-desc-name package))) ; Emacs 24.4
|
|
(Info-make-manuals-xref (concat (symbol-name package) " package")
|
|
nil nil (not (called-interactively-p 'interactive)))))))) ; Link to manuals
|
|
|
|
)
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
(provide 'help-fns+)
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
;;; help-fns+.el ends here
|