This repository has been archived on 2022-11-18. You can view files and clone it, but cannot push or open issues or pull requests.
emacs/emacswiki/info+.el

5422 lines
282 KiB
EmacsLisp
Raw Normal View History

;;; info+.el --- Extensions to `info.el'. -*- coding:utf-8 -*-
;;
;; Filename: info+.el
;; Description: Extensions to `info.el'.
;; Author: Drew Adams
;; Maintainer: Drew Adams (concat "drew.adams" "@" "oracle" ".com")
;; Copyright (C) 1996-2018, Drew Adams, all rights reserved.
;; Created: Tue Sep 12 16:30:11 1995
;; Version: 0
;; Package-Requires: ()
;; Last-Updated: Fri Sep 21 15:14:21 2018 (-0700)
;; By: dradams
;; Update #: 6407
;; URL: https://www.emacswiki.org/emacs/download/info%2b.el
;; Doc URL: https://www.emacswiki.org/emacs/InfoPlus
;; Keywords: help, docs, internal
;; Compatibility: GNU Emacs: 23.x, 24.x, 25.x, 26.x
;;
;; Features that might be required by this library:
;;
;; `apropos', `apropos+', `avoid', `backquote', `bookmark',
;; `bookmark+', `bookmark+-1', `bookmark+-bmu', `bookmark+-key',
;; `bookmark+-lit', `button', `cl', `cmds-menu', `col-highlight',
;; `crosshairs', `fit-frame', `font-lock', `font-lock+',
;; `frame-fns', `help+', `help-fns', `help-fns+', `help-macro',
;; `help-macro+', `help-mode', `hl-line', `hl-line+', `info',
;; `info+', `kmacro', `menu-bar', `menu-bar+', `misc-cmds',
;; `misc-fns', `naked', `pp', `pp+', `second-sel', `strings',
;; `syntax', `thingatpt', `thingatpt+', `view', `vline',
;; `w32browser-dlgopen', `wid-edit', `wid-edit+'.
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;;; Commentary:
;;
;; Extensions to `info.el'.
;;
;; More description below.
;;
;; If you use Emacs 20, 21, or 22 then use library `info+20.el'
;; instead of `info+.el'.
;;(@> "Index")
;;
;; Index
;; -----
;;
;; If you have library `linkd.el', load `linkd.el' and turn on
;; `linkd-mode' now. It lets you easily navigate around the sections
;; of this doc. Linkd mode will highlight this Index, as well as the
;; cross-references and section headings throughout this file. You
;; can get `linkd.el' here:
;; https://www.emacswiki.org/emacs/download/linkd.el.
;;
;; (@> "Things Defined Here")
;; (@> "Documentation")
;; (@> "Macros")
;; (@> "Faces (Customizable)")
;; (@> "User Options (Customizable)")
;; (@> "Internal Variables")
;; (@> "New Commands")
;; (@> "Replacements for Existing Functions")
;; (@> "Non-Interactive Functions")
;;(@* "Things Defined Here")
;;
;; Things Defined Here
;; -------------------
;;
;; Commands defined here:
;;
;; `Info-breadcrumbs-in-mode-line-mode',
;; `Info-change-visited-status' (Emacs 24+),
;; `Info-describe-bookmark' (Emacs 24.2+),
;; `Info-follow-nearest-node-new-window', `Info-goto-node-web',
;; `Info-history-clear', `Info-make-node-unvisited', `info-manual',
;; `Info-merge-subnodes',
;; `Info-mouse-follow-nearest-node-new-window',
;; `Info-outline-demote', `Info-outline-promote',
;; `Info-persist-history-mode' (Emacs 24.4+),
;; `Info-save-current-node', `Info-set-breadcrumbs-depth',
;; `Info-set-face-for-bookmarked-xref' (Emacs 24.2+),
;; `Info-toggle-breadcrumbs-in-header',
;; `Info-toggle-fontify-angle-bracketed',
;; `Info-toggle-fontify-bookmarked-xrefs' (Emacs 24.2+),
;; `Info-toggle-fontify-emphasis',
;; `Info-toggle-fontify-quotations',
;; `Info-toggle-fontify-single-quote',
;; `Info-toggle-node-access-invokes-bookmark' (Emacs 24.4+),
;; `Info-toc-outline', `Info-toc-outline-refontify-region',
;; `Info-url-for-node', `Info-virtual-book'.
;;
;; Faces defined here:
;;
;; `info-command-ref-item', `info-constant-ref-item',
;; `info-double-quoted-name', `info-emphasis', `info-file',
;; `info-function-ref-item',`info-macro-ref-item', `info-menu',
;; `info-node', `info-quoted-name', `info-reference-item',
;; `info-single-quote', `info-special-form-ref-item',
;; `info-string', `info-syntax-class-item',
;; `info-user-option-ref-item', `info-variable-ref-item',
;; `info-xref-bookmarked' (Emacs 24.2+).
;;
;; Options (user variables) defined here:
;;
;; `Info-bookmarked-node-xref-faces' (Emacs 24.2+),
;; `Info-breadcrumbs-in-header-flag',
;; `Info-display-node-header-fn', `Info-emphasis-regexp',
;; `Info-fit-frame-flag', `Info-fontify-angle-bracketed-flag',
;; `Info-fontify-bookmarked-xrefs-flag' (Emacs 24.2+),
;; `Info-fontify-emphasis-flag', `Info-fontify-quotations-flag',
;; `Info-fontify-reference-items-flag',
;; `Info-fontify-single-quote-flag',
;; `Info-node-access-invokes-bookmark-flag' (Emacs 24.4+),
;; `Info-saved-history-file' (Emacs 24.4+), `Info-saved-nodes',
;; `Info-subtree-separator', `Info-toc-outline-no-redundancy-flag'.
;;
;; Macros defined here:
;;
;; `info-user-error'.
;;
;; Non-interactive functions defined here:
;;
;; `Info--pop-to-buffer-same-window', `Info-bookmark-for-node',
;; `Info-bookmark-name-at-point', `Info-bookmark-named-at-point',
;; `Info-bookmark-name-for-node',
;; `Info-display-node-default-header', `info-fontify-quotations',
;; `info-fontify-reference-items',
;; `Info-insert-breadcrumbs-in-mode-line', `Info-isearch-search-p',
;; `Info-node-name-at-point', `Info-read-bookmarked-node-name',
;; `Info-restore-history-list' (Emacs 24.4+),
;; `Info-save-history-list' (Emacs 24.4+), `Info-search-beg',
;; `Info-search-end', `Info-toc-outline-find-node',
;; `Info-toc-outline-refontify-links'.
;;
;; Internal variables defined here:
;;
;; `Info-breadcrumbs-depth-internal', `info-fontify-emphasis',
;; `Info-merged-map', `Info-mode-syntax-table',
;; `info-quotation-regexp', `info-quoted+<>-regexp',
;; `Info-toc-outline-map'.
;;
;;
;; ***** NOTE: The following standard faces defined in `info.el'
;; have been REDEFINED HERE:
;;
;; `info-title-1', `info-title-2', `info-title-3', `info-title-4'.
;;
;;
;; ***** NOTE: The following standard functions defined in `info.el'
;; have been REDEFINED or ADVISED HERE:
;;
;; `info-apropos' - Apropos, not literal string, match by default.
;; Use other window if not already in Info.
;; `Info-apropos-matches' - Added optional arg REGEXP-P.
;; `Info-backward-node' - Prefix arg clones buffer.
;; `Info-directory' - Prefix arg clones buffer.
;; `info-display-manual' - Use completion to input manual name.
;; `Info-find-emacs-command-nodes' - Added arg MSGP and message.
;; `Info-find-file' - Handle virtual books.
;; `Info-find-node', `Info-find-node-2' -
;; Call `fit-frame' if `Info-fit-frame-flag'.
;; Added optional arg NOMSG.
;; `Info-fontify-node' -
;; 1. Show breadcrumbs in header line and/or mode line.
;; 2. File name in face `info-file'.
;; 3. Node names in face `info-node'.
;; 4. Menu items in face `info-menu'.
;; 5. Only 5th and 9th menu items have their `*' colored.
;; 6. Notes in face `info-xref'.
;; 7. If `Info-fontify-emphasis-flag', then fontify _..._.
;; 8. If `Info-fontify-quotations-flag', then fontify ... or
;; `...' in face `info-quoted-name', “...” in face
;; `info-double-quoted-name', and "..." in face `info-string'.
;; 9. If `Info-fontify-angle-bracketed-flag' and
;; `Info-fontify-quotations-flag' then fontify <...> in face
;; `info-quoted-name'.
;; 10. If `Info-fontify-single-quote-flag' and
;; `Info-fontify-quotations-flag', then fontify ' in face
;; `info-single-quote'.
;; `Info-forward-node' - Prefix arg clones buffer.
;; `Info-goto-emacs-command-node' -
;; 1. Uses `completing-read' in interactive spec, with,
;; as default, `symbol-nearest-point'.
;; 2. Added optional arg MSGP.
;; 3. Message if single node found.
;; 4. Returns `num-matches' if found; nil if not.
;; `Info-goto-emacs-key-command-node' -
;; 1. Added optional arg MSGP.
;; 2. If key's command not found, then `Info-search's for key
;; sequence in text and displays message about repeating.
;; `Info-goto-node' - Respect option
;; `Info-node-access-invokes-bookmark-flag' (Emacs 24.4+).
;; `Info-history' - A prefix arg clears the history.
;; `Info-history-back' - Prefix arg clones buffer.
;; `Info-history-forward' - Prefix arg clones buffer.
;; `Info-insert-dir' -
;; Added optional arg NOMSG to inhibit showing progress msgs.
;; `Info-isearch-search' - Respect restriction to active region.
;; `Info-isearch-wrap' - Respect restriction to active region.
;; `Info-mode' - Doc string shows all bindings.
;; `Info-next' - Prefix arg clones buffer.
;; `Info-prev' - Prefix arg clones buffer.
;; `Info-read-node-name' - Added optional arg DEFAULT.
;; `Info-search' - 1. Fits frame.
;; 2. Highlights found regexp if `search-highlight'.
;; `Info-set-mode-line' - Handles breadcrumbs in the mode line.
;; `Info-mouse-follow-nearest-node' - With prefix arg, show node in
;; a new Info buffer.
;; `Info-up' - Prefix arg clones buffer.
;;
;;
;; ***** NOTE: The following standard function
;; has been REDEFINED HERE:
;;
;; `outline-invisible-p' - Fixes Emacs bug #28080.
;;(@* "Documentation")
;;
;; Documentation
;; -------------
;;
;; Library `info+.el' extends the standard Emacs library `info.el' in
;; several ways. It provides:
;;
;; * Association of additional information (metadata) with Info
;; nodes. You do this by bookmarking the nodes. Library Bookmark+
;; gives you the following features in combination with `info+.el'.
;; In many ways an Info node and its default bookmark can be
;; thought of as the same animal.
;;
;; - Rich node metadata. In particular, you can tag nodes with any
;; number of arbitrary tags, to classify them in different and
;; overlapping ways. You can also annotate them (in Org mode, by
;; default).
;;
;; - You can use `C-h C-b' to show the metadata for a (bookmarked)
;; node. This is all of the associated bookmark information,
;; including the annotation and tags for that node and the number
;; of times you have visited it. If invoked with point on a
;; link, the targeted node is described; otherwise, you are
;; prompted for the node name.
;;
;; - Links for bookmarked nodes can have a different face, to let
;; you know that those nodes have associated metadata. Option
;; `Info-fontify-bookmarked-xrefs-flag' controls whether this is
;; done.
;;
;; - The face for this is `info-xref-bookmarked' by default, but
;; you can set the face to use for a given Info bookmark using
;; `C-x f' (command `Info-set-face-for-bookmarked-xref'). This
;; gives you an easy way to classify nodes and show the class of
;; a node by its links. Uses faces to make clear which nodes are
;; most important to you, or which are related to this or that
;; general topic.
;;
;; - If option `Info-node-access-invokes-bookmark-flag' is non-nil
;; then going to a bookmarked Info node invokes its bookmark, so
;; that the node metadata (such as number of visits) gets
;; updated. Command `Info-toggle-node-access-invokes-bookmark'
;; toggles the option value.
;;
;; - You can automatically bookmark nodes you visit, by enabling
;; mode `bmkp-info-auto-bookmark-mode'. Toggle the mode off
;; anytime you do not want to record Info visits.
;;
;; - In the bookmark-list display (from `C-x r l') you can sort
;; bookmarks by the time of last visit (`s d') or by the number
;; of visits (`s v'). This gives you an easy way to see which
;; parts of which Info manuals you have visited most recently and
;; how much you have visited them.
;;
;; * Editable, outline-enabled tables of contents (TOCs). Command
;; `Info-toc-outline' (bound to `O') opens a separate Info buffer
;; showing the table of contents (TOC). This is similar to the
;; standard command `Info-toc' (bound to `T'), but the buffer is
;; cloned from the manual and is in `outline-minor-mode'. Also,
;; there is no redundancy, by default: each TOC entry is listed
;; only once, not multiple times. (This is controlled by option
;; `Info-toc-outline-no-redundancy-flag'.)
;;
;; You can have any number of such TOCs, for the same manual or for
;; different manuals.
;;
;; Outline minor mode lets you hide and show, and promote and
;; demote, various parts of the TOC tree for a manual. And since
;; the TOC is editable you can make other changes to it: sort parts
;; of it, delete parts of it, duplicate parts of it, move parts
;; aroundin an ad hoc way, and so on. Info+ makes the outlining
;; commands behave, so that hidden Info text (e.g. markup text such
;; as `*note'...`::' surrounding links) is kept hidden.
;;
;; Especially when combined with `Info-persist-history-mode',
;; command `Info-change-visited-status' (`C-x DEL', see below), and
;; the Info+ bookmarking enhancements (e.g., special link
;; highlighting and persistently tracking the number of visits per
;; node), `Info-toc-outline' gives you a way to organize access and
;; visibility of a manual's nodes, to reflect how you use it.
;;
;; * Additional, finer-grained Info highlighting. This can make a
;; big difference in readability.
;;
;; - Quoted names, like this: `name-stands-out' or
;; `name-stands-out', and strings, like this: "string-stands-out"
;; are highlighted if `Info-fontify-quotations-flag' is
;; non-`nil'.
;;
;; - Angle-bracketed names, like this: <tab>, are highlighted if
;; `Info-fontify-angle-bracketed-flag' and
;; `Info-fontify-quotations-flag' are non-`nil'.
;;
;; - Isolated single quotes, like this: 'foobar, are highlighted if
;; `Info-fontify-single-quote-flag' and
;; `Info-fontify-quotations-flag' are non-`nil'.
;;
;; - Emphasized text, that is, text enclosed in underscore
;; characters, like this: _this is emphasized text_, is
;; highlighted if `Info-fontify-emphasis-flag' is non-`nil'.
;; (But if internal variable `info-fontify-emphasis' is `nil'
;; then there is no such highlighting, and that option has no
;; effect.)
;;
;; - In the Emacs Lisp manual, reference items are highlighted, so
;; they stand out. This means: constants, commands, functions,
;; macros, special forms, syntax classes, user options, and other
;; variables.
;;
;; Be aware that such highlighting is not 100% foolproof.
;; Especially for a manual such as Emacs or Elisp, where arbitrary
;; keys and characters can be present anywhere, the highlighting
;; can be thrown off.
;;
;; You can toggle each of the `Info-fontify-*-flag' options from
;; the `Info' menu or using an `Info-toggle-fontify-*' command.
;; For example, command `Info-toggle-fontify-emphasis' toggles
;; option `Info-fontify-emphasis-flag'.
;;
;; * You can show breadcrumbs in the mode line or the header line, or
;; both. See where you are in the Info hierarchy, and access higher
;; nodes directly.
;;
;; - In the mode line. Turned on by default.
;;
;; See Toggle Breadcrumbs in the `mouse-3' mode-line menu and
;; `Toggle Breadcrumbs in Mode Line' in the `Info' menu (in the
;; menu-bar or in the minor-mode indicator). You can customize
;; option `Info-breadcrumbs-in-mode-line-mode' if you want to
;; turn this off by default. (Available for Emacs 23+ only.)
;;
;; - In the header (just below the header line).
;;
;; (I also added this to vanilla Emacs 23.) This is OFF by
;; default in `Info+'. See `Toggle Breadcrumbs in Header Line'
;; in `Info' menu. Be aware that unlike breadcrumbs in the mode
;; line, this can occasionally throw off the destination accuracy
;; of cross references and searches slightly.
;;
;; * Some of the commands defined here:
;;
;; - `Info-virtual-book' (bound to `v') Open a virtual Info
;; manual of saved nodes from any number of manuals. The nodes
;; are those saved in option `Info-virtual-book'. With `C-u',
;; bookmarked Info nodes are also included. (If you use Icicles,
;; see also `icicle-Info-virtual-book'.)
;;
;; - `Info-persist-history-mode' - Enabling this minor mode saves
;; the list of your visited Info nodes between Emacs sessions.
;; Together with command `Info-history' (bound to `L' by
;; default), this gives you a persistent virtual manual of the
;; nodes you have visited in the past. If the mode is enabled
;; then the list of visited nodes is saved to the file named by
;; option `Info-saved-history-file' when you quit Emacs (not
;; Info) or when you kill an Info buffer.
;;
;; (If you also use library Bookmark+ then you can bookmark Info
;; nodes, including automatically. This records how many times
;; you have visited each node and when you last did so.)
;;
;; - `Info-change-visited-status' (bound to `C-x DEL') - Toggle or
;; set the visited status of the node at point or the nodes in
;; the active region. Useful if you use
;; `Info-fontify-visited-nodes' to show you which nodes you have
;; visited. No prefix arg: toggle. Non-negative prefix arg: set
;; to visited. Negative prefix arg: set to unvisited.
;;
;; - `Info-save-current-node' (bound to `.') Save the name of the
;; current node to list `Info-saved-nodes', for use by `v'
;; (`Info-virtual-book').
;;
;; - `Info-merge-subnodes' Integrate the current Info node with
;; its subnodes (the nodes in its Menu), perhaps recursively.
;;
;; Use `Info-merge-subnodes' to extract a self-contained report
;; (possibly the whole manual) from an Info manual. The report
;; is itself an Info buffer, with hyperlinks and normal Info
;; behavior.
;;
;; There are various prefix-argument possibilities that govern
;; just how subnodes are treated (recursively or not, for
;; instance). There are a few user options that let you
;; customize the report appearance.
;;
;;
;; The following bindings are made here for Info-mode:
;;
;; `?' `describe-mode' (replaces `Info-summary')
;; `+' `Info-merge-subnodes'
;; `.' `Info-save-current-node'
;; `a' `info-apropos'
;; `G' `Info-goto-node-web'
;; `O' `Info-toc-outline'
;; `v' `Info-virtual-book'
;; `mouse-4' `Info-history-back'
;; `mouse-5' `Info-history-forward'
;; `S-down-mouse-2' `Info-mouse-follow-nearest-node-new-window'
;; `S-RET' `Info-follow-nearest-node-new-window'
;;
;; The following bindings are made here for merged Info buffers:
;;
;; `.' `beginning-of-buffer'
;; `b' `beginning-of-buffer'
;; `q' `quit-window'
;; `s' `nonincremental-re-search-forward'
;; `M-s' `nonincremental-re-search-forward'
;; `TAB' `Info-next-reference'
;; `ESC TAB' `Info-prev-reference'
;;
;; The global binding `C-h r' is changed from `info-emacs-manual' to
;; `info-manual', which behaves the same except if you use a prefix
;; arg. With a prefix arg you can open any manual, choosing either
;; from all installed manuals or from those that are already shown in
;; Info buffers.
;;
;; The following behavior defined in `info.el' has been changed:
;; "*info" has been removed from `same-window-buffer-names', so that
;; a separate window can be used if you so choose.
;;
;; Suggestion: Use a medium-dark background for Info. Try, for
;; example, setting the background to "LightSteelBlue" in your
;; `~/.emacs' file. You can do this as follows:
;;
;; (setq special-display-buffer-names
;; (cons '("*info*" (background-color . "LightSteelBlue"))
;; special-display-buffer-names))
;;
;; Alternatively, you can change the background value of
;; `special-display-frame-alist' and set `special-display-regexps' to
;; something matching "*info*":
;;
;; (setq special-display-frame-alist
;; (cons '(background-color . "LightSteelBlue")
;; special-display-frame-alist))
;; (setq special-display-regexps '("[ ]?[*][^*]+[*]"))
;;
;; If you do use a medium-dark background for Info, consider
;; customizing face to a lighter foreground color - I use "Yellow".
;;
;; Also, consider customizing face `link' to remove its underline
;; attribute.
;;
;; This file should be loaded after loading the standard GNU file
;; `info.el'. So, in your `~/.emacs' file, do this:
;; (eval-after-load "info" '(require 'info+))
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;;; Change Log:
;;
;; 2018/09/21 dadams
;; Added: Info--pop-to-buffer-same-window.
;; Info-find-node, Info-(next|prev|up), info-display-manual: Use Info--pop-to-buffer-same-window, not switch-to-buffer.
;; 2018/09/09 dadams
;; Added: redefinitions of Info-backward-node, Info-directory, Info-forward-node, Info-history-back, Info-history-forward,
;; Info-next, Info-prev, Info-up.
;; 2018/06/14 dadams
;; Added: redefinitions of info-apropos, Info-apropos-matches.
;; bmkp-string-match-p - > string-match-p everywhere.
;; 2018/06/03 dadams
;; info-quotation-regexp, info-quoted+<>-regexp:
;; Use shy groups everywhere. [CHAR] -> CHAR, \\CHAR -> CHAR. Added equivalent rx sexps.
;; info-fontify-reference-items, Info-node-name-at-point: [\n] -> \n.
;; 2018/04/12 dadams
;; Info-read-node-name: Use Info-minibuffer-history, not Info-history, for completing-read.
;; Info-find-node(-2): Added arg STRICT-CASE, moving arg MSG to the end.
;; Info-find-node-2: Updated for Emacs 25+ - use filepos-to-bufferpos etc.
;; 2017/11/17 dadams
;; Info-TOC-outline stuff needs Info-virtual-nodes. Thx to Mike Fitzgerald.
;; http -> https everywhere.
;; 2017/11/09 dadams
;; info-quotation-regexp, info-quoted+<>-regexp: Added \\ to first alternative of each ... type, to exclude \ from it.
;; 2017/09/23 dadams
;; Info-url-for-node: Fix per TeXInfo manual - encode embedded hyphens etc.
;; 2017/08/30 dadams
;; Renamed: Info-refontify-toc-outline-region to Info-toc-outline-refontify-region.
;; 2017/08/28 dadams
;; Added: Info-refontify-toc-outline-region.
;; Info-refontify-toc-outline-region: Add Info-refontify-toc-outline-region to post-command-hook and bind to C-x M-l.
;; Info-toc-outline: Turn off Info-breadcrumbs-in-mode-line-mode in TOC buffer.
;; Info-change-visited-status: Typo: go-to-char.
;; 2017/08/25 dadams
;; Added: Info-change-visited-status. Bound to `C-x DEL (instead of Info-make-node-unvisited).
;; Info-node-name-at-point: Replace newline chars by spaces.
;; Info-toc-outline: Pass NEWNAME arg to clone-buffer, instead of explicitly renaming buffer.
;; 2017/08/22 dadams
;; Added: Info-toc-outline, Info-outline-demote, Info-outline-promote, Info-toc-outline-no-redundancy-flag,
;; Info-toc-outline-find-node, Info-toc-outline-map, Info-toc-outline-refontify-links, redefinition of
;; outline-invisible-p.
;; Bind Info-toc-outline to O.
;; Info-mode-menu: Added Editable Outline TOC item for Info-toc-outline.
;; Info-node-access-invokes-bookmark-flag, Info-toggle-node-access-invokes-bookmark, Info-goto-node advice:
;; Reserve for Emacs 24.4+.
;; 2017/08/10 dadams
;; Info-goto-node: Define it for Emacs 23 also.
;; Info-mode-menu: Add menu items for Info-toggle-node-access-invokes-bookmark, Info-toggle-fontify-bookmarked-xrefs.
;; 2017/08/07 dadams
;; Added: Info-make-node-unvisited. Bound to C-x DEL.
;; 2017/08/06 dadams
;; Added: Info-bookmarked-node-xref-faces, Info-read-bookmarked-node-name,
;; Info-set-face-for-bookmarked-xref.
;; Bind Info-set-face-for-bookmarked-xref to C-x f.
;; Info-describe-bookmark: If no bookmarked node name at point, use Info-read-bookmarked-node-name.
;; Info-bookmark-for-node: Made NODE arg optional - if nil then read the node name. Added LOCALP arg.
;; Info-fontify-node (Emacs 24.2+): Get face for bookmarked xref from bmkp-info-face tag value, if any.
;; Call Info-bookmark-for-node with arg LOCALP.
;; 2017/08/04 dadams
;; Info-describe-bookmark: Use Info-bookmark-name-at-point, not Info-node-name-at-point.
;; Info-goto-node: Do it only for Emacs 24.2+.
;; 2017/08/02 dadams
;; Info-goto-node: Define only if can soft-require bookmark+.el.
;; No-op if NODE is in Info-index-nodes.
;; Bind Info-node-access-invokes-bookmark-flag to nil while invoking bookmark.
;; Use bookmark--jump-via with ignore as display function, instead of bookmark-jump.
;; 2017/07/30 dadams
;; Added advice of Info-goto-node, to respect Info-node-access-invokes-bookmark-flag.
;; Removed redefinitions of Info-follow-nearest-node, Info-try-follow-nearest-node.
;; Replaced Info-follow-xref-bookmarks-flag by Info-node-access-invokes-bookmark-flag.
;; Replaced Info-toggle-follow-bookmarked-xrefs by Info-toggle-node-access-invokes-bookmark.
;; Info-bookmark-for-node, Info-bookmark-named-at-point: Include manual name in bookmark name.
;; 2017/07/29 dadams
;; Added: Info-fontify-bookmarked-xrefs-flag, face info-xref-bookmarked, Info-describe-bookmark,
;; Info-bookmark-for-node, Info-bookmark-name-at-point, Info-bookmark-named-at-point,
;; Info-bookmark-name-for-node, Info-toggle-fontify-bookmarked-xrefs,
;; Info-follow-xref-bookmarks-flag, Info-toggle-follow-bookmarked-xrefs.
;; Added (redefinition of): Info-follow-nearest-node, Info-try-follow-nearest-node.
;; Info-fontify-node (24.2+): Respect Info-fontify-bookmarked-xrefs-flag.
;; Bind Info-describe-bookmark to C-h C-b.
;; 2017/02/20 dadams
;; Added: Info-saved-history-file, Info-persist-history-mode, Info-save-history-list,
;; Info-restore-history-list.
;; Added autoload cookies: Info-breadcrumbs-in-mode-line-mode, Info-set-breadcrumbs-depth,
;; Info-search, Info-mouse-follow-nearest-node, info-display-manual.
;; 2017/01/09 dadams
;; Info-find-emacs-command-nodes: Updated to handle LINE-NUMBER (Emacs 24.5+).
;; 2016/12/13 dadams
;; Removed obsolete face aliases: info-menu-5, Info-title-*-face.
;; 2016/12/11 dadams
;; Added defvars for isearch(-regexp)-lax-whitespace for Emacs 24.1 and 24.2.
;; 2016/12/10 dadams
;; Use string as 3rd arg to make-obsolete.
;; 2016/10/31 dadams
;; info-quotation-regexp: Typo: misplaced curly double-quote. Thx to Don March.
;; 2016/07/02 dadams
;; Added: Info-toggle-fontify-emphasis, Info-breadcrumbs-in-header-flag, Info-emphasis-regexp,
;; Info-fontify-emphasis-flag, info-fontify-emphasis, and face info-emphasis.
;; Added some doc from Emacs Wiki to commentary.
;; Info-mode-menu:
;; Add toggle indicators. Moved toggle commands to Toggle submenu. Added Info-toggle-fontify-emphasis.
;; Info-fontify-node: Fontify emphasis.
;; 2015/09/14 dadams
;; info-double-quoted-name: Changed default colors.
;; 2015/09/13 dadams
;; Added face info-double-quoted-name.
;; info-quotation-regexp, info-quoted+<>-regexp: Added pattern for curly double-quotes (“...”).
;; Use shy groups for all parts.
;; info-fontify-quotations: Fontify text between curly double-quotes (“...”).
;; 2015/03/19 dadams
;; info-quoted+<>-regexp: Highlight <...> only if the first char is alphabetic.
;; 2015/03/06 dadams
;; Added: info-manual. Bound it globally to C-h r.
;; Info-fontify-node (Emacs 24.1.N+): Updated per Emacs 24.4: allow Info-fontify-maximum-menu-size to be t.
;; info-display-manual: Updated for Emacs 25: use info--manual-names with prefix arg.
;; 2015/02/28 dadams
;; Added: redefinition of Info-read-node-name.
;; Info-goto-node-web, Info-url-for-node: Use Info-current-node as default.
;; 2014/12/21 dadams
;; Added: Info-goto-node-web, Info-url-for-node.
;; Reorganized. Code cleanup. Improved commentary. Added index links.
;; Info-toggle-breadcrumbs-in-header-line: Added 3rd arg to make-obsolete.
;; Info-breadcrumbs-in-mode-line-mode: (default-value 'mode-line-format), not default-mode-line-format,
;; Info-display-node-default-header: (goto-char (point-min)), not (beginning-of-buffer).
;; Info-merge-subnodes: with-current-buffer, not save-excursion + set-buffer.
;; 2014/05/04 dadams
;; REMOVED SUPPORT for Emacs 20-22. That support is offered by a new library now: info+20.el.
;; Added coding:utf-8 declaration. Replace \x2018, \x2019 with literal and , since now Emacs 23+.
;; 2014/05/03 dadams
;; info-quotation-regexp, info-quoted+<>-regexp: Handle also curly single quotes (Emacs 24.4+).
;; Removed double * and moved openers outside \(...\) group.
;; info-fontify-quotations: Handle also curly single quotes (Emacs 24.4+).
;; 2014/03/04 dadams
;; Renamed Info-toggle-breadcrumbs-in-header-line to Info-toggle-breadcrumbs-in-header.
;; Declared old name obsolete.
;; 2014/03/02 dadams
;; Info-find-file: Go to directory if no previous file (per Emacs 24.4+).
;; Info-find-node-2 (Emacs > 22): Go to Top node at end, if no history.
;; 2013/10/17 dadams
;; Added: Info-search-beg, Info-search-end, Info-isearch-search-p.
;; Added redefinition: Info-isearch-wrap, Info-isearch-search.
;; Info-display-node-default-header, Info-merge-subnodes: Renamed node-name to infop-node-name.
;; 2013/03/17 dadams
;; Added: Info-history-clear, macro info-user-error (and font-lock it). Advised: Info-history.
;; Use info-user-error instead of error, where appropriate.
;; 2013/02/26 dadams
;; Info-mode-menu and Info-mode doc string: Removed Info-edit, Info-enable-edit (now obsolete).
;; 2013/02/09 dadams
;; Info-read-node-name-1: Removed Emacs 23+ redefinition.
;; 2013/02/03 dadams
;; Added: Info-fontify-angle-bracketed-flag, Info-toggle-fontify-angle-bracketed,
;; Info-toggle-fontify-quotations, Info-toggle-fontify-single-quote, info-quoted+<>-regexp.
;; info-fontify-quotations: Fixed case for Info-toggle-fontify-single-quote = nil.
;; Handle also Info-fontify-angle-bracketed-flag.
;; Added Info-fontify-*-flag to Info menu (so menu bar and C-mouse-3).
;; 2012/09/24 dadams
;; Info-search. Info-mode: Applied latest Emacs 24 updates by Juri (from 2012-09-12).
;; 2012/08/25 dadams
;; Info-fontify-node: Hide any empty lines at end of node (fixes bug #12272).
;; 2012/08/24 dadams
;; info-fontify-reference-items: Fontify parameters on continuation lines also.
;; Info-fontify-node: Juri's fix for Emacs bug #12187.
;; Reverted Juri's change from 08/20, since Juri fixed it elsewhere afterward.
;; 2012/08/21 dadams
;; Call tap-put-thing-at-point-props after load thingatpt+.el.
;; 2012/08/20 dadams
;; Applied Juri's fix for Emacs bug #12230:
;; Added: Info-file-attributes.
;; Info-find-file: Clear caches of modified Info files.
;; 2012/08/18 dadams
;; Invoke tap-define-aliases-wo-prefix if thingatpt+.el is loaded.
;; 2012/08/12 dadams
;; Added: info-constant-ref-item (face).
;; info-fontify-reference-items: Handle constants, using face info-constant-ref-item.
;; Info-toggle-breadcrumbs-in-header-line, Info-save-current-node: Added MSGP arg.
;; 2012/08/10 dadams
;; Info-search: Use latest Emacs 24 msg: _end of node_, not _initial node_.
;; 2012/08/09 dadams
;; Info-fontify-node: Updated guards for Emacs 24 versions.
;; 2012/07/28 dadams
;; Info-fontify-node: Typo on guard: (/= 1 emacs-minor-version) should have been =, not /=.
;; 2012/07/17 dadams
;; Added redefinition of Info-fontify-node for post-Emacs 24.1.
;; Added redefinitions of Info-insert-dir, Info(-directory)-find-node, with args controlling msgs.
;; info-find-node-2: Added optional arg NOMSG.
;; Info-find-emacs-command-nodes, Info-goto-emacs(-key)-command-node: Added optional arg MSGP.
;; Info-search, Info-save-current-node: Show messages only if interactive-p.
;; 2012/01/15 dadams
;; Added: info-display-manual (redefinition).
;; Info-find-file: Do not define for < Emacs 23.2 - no virtual books.
;; 2011/11/15 dadams
;; Added: redefinition of Info-find-file for Emacs 23+, to handle virtual books.
;; 2011/08/23 dadams
;; Removed hard-code removal of info from same-window-(regexps|buffer-names). Thx to PasJa.
;; 2011/02/06 dadams
;; info-user-option-ref-item: Corrected background for light-bg case.
;; 2011/02/03 dadams
;; All deffaces: Provided default values for dark-background screens too.
;; 2011/01/04 dadams
;; Removed autoload cookies from non def* sexps. Added for defgroup and defface.
;; 2010/05/27 dadams
;; Added: Info-set-mode-line.
;; Info-find-node-2:
;; Added redefinition of it for Emacs 23.2 (they keep twiddling it).
;; Do not call Info-insert-breadcrumbs-in-mode-line. Do that in Info-set-mode-line now.
;; 2010/04/06 dadams
;; Added: Info-breadcrumbs-in-header-flag, Info-toggle-breadcrumbs-in-header-line,
;; Info-breadcrumbs-in-mode-line-mode, Info-set-breadcrumbs-depth,
;; Info-insert-breadcrumbs-in-mode-line, Info-breadcrumbs-depth-internal.
;; Added to Info-mode-menu (Emacs 23+): Info-breadcrumbs-in-mode-line-mode.
;; Info-find-node-2 (Emacs 23+): Add breadcrumbs to header line & mode line only according to vars.
;; Info-fontify-node (Emacs 23+): Handle breadcrumbs in header only if flag says to.
;; 2010/01/12 dadams
;; Info-find-node for Emacs 20, Info-find-node-2 for Emacs 21, 22, Info-search:
;; save-excursion + set-buffer -> with-current-buffer.
;; 2010/01/10 dadams
;; Info-find-node-2 for Emacs 23+: Updated for Emacs 23.2 (pretest) - virtual function stuff.
;; 2009/12/13 dadams
;; Typo: Incorrectly used Emacs 22 version for Emacs 21 also.
;; 2009/12/11 dadams
;; info-fontify-(node|quotations|reference-items), Info-merge-subnodes:
;; Use font-lock-face property, not face, if > Emacs 21.
;; 2009/08/03 dadams
;; Updated for Emacs 23.1 release: Info-find-node-2, Info-fontify-node, Info-search: new version.
;; 2009/06/10 dadams
;; Added: Info-fontify-reference-items-flag, Info-mode-syntax-table.
;; Info-mode: Use Info-mode-syntax-table, not text-mode-syntax-table.
;; Info-fontify-node: Fontify ref items if *-reference-items-flag, not just for Elisp manual.
;; Renamed: info-elisp-* to info-*.
;; 2009/06/09 dadams
;; info-fontify-quotations: Allow \ before ', just not before`.
;; 2009/06/08 dadams
;; info-fontify-quotations: Rewrote, using better regexp. Don't fontify escaped ` or '.
;; Fontify `\', `\\', etc. Respect Info-fontify-single-quote-flag.
;; Added: info-single-quote, Info-fontify-single-quote-flag, info-quotation-regexp.
;; info-quoted-name: Changed face spec to (:inherit font-lock-string-face :foreground "DarkViolet")
;; 2009/05/25 dadams
;; Info-virtual-book: Treat info-node bookmarks too.
;; 2009/05/23 dadams
;; Added: Info-mode for Emacs 23.
;; They added Info-isearch-filter, Info-revert-buffer-function, Info-bookmark-make-record.
;; 2009/05/22 dadams
;; Added: Info-saved-nodes, Info-save-current-node, Info-virtual-book. Added to Info-mode-menu.
;; Bind info-apropos, Info-save-current-node, Info-virtual-book to a, ., and v.
;; Info-mode: Updated doc string.
;; 2009/04/26 dadams
;; Info-merge-subnodes: Bind inhibit-field-text-motion to t, for end-of-line.
;; 2008/10/07 dadams
;; Require cl.el at compile time for all Emacs versions, because of case.
;; 2008/10/05 dadams
;; Added: Info-read-node-name-1, Info-read-node-name-2.
;; 2008-07-11 dadams
;; Info-fontify-node (Emacs 22+): Protect histories when getting ancestor nodes for breadcrumbs.
;; (Emacs 22+) Don't change faces info-menu-header, *-title-*, *(-header)-node, header-line.
;; (Emacs 20, 21): Removed bold and italic attributes from info-node and info-xref.
;; Removed commented out defface for info-xref and info-node.
;; Face info-file: Blue, not DarkBlue, foreground, by default.
;; 2008/06/12 dadams
;; Info-fontify-node (Emacs 22+):
;; Prevent infinite recursion from Info-goto-node calling Info-fontify-node.
;; Fixed for nil Info-hide-note-references.
;; 2008/06/10 dadams
;; Info-fontify-node (Emacs 22+): Added breadcrumbs.
;; 2008/03/06 dadams
;; info-mode: Use fboundp for Info-clone-buffer, not version test, for Emacs 22+. Thx to Sebastien Vauban.
;; 2008/02/01 dadams
;; Info-mode: Renamed Info-clone-buffer-hook to Info-clone-buffer for Emacs 22.1.90.
;; 2008/01/08 dadams
;; Info-search (Emacs 22): Removed phony pred arg.
;; 2008/01/06 dadams
;; Removed soft require of Icicles due to cirular dependency. Thx to Tennis Smith.
;; 2007/11/27 dadams
;; Info-search: Use icicle-read-string-completing, if available.
;; Added soft require Icicles.
;; 2007/11/20 dadams
;; Info-subtree-separator: Escaped slashes in doc string: \f -> \\f.
;; 2007/09/26 dadams
;; Better default color for info-quoted-name. Added group face to all deffaces.
;; 2007/09/25 dadams
;; Bound Info-mouse-*-new-* to S-down-mouse-2, not S-mouse-2, because of mouse-scan-lines-or-M-:.
;; Info-goto-emacs-command-node: Convert completion default value to string.
;; 2007/08/27 dadams
;; Info-fontify-node: Ensure Info-fontify-node is a string when fontifiy quotations. Updated for Emacs 22.
;; 2007/07/13 dadams
;; Info-find-node: Redefine only for Emacs < 21.
;; 2006/09/15 dadams
;; Info-mouse-follow-nearest-node redefinition is only for Emacs >= 22.
;; Changed Emacs 22 tests to just (>= emacs-major-version 22).
;; Bind tool-bar-map for Emacs 21. Otherwise, binding of [tool-bar] gives an error (why?).
;; 2006/08/18 dadams
;; Everywhere: Corrected previous change: minibuffer-selected-window to window-minibuffer-p.
;; 2006/08/14 dadams
;; Everywhere: fit-frame only if not a minibuffer window.
;; 2006/08/12 dadams
;; Info-merge-subnodes: Bug fixes:
;; Added concat for insertion of main node when recursive-display-p is negative.
;; Don't recurse down Index menus.
;; When checking for subnodes menu, check for nonfile menu item also.
;; After come back from recursion, go back to Info buffer before trying to go back in history.
;; Call fit-frame at end.
;; 2006/06/10 dadams
;; Added: Info(-mouse)-follow-nearest-node-new-window. Bound to S-RET, S-mouse-2.
;; 2006/03/31 dadams
;; info-menu-header: Removed :underline, because links are underlined in Emacs 22.
;; No longer use display-in-minibuffer.
;; 2006/01/08 dadams
;; Added: redefinition of Info-mouse-follow-nearest-node.
;; 2006/01/07 dadams
;; Added :link for sending bug report.
;; 2006/01/06 dadams
;; Added defgroup Info-Plus and used it. Added :link.
;; 2005/12/30 dadams
;; Moved everything from setup-info.el to here, after getting rid of some of it.
;; Use defface for all faces. Renamed faces, without "-face".
;; Use minibuffer-prompt face, not info-msg-face.
;; No longer require setup-info.el. No longer require cl.el when compile.
;; 2005/11/21 dadams
;; Info-search for Emacs 22: Don't display repeat `s' message if isearch-mode.
;; 2005/11/09 dadams
;; Info-fontify-node: Updated to reflect latest CVS (replaced Info-escape-percent header).
;; 2005/10/31 dadams
;; Use nil as init-value arg in calls to completing-read, everywhere.
;; 2005/07/04 dadams
;; info-fontify-quotations: Use font-lock-face property, instead of face, for Emacs 22.
;; Wrap re-search-forward in condition-case for stack overflow.
;; 2005/07/02 dadams
;; Info-search: fit-frame. Added Emacs 22 version too.
;; Info-goto-emacs-command-node, Info-goto-emacs-key-command-node, Info-merge-subnodes:
;; Use Info-history-back for Emacs 22.
;; Info-mode: Added Emacs 22 version.
;; 2005/06/23 dadams
;; Info-fontify-node: Fontify reference items if in Emacs-Lisp manual.
;; Added: info-fontify-reference-items
;; 2005/05/17 dadams
;; Updated to work with Emacs 22.x.
;; 2004/11/20 dadams
;; Info-find-emacs-command-nodes: bug fix: regexp (cmd-desc) was only for Emacs 21.
;; Refined to deal with Emacs 21 < 21.3.50 (soon to be 22.x)
;; 2004/10/09 dadams
;; info-fontify-quotations:
;; 1) Allow all characters inside `...'.
;; 2) Treat case of "..." preceded by backslashes
;; Info-fontify-node (for Emacs 21): Moved info-fontify-quotations before fontification of titles.
;; 2004/10/07 dadams
;; Renamed Info-resize-frame-p to Info-fit-frame-flag.
;; 2004/10/05 dadams
;; Improved regexp treatment further for fontifying quotations.
;; 2004/10/04 dadams
;; Improved regexp treatment for fontifying quotations.
;; Added info-fontify-quotations. Removed info-fontify-strings-p.
;; Renamed Info-fontify-quotations-p to Info-fontify-quotations-flag.
;; 2004/10/03/dadams
;; Major update: updated to work with Emacs 21 also.
;; Made require of setup-info.el mandatory.
;; Removed all variables and keys to setup-info.el.
;; Renamed to Emacs 21 names and only define for Emacs < 21: emacs-info -> info-emacs-manual.
;; 2004/09/28 dadams
;; Removed dir-info (same as Info-directory).
;; Renamed to Emacs 21 names and only define for Emacs < 21: emacs-lisp-info -> menu-bar-read-lispref
;; 2004/06/01 dadams
;; Renamed: Info-fit-frame-p to Info-resize-frame-p, shrink-frame-to-fit to resize-frame.
;; 2000/09/27 dadams
;; 1. Added: Info-fit-frame-p.
;; 2. Info-find-node: added shrink-frame-to-fit.
;; 1999/04/14 dadams
;; Info-fontify-node: Fontify indexes too.
;; 1999/04/14 dadams
;; 1. Added vars: info-file-face, info-menu-face, info-node-face, info-quoted-name-face, info-string-face,
;; info-xref-face.
;; 2. No longer use (or define) faces: info-node, info-file, info-xref, info-menu-5, info-quoted-name,
;; info-string.
;; 3. Info-fontify-node: Use new face variables instead of faces in #2, above.
;; Corrected: node names in info-node-face (was xref). Use info-menu-face for * and menu item.
;; 4. Info-mode: Redefined like original, but: no make-face's; use face vars.
;; Added user options description to doc string.
;; 1999/04/08 dadams
;; Info-goto-emacs-key-command-node: regexp-quote pp-key for Info-search.
;; 1999/04/07 dadams
;; Info-goto-emacs-key-command-node: a) msgs only if interactive, b) return nil if not found, else non-nil,
;; c) "is undefined" -> "doc not found", d) use display-in-minibuffer more, e) corrected error handler.
;; 1999/04/01 dadams
;; 1. Added: (remove-hook 'same-window-buffer-names "*info*").
;; 2. Info-find-node: switch-to-buffer-other-window -> pop-to-buffer.
;; 1999/03/31 dadams
;; 1. Added (put 'Info-goto-emacs-(key-)command-node 'info-file "emacs").
;; 2. Info-find-node: Mention searched file in error messages.
;; 3. Added (replacement): Info-find-emacs-command-nodes, with progress msg.
;; 4. a. Info-goto-emacs-key-command-node: Use global-map, unless menu item.
;; b. Added message "Not found using Index ...".
;; 1999/03/31 dadams
;; 1. Info-goto-emacs(-key)-command-node: Only display-in-minibuffer if
;; interactive-p.
;; 2. Info-goto-emacs-key-command-node: Messages: "key"; other entries.
;; 1999/03/31 dadams
;; 1. Added (put 'info 'info-file "emacs") so find doc on `info' cmd.
;; 2. Info-goto-emacs-command-node:
;; a. Added message when =< 1 match.
;; b. Return num-matches if found.
;; c. Uses `display-in-minibuffer' instead of `message'.
;; 3. a. Wrapped call to Info-search in condition-case, not if.
;; b. Info-goto-emacs-key-command-node: Return num-matches if found.
;; 1999/03/30 dadams
;; 1. Added Info menu bar menu.
;; 2. Info-goto-emacs-command-node: Only error if interactive-p.
;; 3. Info-goto-emacs-key-command-node:
;; a. Print key in msgs
;; b. If Info-goto-emacs-command-node doesn't find it, then try Info-search.
;; If found & interactive-p, then msg ("repeat"). Else error.
;; 4. Info-search: Msg ("repeat") if found & interactive-p.
;; 1999/03/17 dadams
;; 1. Updated to correspond with Emacs 34.1 version.
;; 2. Protect with fboundp.
;; 1996/07/11 dadams
;; Added redefinitions of Info-goto-emacs-(key-)command-node.
;; 1996/04/26 dadams
;; Put escaped newlines on long-line strings.
;; 1996/04/16 dadams
;; Added: info-file, info-quoted-name, info-string, Info-fontify-quotations-flag, info-fontify-strings-p.
;; Take into account in Info-fontify-node.
;; 1996/02/23 dadams
;; 1. Changed binding of Info-merge-subnodes back to `r', but now requires user confirmation when invoked.
;; 2. Info-subtree-separator: Incorporates "\n* ". variable-interactive prop.
;; 1996/02/22 dadams
;; display-Info-node-subtree:
;; 1. display-Info-node-subtree -> Info-merge-subnodes (renamed).
;; 2. Changed binding of Info-merge-subnodes from `r' to `C-d'.
;; 3. Don't pick up text between menu-item-line and "\n* ". Hardwire "\n* ".
;; 4. Untabify menu-item-line, so can count chars to underline.
;; 5. indent-rigidly, not indent-region.
;; 1996/02/22 dadams
;; 1. Bind describe-mode and display-Info-node-subtree.
;; 2. Added redefinition of Info-mode: Only the doc string was changed.
;; 3. Added Info-subtree-separator.
;; 3. display-Info-node-subtree: Info-subtree-separator. Doc. Garbage-collect.
;; 1996/02/22 dadams
;; Info-merge-subnodes: Rewrote, adding optional args. Renamed (defaliased) to display-Info-node-subtree.
;; 1996/02/22 dadams
;; Added redefinition of Info-merge-subnodes (cleanup, corrections).
;; 1996/02/20 dadams
;; 1. Make info-node, info-xref, info-menu-5 here. (Diff faces than before.)
;; 2. Added redefinition of Info-find-node. (Uses other window.)
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; 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 'info)
(eval-when-compile (require 'cl)) ;; case
;; These are optional, for cosmetic purposes.
(require 'thingatpt nil t) ;; (no error if not found): symbol-at-point
(when (and (require 'thingatpt+ nil t) ;; (no error if not found): symbol-nearest-point
(fboundp 'tap-put-thing-at-point-props)) ; >= 2012-08-21
(tap-define-aliases-wo-prefix)
(tap-put-thing-at-point-props))
(require 'strings nil t) ;; (no error if not found): concat-w-faces
(require 'fit-frame nil t) ;; (no error if not found): fit-frame
;; Took this out because it leads to a circular `require' dependency.
;; (require 'icicles nil t) ;; (no error if not found): icicle-read-string-completing
;; Quiet the byte compiler a bit.
;;
(defvar browse-url-new-window-flag) ; In `browse-url.el'
(defvar desktop-save-buffer)
(defvar header-line-format)
(defvar Info-breadcrumbs-depth)
(defvar Info-breadcrumbs-depth-internal)
(defvar Info-breadcrumbs-in-header-flag)
(defvar Info-breadcrumbs-in-mode-line-mode)
(defvar Info-current-node-virtual)
(defvar isearch-filter-predicate)
(defvar Info-bookmarked-node-xref-faces) ; Here, Emacs 24.2+, with Bookmark+.
(defvar Info-fontify-bookmarked-xrefs-flag) ; Here, Emacs 24.2+, with Bookmark+.
(defvar Info-fontify-visited-nodes)
(defvar Info-hide-note-references)
(defvar Info-history-list)
(defvar Info-history-skip-intermediate-nodes) ; Emacs 24+
(defvar Info-isearch-initial-node)
(defvar Info-isearch-search)
(defvar Info-last-search)
(defvar Info-link-keymap)
(defvar Info-menu-entry-name-re)
(defvar Info-next-link-keymap)
(defvar Info-mode-line-node-keymap)
(defvar Info-node-spec-re)
(defvar Info-persist-history-mode)
(defvar Info-point-loc)
(defvar Info-prev-link-keymap)
(defvar Info-read-node-completion-table)
(defvar Info-refill-paragraphs)
(defvar Info-saved-history-file)
(defvar Info-saved-nodes)
(defvar Info-search-case-fold)
(defvar Info-search-history)
(defvar Info-search-whitespace-regexp)
(defvar info-tool-bar-map)
(defvar Info-up-link-keymap)
(defvar Info-use-header-line)
(defvar isearch-lax-whitespace) ; In `isearch.el'.
(defvar isearch-regexp-lax-whitespace) ; In `isearch.el'.
(defvar infop-node-name) ; Here, in `Info-merge-subnodes'.
(defvar outline-heading-alist) ; In `outline.el'.
(defvar widen-automatically)
;;;;;;;;;;;;;;;;;;;;
(provide 'info+)
(require 'info+) ;; Ensure loaded before compiling.
;;;;;;;;;;;;;;;;;;;;
;;(@* "Macros")
;;; Macros -----------------------------------------------------------
(defmacro info-user-error (&rest args)
"`user-error' if defined, otherwise `error'."
`(if (fboundp 'user-error) (user-error ,@args) (error ,@args)))
(font-lock-add-keywords
'emacs-lisp-mode
'(("(\\(info-user-error\\)\\>" 1 font-lock-warning-face)))
;;; KEYS & MENUS ;;;;;;;;;;;;;;;;;;;;;;;;
(define-key Info-mode-map "?" 'describe-mode) ; Don't use `Info-summary'.
(define-key Info-mode-map "+" 'Info-merge-subnodes)
(define-key Info-mode-map "." 'Info-save-current-node)
(define-key Info-mode-map "a" 'info-apropos)
(define-key Info-mode-map "G" 'Info-goto-node-web)
(define-key Info-mode-map "O" 'Info-toc-outline)
(define-key Info-mode-map "v" 'Info-virtual-book)
(define-key Info-mode-map (kbd "C-x DEL") 'Info-change-visited-status)
;; Mouse back and forward buttons
(define-key Info-mode-map [S-down-mouse-2] 'Info-mouse-follow-nearest-node-new-window)
(define-key Info-mode-map [S-return] 'Info-follow-nearest-node-new-window)
(define-key Info-mode-map [mouse-4] 'Info-history-back)
(define-key Info-mode-map [mouse-5] 'Info-history-forward)
;;(@* "Faces (Customizable)")
;;; Faces (Customizable) ---------------------------------------------
;;;###autoload
(defgroup Info-Plus nil
"Various enhancements to Info."
:group 'info
:link `(url-link :tag "Send Bug Report"
,(concat "mailto:" "drew.adams" "@" "oracle" ".com?subject=\
info+.el bug: \
&body=Describe bug here, starting with `emacs -q'. \
Don't forget to mention your Emacs and library versions."))
:link '(url-link :tag "Other Libraries by Drew"
"https://www.emacswiki.org/emacs/DrewsElispLibraries")
:link '(url-link :tag "Download" "https://www.emacswiki.org/emacs/download/info%2b.el")
:link '(url-link :tag "Description" "https://www.emacswiki.org/emacs/InfoPlus")
:link '(emacs-commentary-link :tag "Commentary" "info+")
)
;; FWIW, I use a `LightSteelBlue' background for `*info*', and I use `red3' for this face.
;;;###autoload
(defface info-double-quoted-name ; For “...”
'((((background dark)) (:inherit font-lock-string-face :foreground "Cyan"))
(t (:inherit font-lock-string-face :foreground "DarkOrange")))
"*Face for names enclosed in curly double-quotes (“...”) in `info'."
:group 'Info-Plus :group 'faces)
;;;###autoload
(defface info-emphasis ; For _..._
'((t (:inherit italic)))
"*Face for emphasizing text enclosed with underscores (_..._) in `info'."
:group 'Info-Plus :group 'faces)
;;;###autoload
(defface info-file
'((((background dark)) (:foreground "Yellow" :background "DimGray"))
(t (:foreground "Blue" :background "LightGray")))
"*Face for file heading labels in `info'." :group 'Info-Plus :group 'faces)
;;;###autoload
(defface info-menu
'((((background dark)) (:foreground "Yellow"))
(t (:foreground "Blue")))
"*Face used for menu items in `info'." :group 'Info-Plus :group 'faces)
;; FWIW, I use a `LightSteelBlue' background for `*info*', and I use `yellow' for this face.
;;;###autoload
(defface info-quoted-name ; For ... and `...'
'((((background dark)) (:inherit font-lock-string-face :foreground "#6B6BFFFF2C2C")) ; ~ bright green
(((background light)) (:inherit font-lock-string-face :foreground "DarkViolet"))
(t (:foreground "yellow")))
"*Face for quoted names (... or `...') in `info'."
:group 'Info-Plus :group 'faces)
;; FWIW, I use a `LightSteelBlue' background for `*info*', and I use `red3' for this face.
;;;###autoload
(defface info-string ; For "..."
'((((background dark)) (:inherit font-lock-string-face :foreground "Orange"))
(t (:inherit font-lock-string-face :foreground "red3")))
"*Face for strings (\"...\") in `info'."
:group 'Info-Plus :group 'faces)
;;;###autoload
(defface info-single-quote ; For '
'((((background dark)) (:inherit font-lock-keyword-face :foreground "Green"))
(t (:inherit font-lock-keyword-face :foreground "Magenta")))
"*Face for isolated single-quote marks (') in `info'."
:group 'Info-Plus :group 'faces)
;; Standard faces from vanilla Emacs `info.el', but without `:weight', `:height' and `:inherit'.
;;;###autoload
(defface info-title-1
'((((type tty pc) (class color) (background dark)) :foreground "yellow" :weight bold)
(((type tty pc) (class color) (background light)) :foreground "brown" :weight bold))
"*Face for info titles at level 1."
:group 'info)
;;;###autoload
(defface info-title-2
'((((type tty pc) (class color)) :foreground "lightblue" :weight bold))
"*Face for info titles at level 2."
:group 'info)
;;;###autoload
(defface info-title-3
'((((type tty pc) (class color)) :weight bold))
"*Face for info titles at level 3."
:group 'info)
;;;###autoload
(defface info-title-4
'((((type tty pc) (class color)) :weight bold))
"*Face for info titles at level 4."
:group 'info)
;;; Faces for highlighting reference items
;;;###autoload
(defface info-command-ref-item
'((((background dark)) (:foreground "#7474FFFF7474" :background "DimGray")) ; ~ light green
(t (:foreground "Blue" :background "LightGray")))
"*Face used for \"Command:\" reference items in `info' manual."
:group 'Info-Plus :group 'faces)
;;;###autoload
(defface info-constant-ref-item
'((((background dark))
(:foreground "DeepPink" :background "DimGray"))
(t (:foreground "DeepPink" :background "LightGray")))
"*Face used for \"Constant:\" reference items in `info' manual."
:group 'Info-Plus :group 'faces)
;;;###autoload
(defface info-function-ref-item
'((((background dark))
(:foreground "#4D4DDDDDDDDD" :background "DimGray")) ; ~ cyan
(t (:foreground "DarkBlue" :background "LightGray")))
"*Face used for \"Function:\" reference items in `info' manual."
:group 'Info-Plus :group 'faces)
;;;###autoload
(defface info-macro-ref-item
'((((background dark))
(:foreground "Yellow" :background "DimGray"))
(t (:foreground "DarkMagenta" :background "LightGray")))
"*Face used for \"Macro:\" reference items in `info' manual."
:group 'Info-Plus :group 'faces)
;;;###autoload
(defface info-reference-item
'((((background dark)) (:background "DimGray"))
(t (:background "LightGray")))
"*Face used for reference items in `info' manual."
:group 'Info-Plus :group 'faces)
;;;###autoload
(defface info-special-form-ref-item
'((((background dark))
(:foreground "Yellow" :background "DimGray"))
(t (:foreground "DarkMagenta" :background "LightGray")))
"*Face used for \"Special Form:\" reference items in `info' manual."
:group 'Info-Plus :group 'faces)
;;;###autoload
(defface info-syntax-class-item
'((((background dark))
(:foreground "#FFFF9B9BFFFF" :background "DimGray")) ; ~ pink
(t (:foreground "DarkGreen" :background "LightGray")))
"*Face used for \"Syntax Class:\" reference items in `info' manual."
:group 'Info-Plus :group 'faces)
;;;###autoload
(defface info-user-option-ref-item
'((((background dark)) (:foreground "Red" :background "DimGray"))
(t (:foreground "Red" :background "LightGray")))
"*Face used for \"User Option:\" reference items in `info' manual."
:group 'Info-Plus :group 'faces)
;;;###autoload
(defface info-variable-ref-item
'((((background dark))
(:foreground "Orange" :background "DimGray"))
(t (:foreground "FireBrick" :background "LightGray")))
"*Face used for \"Variable:\" reference items in `info' manual."
:group 'Info-Plus :group 'faces)
(when (and (require 'bookmark+ nil t) ; Emacs 24.2+ (do not bother for Emacs 23-24.1)
(or (> emacs-major-version 24) (and (= emacs-major-version 24) (> emacs-minor-version 1))))
(defface info-xref-bookmarked
'((((background dark)) (:foreground "violet"))
(t (:foreground "DarkGreen")))
"Face for bookmarked Info nodes."
:group 'Info-Plus :group 'faces)
)
;;(@* "User Options (Customizable)")
;;; User Options (Customizable) --------------------------------------
;;;###autoload
(defcustom Info-breadcrumbs-in-header-flag nil
"*Non-nil means breadcrumbs are shown in the header line."
:type 'boolean :group 'Info-Plus)
;;;###autoload
(defcustom Info-display-node-header-fn 'Info-display-node-default-header
"*Function to insert header by `Info-merge-subnodes'."
:type 'function :group 'Info-Plus)
;;;###autoload
(defcustom Info-emphasis-regexp
"_\\(\\(\\sw\\(\\s-\\|\\sw\\|\\s.\\)*\\)\\|\\(\\(\\s-\\|\\sw\\|\\s.\\)\\sw*\\)\\)_"
"Regexp to match text enclosed in underscore (`_') characters.
The default value matches the following (enclosed in underscores):
word, punctuation, and whitespace characters, plus hyphens, with at
least one word character. Hyphen is included explicitly because it
generally has symbol syntax in Info.
Some possible values include:
_\\(\\(\\sw\\(\\s-\\|\\sw\\|\\s.\\)*\\)\\|\\(\\(\\s-\\|\\sw\\|\\s.\\)\\sw*\\)\\)_ (default)
_\\(\\(\\s-\\|\\sw\\|\\s.\\)+\\)_ (word, punctuation, whitespace)
_\\(\\sw+\\)_\t\t (single words)
_\\(\\s-*\\sw+\\s-*\\)_\t (single words, maybe whitespace-separated)
_\\([^_\\n]+\\)_\t\t (anything except newlines)
_\\([^_]+\\)_\t\t (anything)
Note that any value can be problematic for some Info text - see
`Info-fontify-emphasis-flag'."
:type 'regexp :group 'Info-Plus)
;;;###autoload
(defcustom Info-fit-frame-flag t
"*Non-nil means call `fit-frame' on Info buffer."
:type 'boolean :group 'Info-Plus :group 'Fit-Frame)
(when (and (require 'bookmark+ nil t) ; Emacs 24.4+
(or (> emacs-major-version 24) (and (= emacs-major-version 24) (> emacs-minor-version 3))))
(defcustom Info-node-access-invokes-bookmark-flag t
"*Non-nil means invoke the bookmark when you access an Info node.
This applies to Info bookmarks whose names correspond to the default
name. This is normally the full node name, `(MANUAL) NODE', where
MANUAL is the lowercase name of the Info manual. For example, node
`Modes' in the Emacs manual has full name `(emacs) Modes', and the
bookmark must have that same name.
This automatic bookmark invocation can be useful to update the
bookmark data, such as the number of visits to the node."
:type 'boolean :group 'Info-Plus)
)
;;;###autoload
(defcustom Info-fontify-angle-bracketed-flag t
"*Non-nil means `info' fontifies text within <...>.
A non-nil value has no effect unless `Info-fontify-quotations-flag' is
also non-nil.
Note: This fontification can never be 100% reliable. It aims to be
useful in most Info texts, but it can occasionally result in
fontification that you might not expect. This is not a bug; it is
part of the design to be able to appropriately fontify a great variety
of texts. Set this flag to nil if you do not find this fontification
useful. You can use command `Info-toggle-fontify-angle-bracketed' to
toggle the option value."
:type 'boolean :group 'Info-Plus)
(when (and (require 'bookmark+ nil t) ; Emacs 24.2+ (do not bother for prior)
(or (> emacs-major-version 24) (and (= emacs-major-version 24) (> emacs-minor-version 1))))
(defcustom Info-bookmarked-node-xref-faces '()
"List of faces to use to classify bookmarked nodes.
The faces are used for links to bookmarked nodes. They classify nodes
by serving as the values of bookmark tag \"bmkp-info-face\".
You can use any face for such a link. The faces in this option list
are just provided as defaults when you are asked to enter a face for a
node link. "
:type '(repeat face) :group 'Info-Plus)
(defcustom Info-fontify-bookmarked-xrefs-flag t
"Non-nil means fontify references to bookmarked nodes.
The face used is `info-xref-bookmarked'."
:type 'boolean :group 'Info-Plus)
)
;;;###autoload
(defcustom Info-fontify-emphasis-flag t
"*Non-nil means `info' fontifies text between underscores (`_').
The text that is highlighted matches the value of option
`Info-emphasis-regexp'.
Note 1:
This fontification hides the underscores that surround text that is
emphasized. Because this fontification is not 100% reliable (see Note
2), in cases where it is inappropriate or unhelpful you might want to
see the hidden underscore characters. You can toggle showing all
hidden text (not just hidden underscores) using `M-x visible-mode'.
See (info) `Help-Inv' for more information about this.
Note 2:
This fontification can never be 100% reliable. It aims to be useful
in most Info texts, but it can occasionally result in fontification
that you might not expect. This is not a bug; it is part of the
design to be able to appropriately fontify a great variety of texts.
Set this flag to nil if you do not find this fontification useful.
You can use command `Info-toggle-fontify-emphasis' to toggle the
option value.
Note 3:
If internal variable `info-fontify-emphasis' is `nil' then emphasis is
never highlighted, and this option has no effect. This gives you a
way to turn off all matching of `Info-emphasis-regexp'."
:type 'boolean :group 'Info-Plus)
;;;###autoload
(defcustom Info-fontify-quotations-flag t
"*Non-nil means `info' fontifies text between quotes.
This applies to double-quoted text (... or \"...\") and text
between single-quotes (... or `...').
Note: This fontification can never be 100% reliable. It aims to be
useful in most Info texts, but it can occasionally result in
fontification that you might not expect. This is not a bug; it is
part of the design to be able to appropriately fontify a great variety
of texts. Set this flag to nil if you do not find this fontification
useful. You can use command `Info-toggle-fontify-quotations' to
toggle the option value."
:type 'boolean :group 'Info-Plus)
;;;###autoload
(defcustom Info-fontify-reference-items-flag t
"*Non-nil means `info' fontifies reference items such as \"Function:\"."
:type 'boolean :group 'Info-Plus)
(when (fboundp 'advice-add) ; Emacs 24.4+
(defcustom Info-saved-history-file (locate-user-emacs-file "info-history" ".emacs.info-history")
"File where `Info-persist-history-mode' saves `Info-history-list'."
:type '(file :must-match t) :group 'Info-Plus)
)
;;;###autoload
(defcustom Info-saved-nodes ()
"*List of Info node names you can visit using `\\<Info-mode-map>\\[Info-virtual-book]'.
Each node name is a string. The node name can be absolute, including
a filename, such as \"(emacs)Basic\", or it can be relative, such as
\"Basic\".
You can customize this option, but you can also add node names to it
easily using `\\<Info-mode-map>\\[Info-save-current-node]'."
:type '(repeat (string :tag "Node name")) :group 'info)
;;;###autoload
(defcustom Info-fontify-single-quote-flag t
"*Non-nil means `info' fontifies ' when not preceded by `....
A non-nil value has no effect unless `Info-fontify-quotations-flag' is
also non-nil.
Note: This fontification can never be 100% reliable. It aims to be
useful in most Info texts, but it can occasionally result in
fontification that you might not expect. This is not a bug; it is
part of the design to be able to appropriately fontify a great variety
of texts. Set this flag to nil if you do not find this fontification
useful. You can use command `Info-toggle-fontify-single-quote' to
toggle the option value."
:type 'boolean :group 'Info-Plus)
;;;###autoload
(defcustom Info-subtree-separator "\n* "
"*A string used to separate Info node descriptions.
Inserted by `Info-merge-subnodes' just before each node title.
Setting this to a string that includes a form-feed (^L), such as
\"\\f\\n* \", will cause a page break before each node description.
Use command `set-variable' to set this, quoting any control characters
you want to include, such as form-feed (^L) and newline (^J), with ^Q.
For example, type `^Q^L^Q^J* ' to set this to \"\\f\\n* \"."
:type 'string :group 'Info-Plus)
;;;###autoload
(defcustom Info-toc-outline-no-redundancy-flag t
"Non-nil means `Info-toc-outline' TOC has no redundancy.
If nil then section headings from the TOC manual are included, and
nodes can be repeated because they are in more than one section."
:type 'boolean :group 'Info-Plus)
;;(@* "Internal Variables")
;;; Internal Variables -----------------------------------------------
(defvar info-fontify-emphasis t
"Non-nil means allow `Info-fontify-emphasis-flag' to work.
If nil then emphasis is never fontified, regardless of that flag.")
;; I reported this as Emacs bug #3312. If it gets fixed, this can be removed.
(defvar Info-mode-syntax-table
(let ((table (copy-syntax-table text-mode-syntax-table)))
(modify-syntax-entry ?' "." table) ; Punctuation syntax for apostrophe (').
(modify-syntax-entry ?\240 "." table) ; Punctuation syntax for non-breaking space.
table)
"Syntax table for `info'.")
(defvar Info-merged-map nil "Keymap for merged Info buffer.")
(if Info-merged-map
nil
(setq Info-merged-map (make-keymap))
(suppress-keymap Info-merged-map)
(define-key Info-merged-map "." 'beginning-of-buffer)
(define-key Info-merged-map "\t" 'Info-next-reference)
(define-key Info-merged-map "\e\t" 'Info-prev-reference)
(define-key Info-merged-map "b" 'beginning-of-buffer)
(define-key Info-merged-map "q" 'quit-window)
(define-key Info-merged-map "s" 'nonincremental-re-search-forward)
(define-key Info-merged-map "\M-s" 'nonincremental-re-search-forward))
(defvar Info-breadcrumbs-depth-internal Info-breadcrumbs-depth
"Current breadcrumbs depth for Info.")
;; Match has, inside “...”, "...", ..., or `...', zero or more of these characters:
;; - any character except ”, ", , or ', respectively
;; - \ followed by any character
;;
;; The `... in `...' is optional, so the regexp can also match just '.
;;
;; The regexp matches also ..., `...', “...”, and "..." where at least one of the
;; , , `, ', “, ”, or " is escaped by a backslash.
;; So we check those cases explicitly and do not highlight them.
;;
(defvar info-quotation-regexp
(concat
"\"\\(?:[^\"\\]\\|\\\\\\(?:.\\|\n\\)\\)*\"\\|" ; "..."
"`\\(?:[^']\\|\\\\\\(?:.\\|\n\\)\\)*'\\|" ; `...'
"\\(?:[^]\\|\\\\\\(?:.\\|\n\\)\\)*\\|" ; ...
"\\(?:[^”]\\|\\\\\\(?:.\\|\n\\)\\)*”" ; “...”
)
"Regexp to match `...', ..., “...”, \"...\", or just '.
If ... contains an end char then that char must be backslashed.")
;; (rx (or (seq ?\"
;; (zero-or-more (or (not (any ?\" ?\\)) (seq ?\\ anything)))
;; ?\")
;; (seq ?\`
;; (zero-or-more (or (not (any ?')) (seq ?\\ anything)))
;; ?\')
;; (seq ?
;; (zero-or-more (or (not (any ?)) (seq ?\\ anything)))
;; ?)
;; (seq ?“
;; (zero-or-more (or (not (any ?”)) (seq ?\\ anything)))
;; ?”)))
(defvar info-quoted+<>-regexp
(concat
"\"\\(?:[^\"\\]\\|\\\\\\(?:.\\|\n\\)\\)*\"\\|" ; "..."
"`\\(?:[^']\\|\\\\\\(?:.\\|\n\\)\\)*'\\|" ; `...'
"\\(?:[^]\\|\\\\\\(?:.\\|\n\\)\\)*\\|" ; ...
"\\(?:[^”]\\|\\\\\\(?:.\\|\n\\)\\)*”\\|" ; “...”
"<\\(?:[[:alpha:]][^>]*\\|\\(?:\\\\\\(?:.\\|\n\\)\\)*\\)>" ; <...>
)
"Same as `info-quotation-regexp', but matches also <...>.
If ... contains an end char then that char must be backslashed.")
;; (rx (or (seq ?\"
;; (zero-or-more (or (not (any ?\" ?\\)) (seq ?\\ anything)))
;; ?\")
;; (seq ?\`
;; (zero-or-more (or (not (any ?')) (seq ?\\ anything)))
;; ?\')
;; (seq ?
;; (zero-or-more (or (not (any ?)) (seq ?\\ anything)))
;; ?)
;; (seq ?“
;; (zero-or-more (or (not (any ?”)) (seq ?\\ anything)))
;; ?”)
;; (seq ?<
;; (or (seq (any alpha) (zero-or-more (not (any ?>))))
;; (zero-or-more (seq ?\\ anything)))
;; ?>)))
(defvar Info-toc-outline-map (let ((map (make-sparse-keymap))) (set-keymap-parent map Info-mode-map))
"Keymap for Info TOC with outlining.")
;;(@* "New Commands")
;;; New Commands -----------------------------------------------------
;; Make `Info-find-emacs-command-nodes' look for these commands in the Emacs manual.
;; In particular, don't look for command `info' in Info manual, because that has no index.
(put 'info 'info-file "emacs")
(put 'Info-goto-emacs-command-node 'info-file "emacs")
(put 'Info-goto-emacs-key-command-node 'info-file "emacs")
;;;###autoload (autoload 'Info-mouse-follow-nearest-node-new-window "info+")
(defun Info-mouse-follow-nearest-node-new-window (click)
"Open the link at the mouse pointer in a new window."
(interactive "e")
(Info-mouse-follow-nearest-node click t))
;;;###autoload (autoload 'Info-follow-nearest-node-new-window "info+")
(defun Info-follow-nearest-node-new-window ()
"Open the link near the text cursor in a new window."
(interactive)
(Info-follow-nearest-node t))
;;;###autoload (autoload 'Info-history-clear "info+")
(defun Info-history-clear (&optional msgp)
"Clear Info history and reload current manual."
(interactive (progn (unless (y-or-n-p "Clear the Info history? ") (info-user-error "OK, canceled"))
(list t)))
(setq Info-history ()
Info-history-list ())
(when (derived-mode-p 'Info-mode) (revert-buffer nil t))
(when msgp (message "Info history cleared")))
(when (fboundp 'advice-add) ; Emacs 24.4+
(define-minor-mode Info-persist-history-mode
"Automatically persist the Info history in `Info-saved-history-file'."
:init-value nil :global t :group 'Info-Plus
(cond (Info-persist-history-mode
(add-hook 'kill-emacs-hook 'Info-save-history-list)
(advice-add 'Info-kill-buffer :before 'Info-save-history-list)
(advice-add 'Info-directory :after 'Info-restore-history-list))
(t
(remove-hook 'kill-emacs-hook 'Info-save-history-list)
(advice-remove 'Info-kill-buffer 'Info-save-history-list)
(advice-remove 'Info-directory 'Info-restore-history-list))))
)
(when (> emacs-major-version 23) ; Emacs 23 `revert-buffer' invokes a brain-dead `kill-buffer' etc.
(defun Info-change-visited-status (start end &optional arg)
"Change whether the nodes in the region have been visited.
If the region is not active then act on only the node at point.
No prefix arg means toggle the visited status of each node.
A non-negative prefix arg means consider the nodes visited.
A negative prefix arg means consider the nodes not visited."
(interactive "r\nP")
(let ((toggle (not arg))
(visit (and arg (natnump (prefix-numeric-value arg))))
(unvisit (and arg (not (natnump (prefix-numeric-value arg)))))
(opoint (point))
(omark (mark))
(active mark-active)
(file (Info-find-file Info-current-file))
node visitedp trim)
(unless mark-active (setq start (setq end (point))))
(save-excursion
(goto-char start)
(while (<= (point) end)
(unless (setq node (Info-node-name-at-point)) (Info-next-reference))
(when (setq node (Info-node-name-at-point))
(save-match-data
(string-match "\\s *\\((\\s *\\([^\t)]*\\)\\s *)\\s *\\|\\)\\(.*\\)" node)
(setq node (match-string 3 node))
(setq trim (string-match "\\s +\\'" node)))
(when trim (setq node (substring node 0 trim)))
(when (or (not node) (string= node "")) (setq node "Top"))
(setq visitedp (member (list file node) Info-history-list))
(if (and visitedp (or toggle unvisit))
(setq Info-history-list (delete (list file node) Info-history-list))
(if (and (not visitedp) (or toggle visit))
(setq Info-history-list (cons (list file node) Info-history-list)))))
(when (> (forward-line 1) 0) (goto-char (point-max)))))
(when (derived-mode-p 'Info-mode) (revert-buffer nil t))
(when omark (set-mark omark))
(goto-char opoint)
(message (if toggle
"Node visited status toggled"
(format "Node visited status is now %s" (if visit 'VISITED 'UNvisited))))
(if (not active)
(deactivate-mark)
(activate-mark)
(setq deactivate-mark nil))))
)
;; Not bound. Use `Info-change-visited-status' instead.
;;
;;;###autoload (autoload 'Info-make-node-unvisited "info+")
(defun Info-make-node-unvisited (node &optional msgp)
"Reset the visited status of NODE to unvisited."
(interactive (list (or (Info-node-name-at-point) (Info-read-node-name "Node: " Info-current-node))
t))
(let (file trim)
(save-match-data
(string-match "\\s *\\((\\s *\\([^\t)]*\\)\\s *)\\s *\\|\\)\\(.*\\)" node)
(setq file (if (= (match-beginning 1) (match-end 1)) "" (match-string 2 node))
node (match-string 3 node)
trim (string-match "\\s +\\'" file))
(when trim (setq file (substring file 0 trim)))
(setq trim (string-match "\\s +\\'" node)))
(when trim (setq node (substring node 0 trim)))
(when (or (not node) (string= node "")) (setq node "Top"))
(when (or (not file) (string= file "")) (setq file Info-current-file))
(setq file (Info-find-file file)
Info-history-list (remove (list file (substring-no-properties node)) Info-history-list))
;; Emacs 23 has brain-dead `kill-buffer', which is invoked by `revert-buffer' and deletes the window/frame if dedicated.
(when (and (> emacs-major-version 23) (derived-mode-p 'Info-mode)) (revert-buffer nil t))
(when msgp (message "Node %sis now unvisited"
(if (string= "dir" Info-current-file) ""
(format "`%s%s' "
(if (equal file Info-current-file)
""
(format "(%s) " (file-name-nondirectory file)))
node))))))
;; I made this a global minor mode and turned it on by default, contrary to "the rules".
;; I did this so (a) users could easily customize it but (b) it would be on by default, otherwise.
;;
;;;###autoload (autoload 'Info-breadcrumbs-in-mode-line-mode "info+")
(define-minor-mode Info-breadcrumbs-in-mode-line-mode
"Toggle the use of breadcrumbs in Info mode line.
With arg, show breadcrumbs iff arg is positive.
Change the default behavior by customizing option
`Info-breadcrumbs-in-mode-line-mode'."
:init-value t :global t :group 'mode-line :group 'Info-Plus
(if (not Info-breadcrumbs-in-mode-line-mode)
(setq Info-breadcrumbs-depth-internal 0
mode-line-format (default-value 'mode-line-format))
(setq Info-breadcrumbs-depth-internal Info-breadcrumbs-depth)
(Info-insert-breadcrumbs-in-mode-line)))
;;;###autoload (autoload 'Info-toggle-breadcrumbs-in-header "info+")
(defun Info-toggle-breadcrumbs-in-header (&optional msgp)
"Toggle option `Info-breadcrumbs-in-header-flag'.
Toggles showing breadcrumbs in the Info header (top of the node).
This means the area at the top of the node, not the separate header
line from non-nil `Info-use-header-line'."
(interactive "p")
(setq Info-breadcrumbs-in-header-flag (not Info-breadcrumbs-in-header-flag))
(when msgp (message "Showing breadcrumbs in Info header is now %s"
(if Info-breadcrumbs-in-header-flag "ON" "OFF"))))
(defalias 'Info-toggle-breadcrumbs-in-header-line 'Info-toggle-breadcrumbs-in-header)
(make-obsolete 'Info-toggle-breadcrumbs-in-header-line 'Info-toggle-breadcrumbs-in-header "2014/03/04")
(when (boundp 'Info-node-access-invokes-bookmark-flag) ; Emacs 24.4+
(defun Info-toggle-node-access-invokes-bookmark (&optional msgp)
"Toggle option `Info-node-access-invokes-bookmark-flag'."
(interactive "p")
(setq Info-node-access-invokes-bookmark-flag (not Info-node-access-invokes-bookmark-flag))
(when (eq major-mode 'Info-mode)
(font-lock-defontify)
(let ((modp (buffer-modified-p))
(inhibit-read-only t))
(Info-fontify-node))
(when msgp (message "`Info-node-access-invokes-bookmark-flag' is now %s"
(if Info-node-access-invokes-bookmark-flag 'ON 'OFF)))))
)
(when (and (require 'bookmark+ nil t) ; Emacs 24.2+ (do not bother for prior)
(or (> emacs-major-version 24) (and (= emacs-major-version 24) (> emacs-minor-version 1))))
(defun Info-set-face-for-bookmarked-xref (node)
"Specify the face to use for Info links to bookmarked NODE.
Sets the value of bookmark tag \"bmkp-info-face\" to a face symbol you
name.
If option `Info-bookmarked-node-xref-faces' is non-nil then only the
faces in that list are available for completion, but you can enter any
face name. If that option is nil then all faces are available for
completion.
NODE defaults to the bookmarked node named at point. If none then you
are prompted for NODE."
(interactive (list (or (Info-bookmark-name-at-point) (Info-read-bookmarked-node-name))))
(let* ((alist (bmkp-info-alist-only))
(bmk (bmkp-get-bookmark-in-alist node t alist)))
(unless bmk (error "No Info bookmark for node `%s'" node))
(bmkp-set-tag-value bmk "bmkp-info-face"
(if Info-bookmarked-node-xref-faces
(intern (completing-read "Face: " Info-bookmarked-node-xref-faces
(lambda (ff) (memq ff (face-list)))))
(read-face-name "Face" 'info-xref-bookmarked)))))
(define-key Info-mode-map (kbd "C-x f") 'Info-set-face-for-bookmarked-xref)
(defun Info-toggle-fontify-bookmarked-xrefs (&optional msgp)
"Toggle option `Info-fontify-bookmarked-xrefs-flag'."
(interactive "p")
(setq Info-fontify-bookmarked-xrefs-flag (not Info-fontify-bookmarked-xrefs-flag))
(when (eq major-mode 'Info-mode)
(font-lock-defontify)
(let ((modp (buffer-modified-p))
(inhibit-read-only t))
(Info-fontify-node))
(when msgp (message "`Info-fontify-bookmarked-xrefs-flag' is now %s"
(if Info-fontify-bookmarked-xrefs-flag 'ON 'OFF)))))
)
;;;###autoload (autoload 'Info-toggle-fontify-emphasis "info+")
(defun Info-toggle-fontify-emphasis (&optional msgp)
"Toggle option `Info-fontify-emphasis-flag'."
(interactive "p")
(unless info-fontify-emphasis (error "`info-fontify-emphasis' must be non-nil to use this command"))
(setq Info-fontify-emphasis-flag (not Info-fontify-emphasis-flag))
(when (eq major-mode 'Info-mode)
(font-lock-defontify)
(let ((modp (buffer-modified-p))
(inhibit-read-only t))
(Info-fontify-node))
(when msgp (message "`Info-fontify-emphasis-flag' is now %s"
(if Info-fontify-emphasis-flag 'ON 'OFF)))))
;;;###autoload (autoload 'Info-toggle-fontify-quotations "info+")
(defun Info-toggle-fontify-quotations (&optional msgp)
"Toggle option `Info-fontify-quotations-flag'."
(interactive "p")
(setq Info-fontify-quotations-flag (not Info-fontify-quotations-flag))
(when (eq major-mode 'Info-mode)
(font-lock-defontify)
(let ((modp (buffer-modified-p))
(inhibit-read-only t))
(Info-fontify-node))
(when msgp (message "`Info-fontify-quotations-flag' is now %s"
(if Info-fontify-quotations-flag 'ON 'OFF)))))
;;;###autoload (autoload 'Info-toggle-fontify-single-quote "info+")
(defun Info-toggle-fontify-single-quote (&optional msgp)
"Toggle option `Info-fontify-single-quote-flag'."
(interactive "p")
(setq Info-fontify-single-quote-flag (not Info-fontify-single-quote-flag))
(when (eq major-mode 'Info-mode)
(font-lock-defontify)
(let ((modp (buffer-modified-p))
(inhibit-read-only t))
(Info-fontify-node))
(when msgp (message "`Info-fontify-single-quote-flag' is now %s"
(if Info-fontify-single-quote-flag 'ON 'OFF)))))
;;;###autoload (autoload 'Info-toggle-fontify-angle-bracketed "info+")
(defun Info-toggle-fontify-angle-bracketed (&optional msgp)
"Toggle option `Info-fontify-angle-bracketed-flag'."
(interactive "p")
(setq Info-fontify-angle-bracketed-flag (not Info-fontify-angle-bracketed-flag))
(when (eq major-mode 'Info-mode)
(font-lock-defontify)
(let ((modp (buffer-modified-p))
(inhibit-read-only t))
(Info-fontify-node))
(when msgp (message "`Info-fontify-angle-bracketed-flag' is now %s"
(if Info-fontify-angle-bracketed-flag 'ON 'OFF)))))
;;;###autoload (autoload 'Info-save-current-node "info+")
(defun Info-save-current-node (&optional msgp)
"Save name of current Info node to list `Info-saved-nodes'."
(interactive "p")
(unless (eq major-mode 'Info-mode) (info-user-error "You must be in Info to use this command"))
(unless Info-current-node (info-user-error "No current Info node"))
(unless Info-current-file (info-user-error "No Info file"))
(add-to-list 'Info-saved-nodes (concat "(" (file-name-nondirectory Info-current-file) ")"
Info-current-node))
(when msgp (message (format "Node `%s' saved" Info-current-node))))
(when (require 'bookmark+ nil t)
(defun Info-describe-bookmark (&optional node show-definition-p)
"Describe bookmark for NODE (default: bookmarked node named at point).
With a prefix argument, show the internal definition of the bookmark.
If there is no bookmarked node named at point then you are prompted
for the name of one.
When called from Lisp, NODE is a full node name: `(MANUAL) NODE'.
That is, it corresponds to a default Info bookmark name."
(interactive
(list (or (Info-bookmark-name-at-point) (Info-read-bookmarked-node-name))
current-prefix-arg))
(let* ((alist (bmkp-info-alist-only))
(bmk (or (bmkp-get-bookmark-in-alist node t alist)
(bmkp-read-bookmark-for-type "Info" alist nil nil 'bmkp-info-history "Describe "))))
(bmkp-describe-bookmark bmk show-definition-p)))
(defun Info-read-bookmarked-node-name (&optional localp)
"Read and return the name of a bookmarked Info node.
A bookmarked node name has the form \"(MANUAL) NODE\", referring to
NODE in MANUAL.
Optional arg LOCALP means read a node name from the current manual."
(let* ((completion-ignore-case t)
(bmks (remove-if-not
(lambda (bmk) (string-match-p (if (and localp Info-current-file)
(format "\\`(%s) "
(file-name-sans-extension
(file-name-nondirectory Info-current-file)))
"(\\([^)]+\\)) \\([^)]*\\)")
(bmkp-bookmark-name-from-record bmk)))
(bmkp-info-alist-only)))
(bmk (completing-read "Bookmarked node: " bmks nil t nil 'bmkp-info-history)))
(while (equal bmk "") (setq bmk (completing-read "Bookmarked node: " bmks nil t nil 'bmkp-info-history)))
bmk))
)
;;; Support for TOC with Outline support ---------------------
;; REPLACE ORIGINAL in `outline.el':
;;
;; See Emacs bug #28080.
;;
(defun outline-invisible-p (&optional pos)
"Non-nil if the character after point has been made invisible by Outline."
(eq (get-char-property (or pos (point)) 'invisible) 'outline))
(when (boundp 'Info-virtual-nodes) ; Emacs 23.4+
(add-to-list 'Info-virtual-nodes '("\\`\\*TOC Outline\\* (.*)\\'" (find-node . Info-toc-outline-find-node)))
(defun Info-toc-outline (&optional arg)
"Go to a node with a table of contents (TOC) in `outline-minor-mode'.
The TOC is created from the tree structure of Info menus.
In this node you can use the commands and menu items of
`outline-minor-mode' to navigate, hide/show, delete, or rearrange
parts of the TOC. Start with menu-bar menu `Outline' to see what is
possible.
* With no prefix arg:
- If buffer `*TOC Outline* (MANUAL)' exists, where MANUAL is the
manual/file name for the current manual, then pop to it.
- Else create that buffer for the current manual, and pop to it.
* With a plain prefix arg (`C-u'):
Pop to a new TOC buffer, `*TOC Outline* (MANUAL)' (or `*TOC
Outline*<N> (MANUAL)', N=1,2,3...), for the current MANUAL.
* With any other prefix arg (e.g. `M--'):
Reuse the current Info buffer, going to a node `*TOC Outline*'
for the current manual."
(interactive (list (if (consp current-prefix-arg) 'clone current-prefix-arg)))
(unless (derived-mode-p 'Info-mode) (info-user-error "You must be in Info to use this command"))
(unless Info-current-node (info-user-error "No current Info node"))
(unless Info-current-file (info-user-error "No Info file"))
(when (equal Info-current-file "dir") (info-user-error "No TOC for Info Directory - choose a manual"))
(if (and (not arg) (get-buffer (format "*TOC Outline* (%s)" (file-name-nondirectory Info-current-file))))
(pop-to-buffer (format "*TOC Outline* (%s)" (file-name-nondirectory Info-current-file)))
(when (or (not arg) (eq arg 'clone))
(clone-buffer (format "*TOC Outline* (%s)" (file-name-nondirectory Info-current-file)) t))
(Info-breadcrumbs-in-mode-line-mode -1)
(setq mark-ring ()) ;`clone-buffer' causes this to be needed, if `*info*' buffer has no mark.
(Info-find-node Info-current-file (format "*TOC Outline* (%s)" (file-name-nondirectory Info-current-file)))
(let ((prev-node (nth 1 (car Info-history)))
prev-posn)
(goto-char (point-min))
(when (setq prev-posn (search-forward (concat "*Note " prev-node ":") nil t))
(setq prev-posn (- prev-posn (length prev-node) 2)))
(goto-char (or prev-posn (point-min))))))
(defun Info-toc-outline-find-node (filename nodename &optional _no-going-back)
"TOC-specific implementation of `Info-find-node-2'."
(let* ((curr-file (substring-no-properties (or filename Info-current-file)))
(curr-node (substring-no-properties (or nodename Info-current-node)))
(node-list (Info-toc-nodes curr-file)))
(insert (format "\n\^_\nFile: %s, Node: %s, Up: Top\n\n" curr-file curr-node))
(let ((title (format "Contents (%s)" (file-name-nondirectory curr-file))))
(insert title "\n" (make-string (length title) ?*) "\n\n"))
(insert "*Note Top::\n")
(Info-toc-insert (nth 3 (assoc "Top" node-list)) node-list 0 curr-file) ; `Top' nodes
(unless (bobp)
(let ((Info-hide-note-references 'hide)
(Info-fontify-visited-nodes nil))
(setq Info-current-file filename
Info-current-node (format "*TOC Outline* (%s)" (file-name-nondirectory curr-file)))
(goto-char (point-min))
(narrow-to-region (or (re-search-forward "\n[\^_\f]\n" nil t) (point-min)) (point-max))
(Info-fontify-node)
(widen))))
(set (make-local-variable 'inhibit-read-only) t)
(goto-char (point-min))
(search-forward "Contents" nil t)
(forward-line 3)
;; If `Info-toc-outline-no-redundancy-flag' is non-nil then remove redundancies. Else give non-links face `info-title-4'.
(save-excursion
(let ((note-re "^[\t]*[*]Note "))
(while (not (eobp))
(if (re-search-forward note-re (line-end-position) t)
(if (and Info-toc-outline-no-redundancy-flag ; Remove node line if already listed.
(let ((node (buffer-substring-no-properties (point) (line-end-position))))
(save-excursion (beginning-of-line) (re-search-backward (concat note-re node) nil t))))
(delete-region (line-beginning-position) (line-beginning-position 2))
(save-excursion ; Indent previous line to same column, if it was a heading.
(let ((col (current-column)))
(forward-line -1)
(unless (re-search-forward note-re (line-end-position) t)
(insert (propertize " " 'display `(space :align-to ,col))))))
(forward-line 1))
(if Info-toc-outline-no-redundancy-flag
(delete-region (line-beginning-position) (line-beginning-position 2))
;; The line is a section heading. Put face `info-title-4' on it.
(put-text-property (line-beginning-position 1) (line-end-position 1) 'face 'info-title-4)
(unless (looking-at "\\s-") (insert "\n"))
(forward-line 1))))))
(outline-minor-mode 1)
(define-key Info-toc-outline-map [remap outline-promote] 'Info-outline-promote)
(define-key Info-toc-outline-map [remap outline-demote] 'Info-outline-demote)
(define-key Info-toc-outline-map "\C-x\M-l" 'Info-toc-outline-refontify-region)
(use-local-map Info-toc-outline-map)
(setq outline-regexp "[\t]*[*]Note ") ; Include no "^" here.
(set (make-local-variable 'Info-hide-note-references) 'hide)
(add-hook 'post-command-hook 'Info-toc-outline-refontify-region nil 'LOCAL)
(buffer-enable-undo))
(defun Info-toc-outline-refontify-region (&optional start end forcep)
"In Info `*TOC Outline*' buffer, refontify region.
Interactively, if region is not active or is empty, refontify buffer.
From Lisp:
* Do nothing if not in a `*TOC Outline* buffer or if buffer has not
been modified.
* START defaults to `point-min', END defaults to `point-max'."
(interactive (let* ((regionp (use-region-p))
(st (if regionp (region-beginning) (point-min)))
(en (if regionp (region-end) (point-max))))
(list st en t)))
(setq start (or start (point-min))
end (or end (point-max)))
(let ((buff (buffer-name)))
(when (or forcep (and buff
(buffer-modified-p)
(string-match-p "\\`\\*TOC Outline\\* ([^)]+)" buff)
(derived-mode-p 'Info-mode)))
(Info-toc-outline-refontify-links start end))))
(defun Info-outline-promote (&optional which)
"Promote headings higher up the tree.
If `transient-mark-mode' is on and the mark is active, promote
headings in the region (from a Lisp program, pass the symbol `region'
for WHICH).
Otherwise:
* With no prefix arg, promote the current heading and all headings in
the subtree (from a Lisp program, pass symbol `subtree' for WHICH);
* with a prefix arg, promote just the current heading (from a Lisp
program, pass nil for WHICH, or do not pass any argument).
This is a version of `outline-promote' for use with Info. It
refontifies the buffer to hide the link prefix `*Note'."
(interactive (list (if (and transient-mark-mode mark-active)
'region
(outline-back-to-heading)
(and (not current-prefix-arg) 'subtree))))
(let (start end)
(cond ((eq which 'region)
(outline-map-region 'Info-outline-promote
(setq start (copy-marker (region-beginning)))
(prog1 (region-end)
(setq end (save-excursion (goto-char (region-end)) (copy-marker (line-end-position)))))))
(which
(outline-map-region 'Info-outline-promote
(setq start (copy-marker (point)))
(save-excursion (outline-get-next-sibling) (setq end (copy-marker (point))))))
(t
(outline-back-to-heading t)
(setq start (copy-marker (point))
end (copy-marker (line-end-position)))
(let* ((head (match-string-no-properties 0))
(level (save-match-data (funcall outline-level)))
(up-head (or (outline-head-from-level (1- level) head)
(save-excursion ; Use the parent heading, if it is really one level less.
(save-match-data
(outline-up-heading 1 t)
(and (= (1- level) (funcall outline-level)) (match-string-no-properties 0))))
(error "Cannot promote - already at highest level"))))
(unless (rassoc level outline-heading-alist) (push (cons head level) outline-heading-alist))
(replace-match up-head nil t))))
(Info-toc-outline-refontify-links start end)))
(defun Info-outline-demote (&optional which)
"Demote headings lower down the tree.
If `transient-mark-mode' is on and the mark isactive, demote headings
in the region (from a Lisp program, pass symbol `region' for WHICH).
Otherwise:
* Without a prefix arg, demote current heading and all headings in the
subtree (from a Lisp program, pass symbol `subtree' for WHICH).
* With a prefix arg, demote just the current heading (from a Lisp
program, pass nil for WHICH, or do not pass any argument).
This is a version of `outline-demote' for use with Info. It
refontifies the buffer to hide link prefix `*Note'."
(interactive (list (if (and transient-mark-mode mark-active)
'region
(outline-back-to-heading)
(and (not current-prefix-arg) 'subtree))))
(let (start end)
(cond ((eq which 'region)
(outline-map-region 'Info-outline-demote
(setq start (copy-marker (region-beginning)))
(prog1 (region-end)
(setq end (save-excursion (goto-char (region-end)) (copy-marker (line-end-position)))))))
(which
(outline-map-region 'Info-outline-demote
(setq start (copy-marker (point)))
(save-excursion (outline-get-next-sibling) (setq end (copy-marker (point))))))
(t
(setq start (copy-marker (point))
end (copy-marker (line-end-position)))
(let* ((head (match-string-no-properties 0))
(level (save-match-data (funcall outline-level)))
(down-head (or (outline-head-from-level (1+ level) head)
(save-excursion
(save-match-data
(while (and (progn (outline-next-heading) (not (eobp)))
(<= (funcall outline-level) level)))
(when (eobp) ; Try again from beginning of buffer.
(goto-char (point-min))
(while (and (progn (outline-next-heading) (not (eobp)))
(<= (funcall outline-level) level))))
(unless (eobp)
(looking-at outline-regexp)
(match-string-no-properties 0))))
(error "Cannot demote - already at lowest level"))))
(unless (rassoc level outline-heading-alist) (push (cons head level) outline-heading-alist))
(replace-match down-head nil t))))
(Info-toc-outline-refontify-links start end)))
(defun Info-toc-outline-refontify-links (begin end)
"Refontify TOC cross references between buffer positions BEGIN and END."
;; (interactive "r") ; Not really intended as a command, but might be handy sometimes (?).
(save-excursion
(let* ((inhibit-read-only t)
(case-fold-search t)
(fontify-bookmarked-p (and (boundp 'Info-fontify-bookmarked-xrefs-flag) Info-fontify-bookmarked-xrefs-flag))
(node-not-too-large (and (or fontify-bookmarked-p Info-fontify-visited-nodes)
Info-fontify-maximum-menu-size
(or (eq t Info-fontify-maximum-menu-size)
(< (- (point-max) (point-min)) Info-fontify-maximum-menu-size))))
(fontify-bookmarked-p (and node-not-too-large fontify-bookmarked-p))
(fontify-visited-p (and node-not-too-large Info-fontify-visited-nodes))
paragraph-markers rbeg rend)
(goto-char begin)
(while (re-search-forward "\\(\\*Note[ \n\t]+\\)\\([^:]*\\)\\(:[ \t]*\\([^.,:(]*\\)\\(\\(([^)]\
*)\\)[^.,:]*\\)?[,:]?\n?\\)" end t)
(let ((start (match-beginning 0))
(next (point))
other-tag)
(when Info-hide-note-references
(when (not (eq Info-hide-note-references 'hide)) ; *Note is often used where *note should have been.
(goto-char start)
(skip-syntax-backward " ")
(when (memq (char-before) '(?\( ?\[ ?\{))
(skip-syntax-backward " (")) ; Check whether the paren is preceded by an end of sentence.
(setq other-tag (cond ((save-match-data (looking-back "\\<see")) "")
((save-match-data (looking-back "\\<in")) "")
((memq (char-before) '(nil ?\. ?! ??)) "See ")
((save-match-data (save-excursion (search-forward "\n\n" start t))) "See ")
(t "see "))))
(goto-char next)
(add-text-properties
(match-beginning 1)
(or (save-match-data (let ((start1 (match-beginning 1))) ; Don't hide \n after *Note
(and (string-match "\n" (match-string 1)) (+ start1 (match-beginning 0)))))
(match-end 1))
(if other-tag
`(display ,other-tag front-sticky nil rear-nonsticky t)
'(invisible t front-sticky nil rear-nonsticky t))))
(add-text-properties
(match-beginning 2) (match-end 2)
(list 'help-echo (if (or (match-end 5) (not (equal (match-string 4) "")))
(concat "mouse-2: go to " (or (match-string 5) (match-string 4)))
"mouse-2: go to this node")
'mouse-face 'highlight))
(setq rbeg (match-beginning 2)
rend (match-end 2))
(let (node)
(put-text-property
rbeg
rend
'font-lock-face
(if (and (or Info-fontify-visited-nodes fontify-bookmarked-p)
(save-match-data
(setq node (replace-regexp-in-string
"^[ \t]+" ""
(replace-regexp-in-string
"[ \t\n]+" " "
(or (match-string-no-properties 5)
(and (not (equal (match-string 4) "")) (match-string-no-properties 4))
(match-string-no-properties 2)))))
(let* ((hl Info-history-list)
(external-link-p (string-match "(\\([^)]+\\))\\([^)]*\\)" node))
(file (if external-link-p
(file-name-nondirectory (match-string-no-properties 1 node))
Info-current-file))
res)
(when external-link-p
(setq node (if (equal (match-string 2 node) "") "Top" (match-string-no-properties 2 node))))
(or (and fontify-bookmarked-p (Info-bookmark-name-for-node node))
(progn
(while hl
(if (and (string-equal node (nth 1 (car hl)))
(equal file (if (and external-link-p (stringp (caar hl)))
(file-name-nondirectory (caar hl))
(caar hl))))
(setq res (car hl)
hl nil)
(setq hl (cdr hl))))
res)))))
(let ((bmk (and fontify-bookmarked-p (Info-bookmark-for-node node 'LOCALP))))
(if bmk
(or (bmkp-get-tag-value bmk "bmkp-info-face") 'info-xref-bookmarked)
'info-xref-visited))
'info-xref)))
(save-excursion ; For multiline ref, unfontify newline and surrounding whitespace
(goto-char rbeg)
(save-match-data (while (re-search-forward "\\s-*\n\\s-*" rend t nil)
(remove-text-properties (match-beginning 0) (match-end 0) '(font-lock-face t)))))
(when (memq Info-hide-note-references '(t hide))
(add-text-properties (match-beginning 3) (match-end 3)
'(invisible t front-sticky nil rear-nonsticky t))
;; Unhide the file name of the external reference in parens
(when (and (match-string 6) (not (eq Info-hide-note-references 'hide)))
(remove-text-properties (match-beginning 6) (match-end 6) '(invisible t front-sticky nil rear-nonsticky t)))
(save-match-data ; Unhide newline because hidden newlines cause too long lines
(let ((beg3 (match-beginning 3))
(end3 (match-end 3)))
(when (and (string-match "\n[ \t]*" (match-string 3))
(not (save-match-data (save-excursion (goto-char (1+ end3)) (looking-at "[.)]*$")))))
(remove-text-properties (+ beg3 (match-beginning 0)) (+ beg3 (match-end 0))
'(invisible t front-sticky nil rear-nonsticky t))))))
(when (and Info-refill-paragraphs Info-hide-note-references)
(push (set-marker (make-marker) start) paragraph-markers)))))
(goto-char (point-max))
(skip-chars-backward "\n") ; Hide any empty lines at the end of the node.
(when (< (1+ (point)) (point-max)) (put-text-property (1+ (point)) (point-max) 'invisible t))
(set-buffer-modified-p nil)))
)
;; Note: This is not super-clean code (it's kind of a hack job).
;;;###autoload (autoload 'Info-merge-subnodes "info+")
(defun Info-merge-subnodes (&optional recursive-display-p recursive-call-p)
"Integrate current node with nodes referred to in its Menu.
Displays the current Info node, together with the nodes in its Menu.
Buffer `*Info: NODE*' is used for the display, where NODE is the name
of the current node. The contents of this node's subnodes (the nodes
named in this node's Menu) are included in the buffer, following the
contents of the current node.
Optional arg RECURSIVE-DISPLAY-P (prefix arg if interactive) governs
the way menus of subnodes are treated:
If nil, nothing additional happens. Subnode menus are not explored.
Only the current node and its immediate subnodes are documented, in
the single display buffer `*Info: NODE*'.
If non-nil, then the subnodes of a node are treated in the same way
as the parent node, recursively: If any of them has, itself, a Menu,
then that menu's subnodes are also explored, and so on.
If RECURSIVE-DISPLAY-P is zero, then a single display buffer is
used for all of the nodes explored. Otherwise, a separate display
buffer is used for each subnode that has a Menu (see next).
Use this when you want a single, flat compilation of the current
node and all of its subnodes. It is less appropriate when the
current node has several levels of subnodes: The flattened
result can be difficult to read.
If RECURSIVE-DISPLAY-P is positive, then the contents of each
subnode are displayed twice: once in the parent node's display,
and once in the subnode's own display.
Use this when the current node has several levels of subnodes
and you want each display buffer to be self-contained.
If RECURSIVE-DISPLAY-P is negative, then there is no redundancy: A
subnode's contents are only displayed in its parent's buffer. The
subnode's own display buffer only contains the contents of its own
subnodes.
Use this when the current node has several levels of subnodes
and you want no redundancy between the display buffers.
The user option (variable) `Info-subtree-separator' is a string to be
inserted by `Info-merge-subnodes' just before the title of each
node (preceding its description). By default it is \"\\n* \", producing
a node title resembling a menu item. Setting this to \"\\f\\n* \" will
cause a page break before each node description. For more on setting
this variable, type \\<Info-mode-map>`\\[describe-variable] Info-subtree-separator'.
------
Optional second arg RECURSIVE-CALL-P is only for internal use. It is
used to indicate whether (non-nil) or not (nil) this is a recursive
\(i.e. not a top-level) call to `Info-merge-subnodes'. Non-nil
means that this is a subnode, and that its contents should only be
included in the present display if RECURSIVE-DISPLAY-P is also
non-nil. For proper operation when RECURSIVE-DISPLAY-P is zero, the
non-nil value of RECURSIVE-CALL-P should be the node name of the
top-level call to `Info-merge-subnodes'."
(interactive "P")
(when (interactive-p)
(unless (y-or-n-p "Do you really want to integrate this node with its \
subnodes (outside Info)? ")
(info-user-error (substitute-command-keys
"OK. If you are not sure what this command is about, type \
`\\[describe-function] Info-merge-subnodes'.")))) ; Defined in `help.el'.
(garbage-collect)
(setq recursive-display-p (and recursive-display-p (prefix-numeric-value recursive-display-p)))
(let* ((buf (current-buffer)) ; Info buffer
(single-buf-p (and recursive-display-p (zerop recursive-display-p)))
(infop-node-name (or (and single-buf-p recursive-call-p) Info-current-node))
(rep-buf (get-buffer-create (concat "*Info: " infop-node-name "*"))) ; Merge buffer.
(more t)
(inhibit-field-text-motion t) ; Just to be sure, for `end-of-line'.
token oldpt strg menu-item-line ind)
(when (interactive-p)
(message "Processing node `%s' and %ssubnodes..." infop-node-name
(if recursive-display-p "all of its " "its immediate ")))
(save-window-excursion
(goto-char (point-min))
(forward-line 1)
(setq strg (buffer-substring (point) (point-max))) ; Node contents.
(goto-char (point-min))
(setq more (search-forward "* menu" nil t))
(forward-line 1)
;; Merge buffer: Insert buffer header and main node's contents, if not recursive or
;; do want redundancy.
;; Then insert each subnode (unless this is an Index).
(switch-to-buffer-other-window rep-buf)
(unless (and recursive-call-p single-buf-p)
(erase-buffer)
(funcall Info-display-node-header-fn) ; Insert header.
(insert (concat "\n\n" (and (or (not recursive-call-p) ; Top-level call.
(and recursive-display-p ; Redundancy desired.
(> recursive-display-p 0)))
strg)))) ; Insert main node's contents.
(unless (string-match "\\s-*Index$" infop-node-name) ; Don't recurse down Index menus.
;; Insert menu items and possibly their subnodes.
(save-excursion
(while more
;; Info buffer: Get menu item token.
(set-buffer buf)
(end-of-line)
(setq oldpt (point)
more (search-forward "\n* " nil t)) ; Possible next menu item.
(unless more (goto-char (point-max)))
(while (and (not (eobp)) ; Search for a real menu item.
(not (setq token (Info-get-token ; File menu item.
(point) "\\* " "\\* \\([^:]*\\)::")))
(not (setq token (Info-get-token ; Nonfile menu item.
(point) "\\* "
"\\* [^:]*:[ \t]+\\([^\t,.\n]+\\)[\t,.\n]"))))
(setq more (search-forward "\n* " nil t)))
(unless token (setq more nil)) ; No menu item. Done.
;; Treat subnode (menu item).
(when more
;; Merge buffer: Insert separator line.
(set-buffer rep-buf)
(goto-char (point-max))
(insert Info-subtree-separator) ; Ready for next menu item.
;; Info buffer: Go to subnode.
(set-buffer buf)
(Info-goto-node token)
(goto-char (point-min))
(forward-line 1)
(setq strg (buffer-substring (point) (point-max))) ; Pick up subnode contents.
;; Go back to parent node and get menu-item line.
(Info-history-back)
(let ((inhibit-read-only t)) ; Get untabified menu-item line, so can count
(buffer-enable-undo) (undo-start) ; chars to underline.
(untabify (point) (save-excursion (forward-line 1) (point)))
(setq menu-item-line (buffer-substring-no-properties
(save-excursion (beginning-of-line) (forward-char 2) (point))
(save-excursion (forward-line 1) (point))))
(when pending-undo-list (undo-more 1)) ; Only if did something.
(buffer-disable-undo))
;; Merge buffer: Insert menu-item line, underline it, and insert subnode contents.
(set-buffer rep-buf)
(insert menu-item-line)
(setq ind (1+ (length menu-item-line)))
(while (> ind 0) (insert "=") (setq ind (1- ind))) ; Underline menu item.
(insert "\n")
(put-text-property (save-excursion (forward-line -2) (point))
(save-excursion (forward-line 1) (point))
'font-lock-face 'info-file)
(setq oldpt (point))
(insert strg) ; Insert subnode contents.
(indent-rigidly oldpt (point) 2)
;; Recursive call: Insert subnode's subnodes, if there are any.
;; Again, though, don't recurse down Index menus.
(when (and recursive-display-p (not (string-match "\\s-*Index$" token)))
;; Info buffer: Go back to subnode.
;; If it has a menu, then treat its subnodes, recursively.
(with-current-buffer buf
(Info-goto-node token)
(when (search-forward "* menu" nil t)
(forward-line 1) (end-of-line)
(when (and (search-forward "\n* " nil t)
(or (Info-get-token (point) "\\* " "\\* \\([^:]*\\)::") ; file menu item
(Info-get-token (point) "\\* " ; nonfile menu item
"\\* [^:]*:[ \t]+\\([^\t,.\n]+\\)[\t,.\n]")))
(Info-merge-subnodes recursive-display-p infop-node-name)))
(set-buffer buf) ; Info buffer: Go back to parent node.
(Info-history-back)))
(set-buffer buf)))))) ; Info buffer
;; Merge buffer
(switch-to-buffer-other-window rep-buf)
(when (and (one-window-p t) (not (window-minibuffer-p)) (fboundp 'fit-frame) ; Defined in `fit-frame.el'.
Info-fit-frame-flag)
(fit-frame))
(goto-char (point-min))
(set-buffer-modified-p nil)
(use-local-map Info-merged-map)
(when (interactive-p)
(message "Processing node `%s' and %ssubnodes... done" infop-node-name
(if recursive-display-p "all of its " "its immediate ")))))
;;;###autoload (autoload 'Info-virtual-book "info+")
(defun Info-virtual-book (book nodes &optional include-bookmark-nodes-p)
"Open a virtual Info BOOK, with a menu of Info NODES.
BOOK is a string naming the virtual book.
NODES is a list of strings naming Info nodes.
Each node name is normally absolute, that is, a
filename-plus-nodename string such as \"(emacs)Basic\". But if you
call this command from an Info buffer, then a node name can be
relative, such as \"Basic\".
Non-nil optional arg INCLUDE-BOOKMARK-NODES-P means that all Info
nodes recorded as bookmarks are included in the virtual book.
Interactively, you are prompted for the name of the virtual book, and
the nodes are those in `Info-saved-nodes'. Interactively, a prefix
argument says to include Info nodes recorded as bookmarks."
(interactive (list (read-from-minibuffer "Virtual book name: " nil nil nil nil "Virtual Book")
Info-saved-nodes
current-prefix-arg))
(unless nodes (setq nodes Info-saved-nodes))
(when include-bookmark-nodes-p
(unless (require 'bookmark+ nil t) (require 'bookmark nil t))
(bookmark-maybe-load-default-file)
(let ((bm-nodes ())
node file)
(dolist (bm bookmark-alist)
(when (setq node (cdr (assq 'info-node bm)))
(setq file (bookmark-get-filename bm))
(push (concat "(" (file-name-nondirectory (cdr (assq 'filename bm))) ")" node) bm-nodes)))
(setq nodes (append nodes bm-nodes))))
(unless (and nodes (stringp (car nodes))) ; Minimal sanity check.
(info-user-error (if (interactive-p) "No saved Info nodes" "No Info nodes")))
(unless (stringp book) (setq book "Virtual Book")) ; Non-interactive - NODESET is a list.
(let ((file (and (stringp Info-current-file) (concat "(" (file-name-nondirectory Info-current-file) ")"))))
(with-current-buffer (get-buffer-create " *info-toc*")
(let ((inhibit-read-only t)
sans-file filep)
(erase-buffer)
(goto-char (point-min))
(insert "\n\^_\nFile: toc, Node: Top, Up: (dir)\n\n")
(insert book "\n" (make-string (length book) ?*) "\n\n")
(insert "* Menu:\n\n")
(while nodes
(if (setq filep (string-match "^([^)]+)" (car nodes)))
(setq sans-file (substring (car nodes) (match-end 0)))
(setq sans-file (car nodes)))
;; (insert "* " sans-file ": " (car nodes) ".\n")
(insert "* " sans-file ": "
(concat (and (not filep) (or file "(**INFO FILE UNKNOWN**)")) (car nodes))
".\n")
(setq nodes (cdr nodes))))
(unless (bobp)
(let ((Info-hide-note-references 'hide)
(Info-fontify-visited-nodes nil))
(Info-mode)
(setq Info-current-file 'toc Info-current-node "Top")
(goto-char (point-min))
(narrow-to-region (or (re-search-forward "\n[\^_\f]\n" nil t) (point-min)) (point-max))
(Info-fontify-node)
(widen)))))
(info)
(Info-find-node 'toc "Top"))
;;;###autoload (autoload 'Info-goto-node-web "info+")
(defun Info-goto-node-web (node &optional flip-new-win)
"Use `browse-url' to go to Info node NODE using a Web browser.
With a prefix arg, reverse the effect of option
option`browse-url-new-window-flag'.
NODE is the name of a node in the GNU Emacs or Elisp manual.
Alternatively, NODE can have the form (MANUAL)NODE, where MANUAL is
\"emacs\" or \"elisp\" and NODE is the name of the node in that
manual. Empty NODE in (MANUAL) defaults to the `Top' node."
(interactive (list (Info-read-node-name "Go to node: " Info-current-node) current-prefix-arg))
(require 'browse-url)
(unless Info-current-file (error "This command must be invoked from Info"))
(browse-url (Info-url-for-node node) (list (if flip-new-win
(not browse-url-new-window-flag)
browse-url-new-window-flag))))
;; See https://www.gnu.org/software/texinfo/manual/texinfo/html_node/
;; HTML-Xref-Node-Name-Expansion.html
;;
;; 1. The standard ASCII letters (a-z and A-Z) are not modified. All
;; other characters may be changed as specified below.
;;
;; 2. The standard ASCII numbers (0-9) are not modified except when a
;; number is the first character of the node name. In that case, see
;; below.
;;
;; 3. Multiple consecutive space, tab and newline characters are
;; transformed into just one space. (Its not possible to have
;; newlines in node names with the current implementation, but we
;; specify it anyway, just in case.)
;;
;; 4. Leading and trailing spaces are removed.
;;
;; 5. After the above has been applied, each remaining space character is
;; converted into a - character.
;;
;; 6. Other ASCII 7-bit characters are transformed into _00xx, where xx
;; is the ASCII character code in (lowercase) hexadecimal. This includes
;; _, which is mapped to _005f.
;;
;; 7. If the node name does not begin with a letter, the literal string
;; g_t is prefixed to the result. (Due to the rules above, that
;; string can never occur otherwise; it is an arbitrary choice,
;; standing for “GNU Texinfo”.) This is necessary because XHTML
;; requires that identifiers begin with a letter.
;;
;;;###autoload (autoload 'Info-url-for-node "info+")
(defun Info-url-for-node (node)
"Return a URL for NODE, a node in the GNU Emacs or Elisp manual.
Alternatively, NODE can have the form (MANUAL)NODE, where MANUAL is
\"emacs\" or \"elisp\" and NODE is the name of the node in that
manual. Empty NODE in (MANUAL) defaults to the `Top' node."
(interactive (list (Info-read-node-name "Node: " Info-current-node)))
(unless Info-current-file (error "This command must be invoked from Info"))
(let (file url)
(string-match "\\s *\\((\\s *\\([^\t)]*\\)\\s *)\\s *\\|\\)\\(.*\\)" node)
(setq file (if (= (match-beginning 1) (match-end 1)) "" (match-string 2 node))
node (match-string 3 node))
(when (equal node "") (setq node "index")) ; `Top' node.
(let ((trim (string-match "\\s +\\'" file)))
(when trim (setq file (substring file 0 trim))))
(let ((trim (string-match "\\s +\\'" node)))
(when trim (setq node (substring node 0 trim))))
(when (equal file "") (setq file Info-current-file))
(setq file (file-name-sans-extension (file-name-nondirectory file)))
(unless (member file '("emacs" "elisp"))
(error "Manual cannot be `%s'; it can only be `emacs' or `elisp'" file))
(setq node (mapconcat (lambda (ch)
(if (or (< ch 32) ; ^@^A-^Z^[^\^]^^^-
(and (<= 33 ch) (<= ch 47)) ; !"#$%&'()*+,-./
(and (<= 58 ch) (<= ch 64)) ; :;<=>?@
(and (<= 91 ch) (<= ch 96)) ; [\]_`
(and (<= 123 ch) (<= ch 127))) ; {|}~ DEL
(format "_00%x" ch)
(char-to-string ch)))
node
""))
(setq node (replace-regexp-in-string "[ \t]+" "-" node t t))
(unless (string-match-p "[[:alpha:]]" node) (setq node (concat "g_t" node)))
(setq url (concat "https://www.gnu.org/software/emacs/manual/html_node/"
file "/" node ".html"))
(message "URL: %s" url)
url))
;;;###autoload (autoload 'info-manual "info+")
(defun info-manual (arg) ; Bound to `C-h r' globally (replaces `info-emacs-manual').
"Display a manual in Info mode - by default, the Emacs manual.
With a prefix arg, prompt for the manual name.
With a numeric prefix arg, only currently visited manuals are
candidates."
(interactive "P")
(if arg
(let ((current-prefix-arg (numberp arg))) (call-interactively #'info-display-manual))
(info "emacs")))
(global-set-key [remap info-emacs-manual] 'info-manual) ; `C-h r'
(easy-menu-define
Info-merged-menu Info-merged-map
"Menu for merged `info' buffers."
'("Info"
["Next Link" Info-next-reference t]
["Previous Link" Info-prev-reference t]
["Search (regexp)" Info-search t]
["Quit" quit-window t]))
(easy-menu-define
Info-mode-menu Info-mode-map
"Menu for Info files."
'("Info"
("Toggle"
["Highlighting _..._ (emphasis)" Info-toggle-fontify-emphasis
:visible info-fontify-emphasis :style toggle :selected Info-fontify-emphasis-flag
:help "Toggle option `Info-fontify-emphasis-flag'"]
["Highlighting ... or `...', and \"...\"" Info-toggle-fontify-quotations
:style toggle :selected Info-fontify-quotations-flag :help "Toggle option `Info-fontify-quotations-flag'"]
["Highlighting <...>" Info-toggle-fontify-angle-bracketed
:style toggle :selected Info-fontify-angle-bracketed-flag
:help "Toggle option `Info-fontify-angle-bracketed-flag'"]
["Highlighting Single '" Info-toggle-fontify-single-quote
:style toggle :selected Info-fontify-single-quote-flag
:help "Toggle option `Info-fontify-single-quote-flag'"]
["Highlighting Bookmarked Links" Info-toggle-fontify-bookmarked-xrefs
:style toggle :selected (and (boundp 'Info-fontify-bookmarked-xrefs-flag) Info-fontify-bookmarked-xrefs-flag)
:visible (fboundp 'Info-toggle-fontify-bookmarked-xrefs)
:help "Toggle option `Info-fontify-bookmarked-xrefs-flag'"]
["Bookmark Access On Visit" Info-toggle-node-access-invokes-bookmark
:style toggle :selected (and (boundp 'Info-node-access-invokes-bookmark-flag) Info-node-access-invokes-bookmark-flag)
:visible (fboundp 'Info-toggle-node-access-invokes-bookmark)
:help "Toggle option `Info-node-access-invokes-bookmark-flag'"]
["Breadcrumbs in Mode Line" Info-breadcrumbs-in-mode-line-mode
:style toggle :selected Info-breadcrumbs-in-mode-line-mode
:help "Toggle showing breadcrumbs in the mode line"]
["Breadcrumbs in Node Header" Info-toggle-breadcrumbs-in-header
:style toggle :selected Info-breadcrumbs-in-header-flag
:help "Toggle showing breadcrumbs in the node header"])
["Table of Contents (TOC)" Info-toc :help "Go to table of contents"]
["Editable Outline TOC" Info-toc-outline :help "Go to editable table of contents with outline support"]
["Virtual Book" Info-virtual-book :help "Open table of contents of a virtual book" :active Info-saved-nodes]
["Save Current Node" Info-save-current-node :help "Save current node name for virtual book"]
["Find...(Regexp)" Info-search :help "Search for regular expression in this Info file"]
["Find Case-Sensitively..." Info-search-case-sensitively
:help "Search case sensitively for regular expression"]
["Find Again" Info-search-next :help "Search for another occurrence of same regular expression"]
("Index"
["Find with Index..." Info-index :help "Look for a string in the index"]
["Find Again with Index" Info-index-next
:active Info-index-alternatives :help "Look for string again in index"]
["Find In All Indexes..." info-apropos :help "Look for a string in the indexes of all manuals"])
"--"
["Back (History)" Info-history-back
:active Info-history :help "Go back in history to the last node you were at"]
["Forward (History)" Info-history-forward :active Info-history-forward :help "Go forward in history"]
["History List" Info-history :active Info-history-list :help "Go to menu of visited nodes"]
"--"
["Top" Info-directory :help "Go to the list of manuals (Info top level)"]
["Up" Info-up :active (Info-check-pointer "up") :help "Go up in the Info tree"]
["Next" Info-next :active (Info-check-pointer "next") :help "Go to the next node"]
["Previous" Info-prev :active (Info-check-pointer "prev[ious]*") :help "Go to the previous node"]
("Menu Item" ["You should never see this" report-emacs-bug t])
("Reference" ["You should never see this" report-emacs-bug t])
["Go to Node on Web..." Info-goto-node-web :help "Go to a named node on the Web (HTML doc)"]
["Go to Node..." Info-goto-node :help "Go to a named node"]
"--"
["Forward" Info-forward-node :help "Go forward one node, considering all as a sequence"]
["Backward" Info-backward-node :help "Go backward one node, considering all as a sequence"]
["First in File" Info-top-node :help "Go to top node of file"]
["Last in File" Info-final-node :help "Go to final node in this file"]
["Beginning of This Node" beginning-of-buffer :help "Go to beginning of this node"]
"--"
["Clone Info Buffer" clone-buffer :help "Create a twin copy of the current Info buffer."]
["Copy Node Name" Info-copy-current-node-name :help "Copy the name of the current node into the kill ring"]
["Merge Subnodes" Info-merge-subnodes :help "Integrate current node with nodes referred to in its Menu"]
"--"
["Quit Info" Info-exit :help "Exit from Info"]))
;;(@* "Replacements for Existing Functions")
;;; Replacements for Existing Functions -------------------------------
;; REPLACE ORIGINAL in `info.el':
;;
;; Added prefix arg.
;;
(defadvice Info-history (around clear-info-hist-with-prefix-arg first (&optional clearp) activate)
"With a prefix arg, clear the history instead, upon confirmation."
(interactive "P")
(if (not (ad-get-arg 0))
ad-do-it
(when (equal "*History*" Info-current-file) (Info-up))
(call-interactively #'Info-history-clear)))
;; REPLACE ORIGINAL in `info.el':
;;
;; Respect option `Info-node-access-invokes-bookmark-flag'.
;;
(when (boundp 'Info-node-access-invokes-bookmark-flag) ; Emacs 24.4+
(defadvice Info-goto-node (around bmkp-invoke-Info-bookmark activate)
"Respect option `Info-node-access-invokes-bookmark-flag'.
If the option is non-nil then a bookmark for the node is invoked when
the node is visited, provided that the bookmark name has the default
form: `(MANUAL) NODE' (e.g.,`(emacs) Modes')."
(if Info-node-access-invokes-bookmark-flag
(let ((node (ad-get-arg 0)))
(if (member node (Info-index-nodes))
ad-do-it
(let ((bmk (and Info-node-access-invokes-bookmark-flag (Info-bookmark-for-node node))))
(if bmk
(let ((Info-node-access-invokes-bookmark-flag nil)) (bookmark--jump-via bmk 'ignore))
ad-do-it))))
ad-do-it))
)
;; REPLACE ORIGINAL in `info.el':
;;
;; Added optional arg DEFAULT.
;;
(defun Info-read-node-name (prompt &optional default)
(let* ((completion-ignore-case t)
(Info-read-node-completion-table (Info-build-node-completions))
(nodename (completing-read
prompt 'Info-read-node-name-1 nil t nil 'Info-minibuf-history default)))
(if (equal nodename "")
(or default (Info-read-node-name prompt))
nodename)))
;; REPLACE ORIGINAL in `info.el':
;;
;; Added optional arg LITERALP. Use apropos matching, not literal-string matching, by default. Prefix arg matches literally.
;; Use other window, unless already in Info.
;;
(defun info-apropos (pattern &optional literalp)
"Search indexes of all known Info files on your system for apropos PATTERN.
Build a menu of the possible matches.
With a prefix arg, match PATTERN as a literal string, not as a regexp
or keywords.
Just as for commands such as `apropos', PATTERN can be a word, a list
of words (separated by spaces), or a regexp (using some regexp special
characters). If it is a word, search for matches for that word as a
substring. If it is a list of words, search for matches for any
two (or more) of those words."
(interactive (list (apropos-read-pattern "index entries") current-prefix-arg))
(apropos-parse-pattern pattern)
(if (equal apropos-regexp "")
(Info-find-node Info-apropos-file "Top")
(let ((nodes Info-apropos-nodes)
nodename)
(while (and nodes (not (string-match apropos-regexp (nth 1 (car nodes)))))
(setq nodes (cdr nodes)))
;; Use another window, if not already in Info.
(unless (eq major-mode 'Info-mode) (pop-to-buffer "*info*"))
(if nodes
(Info-find-node Info-apropos-file (caar nodes))
(setq nodename (format "Index for %s" apropos-regexp))
(push (list nodename
apropos-regexp
(Info-apropos-matches apropos-regexp (and (not literalp) 'REGEXP)))
Info-apropos-nodes)
(Info-find-node Info-apropos-file nodename)))))
;; REPLACE ORIGINAL in `info.el':
;;
;; Added optional arg REGEXP-P.
;;
(defun Info-apropos-matches (string &optional regexp-p)
"Collect STRING matches from all known Info files on your system.
Return a list of matches where each element is in the format
\((FILENAME INDEXTEXT NODENAME LINENUMBER)).
Non-nil optional REGEXP-P means interpret STRING as a regexp, instead
of trying to match it literally."
;; Emacs 23 has an `interactive' spec here, for no reason.
(unless (string= string "")
(let ((pattern (format "\n\\* +\\([^\n]*\\(%s\\)[^\n]*\\):[ \t]+\
\\([^\n]+\\)\\.\\(?:[ \t\n]*(line +\\([0-9]+\\))\\)?"
(if regexp-p string (regexp-quote string))))
(ohist Info-history)
(ohist-list Info-history-list)
(current-node Info-current-node)
(current-file Info-current-file)
manuals matches node nodes)
(let ((Info-fontify-maximum-menu-size nil))
(Info-directory)
;; `current-node' and `current-file' are nil if you invoke `info-apropos' as the first Info command.
;; (`info-apropos' loads `info.el'.) In that case, use `(DIR)Top', to avoid an error after search is complete.
(unless current-node (setq current-file Info-current-file
current-node Info-current-node))
(message "Searching indices...")
(goto-char (point-min))
(re-search-forward "\\* Menu: *\n" nil t)
(while (re-search-forward "\\*.*: *(\\([^)]+\\))" nil t)
(add-to-list 'manuals (match-string 1))) ; Ensure no duplicates in MANUALS, so the `dolist' runs faster.
(dolist (manual (nreverse manuals))
(message "Searching %s" manual)
(condition-case err
(if (setq nodes (Info-index-nodes (Info-find-file manual)))
(save-excursion
(Info-find-node manual (car nodes))
(while
(progn
(goto-char (point-min))
(while (re-search-forward pattern nil t)
(let ((entry (match-string-no-properties 1))
(nodename (match-string-no-properties 3))
(line (match-string-no-properties 4)))
(add-text-properties
(- (match-beginning 2) (match-beginning 1))
(- (match-end 2) (match-beginning 1))
'(face info-index-match) entry)
(setq matches (cons (list manual entry nodename line)
matches))))
(setq nodes (cdr nodes)
node (car nodes)))
(Info-goto-node node))))
(error (message "%s" (if (eq (car-safe err) 'error) (nth 1 err) err))
(sit-for 1 t)))))
(Info-find-node current-file current-node)
(setq Info-history ohist
Info-history-list ohist-list)
(message "Searching indices...done")
(or (nreverse matches) t))))
;; REPLACE ORIGINAL in `info.el':
;;
;; Added final clause to `cond', to handle virtual books. (Emacs 23.2+)
;;
(when (or (> emacs-major-version 23) (and (= emacs-major-version 23) (> emacs-minor-version 1)))
(defun Info-find-file (filename &optional noerror)
"Return expanded FILENAME, or t if FILENAME is \"dir\".
Optional second argument NOERROR, if t, means if file is not found
just return nil (no error)."
;; Convert filename to lower case if not found as specified.
;; Expand it.
(cond
((Info-virtual-call (Info-virtual-fun 'find-file filename nil) filename noerror))
((stringp filename)
(let (temp temp-downcase found)
(setq filename (substitute-in-file-name filename))
(let ((dirs (if (string-match "^\\./" filename)
'("./") ; If specified name starts with `./' then just try current dir.
(if (file-name-absolute-p filename)
'(nil) ; No point in searching for an absolute file name
(if Info-additional-directory-list
(append Info-directory-list Info-additional-directory-list)
Info-directory-list)))))
;; Fall back on the installation directory if we can't find the info node anywhere else.
(when installation-directory
(setq dirs (append dirs (list (expand-file-name "info" installation-directory)))))
;; Search the directory list for file FILENAME.
(while (and dirs (not found))
(setq temp (expand-file-name filename (car dirs))
temp-downcase (expand-file-name (downcase filename) (car dirs)))
;; Try several variants of specified name.
(let ((suffix-list Info-suffix-list)
(lfn (if (fboundp 'msdos-long-file-names) (msdos-long-file-names) t)))
(while (and suffix-list (not found))
(cond ((info-file-exists-p (info-insert-file-contents-1 temp (car (car suffix-list)) lfn))
(setq found temp))
((info-file-exists-p (info-insert-file-contents-1 temp-downcase (car (car suffix-list))
lfn))
(setq found temp-downcase))
((and (fboundp 'msdos-long-file-names)
lfn
(info-file-exists-p (info-insert-file-contents-1 temp (car (car suffix-list))
nil)))
(setq found temp)))
(setq suffix-list (cdr suffix-list))))
(setq dirs (cdr dirs))))
(if found
(setq filename found)
(if noerror
(setq filename nil)
(unless Info-current-file (Info-directory)) ; If no previous Info file, go to directory.
(info-user-error "Info file `%s' does not exist" filename)))
filename))
((member filename '(apropos history toc)) filename))) ; Handle virtual books - `toc'.
)
;; REPLACE ORIGINAL in `info.el':
;;
;; Added optional arg NOMSG.
;;
(defun Info-find-node (filename nodename &optional no-going-back strict-case nomsg)
"Go to an Info node specified as separate FILENAME and NODENAME.
NO-GOING-BACK is non-nil if recovering from an error in this function;
it says do not attempt further (recursive) error recovery.
Look first for a case-sensitive match for NODENAME. If none is found
then try a case-insensitive match, unless STRICT-CASE is non-nil.
Non-nil NOMSG means do not show a status message."
(info-initialize)
(setq filename (Info-find-file filename))
;; Go into Info buffer.
(or (eq major-mode 'Info-mode) (Info--pop-to-buffer-same-window "*info*"))
;; Record the node we are leaving, if we were in one.
(and (not no-going-back)
Info-current-file
(push (list Info-current-file Info-current-node (point)) Info-history))
(Info-find-node-2 filename nodename no-going-back strict-case nomsg))
;; REPLACE ORIGINAL in `info.el':
;;
;; 1. Call `fit-frame' if `Info-fit-frame-flag'.
;; 2. Added optional arg NOMSG.
;;
(defun Info-find-node-2 (filename nodename &optional no-going-back strict-case nomsg)
(buffer-disable-undo (current-buffer))
(or (eq major-mode 'Info-mode) (Info-mode))
(widen)
(setq Info-current-node nil)
(unwind-protect
(let ((case-fold-search t)
(virtual-fun (and (fboundp 'Info-virtual-fun) ; Emacs 23.2.
(Info-virtual-fun 'find-node (or filename Info-current-file) nodename)))
anchorpos)
(cond ((functionp virtual-fun)
(let ((filename (or filename Info-current-file)))
(setq buffer-read-only nil
Info-current-file filename
Info-current-subfile nil
Info-current-file-completions ()
buffer-file-name nil)
(erase-buffer)
(Info-virtual-call virtual-fun filename nodename no-going-back)
(set-marker Info-tag-table-marker nil)
(setq buffer-read-only t)
(set-buffer-modified-p nil)
(if (> emacs-major-version 24)
(setq Info-current-node-virtual t)
(set (make-local-variable 'Info-current-node-virtual) t))))
((not (and (or (not (boundp 'Info-current-node-virtual)) (not Info-current-node-virtual))
(or (null filename) (equal Info-current-file filename))))
;; Switch files if necessary
(let ((inhibit-read-only t))
(when (and (boundp 'Info-current-node-virtual) Info-current-node-virtual)
;; When moving from a virtual node.
(if (> emacs-major-version 24)
(setq Info-current-node-virtual nil)
(set (make-local-variable 'Info-current-node-virtual) nil))
(unless filename (setq filename Info-current-file)))
(setq Info-current-file nil
Info-current-subfile nil
Info-current-file-completions ()
buffer-file-name nil)
(erase-buffer)
;; Erase any memory of the previous coding-system, so that `info-insert-file-contents' sets the
;; buffer's encoding to what the Info file specifies.
(set-buffer-file-coding-system 'undecided t)
(cond ((eq filename t) (Info-insert-dir nomsg))
((eq filename 'apropos) (insert-buffer-substring " *info-apropos*"))
((eq filename 'history) (insert-buffer-substring " *info-history*"))
((eq filename 'toc) (insert-buffer-substring " *info-toc*"))
(t (info-insert-file-contents filename nil)
(setq default-directory (file-name-directory filename))))
(set-buffer-modified-p nil)
(if (> emacs-major-version 24)
(setq Info-file-supports-index-cookies (Info-file-supports-index-cookies filename))
(set (make-local-variable 'Info-file-supports-index-cookies)
(Info-file-supports-index-cookies filename)))
;; See whether file has a tag table. Record the location if yes.
(goto-char (point-max))
(forward-line -8)
;; Use string-equal, not equal, to ignore text props.
(if (not (or (string-equal nodename "*")
(not (search-forward "\^_\nEnd tag table\n" nil t))))
(let (pos)
;; We have a tag table. Find its beginning. Is this an indirect file?
(search-backward "\nTag table:\n")
(setq pos (point))
(if (save-excursion (forward-line 2) (looking-at "(Indirect)\n"))
;; It is indirect. Copy it to another buffer and record that the tag table is in that buffer.
(let ((buf (current-buffer))
(tagbuf (or Info-tag-table-buffer
(generate-new-buffer " *info tag table*"))))
(setq Info-tag-table-buffer tagbuf)
(with-current-buffer tagbuf
(buffer-disable-undo (current-buffer))
(setq case-fold-search t)
(erase-buffer)
(insert-buffer-substring buf))
(set-marker Info-tag-table-marker (match-end 0) tagbuf))
(set-marker Info-tag-table-marker pos)))
(set-marker Info-tag-table-marker nil))
(setq Info-current-file filename))))
(if (string-equal nodename "*") ; Use `string-equal', not `equal', to ignore text props.
(progn (setq Info-current-node nodename) (Info-set-mode-line))
;; Possibilities:
;;
;; 1. Anchor found in tag table
;; 2. Anchor *not* in tag table
;;
;; 3. Node found in tag table
;; 4. Node *not* found in tag table, but found in file
;; 5. Node *not* in tag table, and *not* in file
;;
;; *Or* the same, but in an indirect subfile.
;;
;;
;; Search file for a suitable node.
(let ((guesspos (point-min))
(regexp (concat "\\(Node:\\|Ref:\\) *\\(" (if (stringp nodename) (regexp-quote nodename) "")
"\\) *[,\t\n\177]")))
(catch 'foo
;; First, search a tag table, if any
(when (marker-position Info-tag-table-marker)
(let* ((m Info-tag-table-marker)
(found (if (or (> emacs-major-version 24) ; Emacs 24.4+
(and (= emacs-major-version 24) (> emacs-minor-version 3)))
(Info-find-in-tag-table m regexp strict-case)
(Info-find-in-tag-table m regexp))))
(when found ; FOUND is (ANCHOR POS MODE)
(let ((filepos (nth 1 found))) ; File position in bytes.
;; If this is an indirect file, determine which file really holds this node and read it in.
(unless (eq (nth 2 found) 'Info-mode)
;; Note that the current buffer must be the *info* buffer on entry to `Info-read-subfile'.
;; Thus the hackery above.
(setq filepos (Info-read-subfile filepos)))
(setq guesspos (if (fboundp 'filepos-to-bufferpos)
(filepos-to-bufferpos filepos 'approximate)
filepos)))
(when (nth 0 found) ; Handle anchor
(goto-char (setq anchorpos guesspos)) (throw 'foo t)))))
;; Else we may have a node, which we search for.
(goto-char (max (point-min) (- (if (> emacs-major-version 24) guesspos (byte-to-position guesspos))
1000)))
;; Now search from our advised position (or from beg of buffer) to find the actual node.
;; First, check whether the node is right where we are, in case the buffer begins with a node.
(when (> emacs-major-version 24)
(let ((pos (Info-find-node-in-buffer regexp strict-case))) (when pos (goto-char pos) (throw 'foo t)))
;; If the Texinfo source had an @ifnottex block of text before the Top node, makeinfo 5.0 and 5.1 mistakenly
;; omitted that block's size from the starting position of the 1st subfile, which makes GUESSPOS overshoot
;; the correct position by the length of that text. So we try again with a larger slop.
(goto-char (max (point-min) (- guesspos 10000))))
(let ((pos (if (<= emacs-major-version 23)
(Info-find-node-in-buffer regexp)
(Info-find-node-in-buffer regexp strict-case))))
(when pos (goto-char pos) (throw 'foo t)))
(when (string-match "\\([^.]+\\)\\." nodename)
(let (Info-point-loc) (Info-find-node-2 filename (match-string 1 nodename) no-going-back strict-case nomsg))
(widen)
(throw 'foo t))
;; No such anchor in tag table or node in tag table or file
(info-user-error "No such node or anchor: `%s'" nodename))
(Info-select-node)
(goto-char (point-min))
(forward-line 1) ; skip header line
(when (and (not (fboundp 'Info-breadcrumbs)) ; Before Emacs 23.2
Info-breadcrumbs-in-header-flag
(> Info-breadcrumbs-depth 0))
(forward-line 1)) ; skip breadcrumbs line
(cond (anchorpos
(let ((new-history (list Info-current-file (substring-no-properties nodename))))
;; Add anchors to the history too
(setq Info-history-list (cons new-history (delete new-history Info-history-list))))
(goto-char anchorpos))
((numberp Info-point-loc)
(forward-line (- Info-point-loc 2))
(setq Info-point-loc nil))
((stringp Info-point-loc)
(Info-find-index-name Info-point-loc)
(setq Info-point-loc nil)))))
(when (and (one-window-p t)
(not (window-minibuffer-p))
(fboundp 'fit-frame) ; Defined in `fit-frame.el'.
Info-fit-frame-flag)
(fit-frame)))
;; If we did not finish finding the specified node, go to the previous one or to `Top' node.
(unless (or Info-current-node no-going-back)
(if Info-history
(let ((hist (car Info-history)))
(setq Info-history (cdr Info-history))
(Info-find-node (nth 0 hist) (nth 1 hist) t t nomsg) ; Pass STRICT-CASE? Emacs bug #31137.
(goto-char (nth 2 hist)))
(Info-find-node Info-current-file "Top" t t nomsg)))) ; Pass STRICT-CASE? Emacs bug #31137.
(Info-set-mode-line))
;; REPLACE ORIGINAL in `info.el':
;;
;; Pass non-nil NOMSG arg to `Info-insert-dir'.
;;
(defun Info-directory-find-node (filename nodename &optional no-going-back nomsg)
"Directory-specific implementation of Info-find-node-2."
(Info-insert-dir 'NOMSG))
;; REPLACE ORIGINAL in `info.el':
;;
;; Added optional arg NOMSG. If non-nil then do not show progress messages.
;;
(defun Info-insert-dir (&optional nomsg)
(if (and Info-dir-contents
Info-dir-file-attributes
(eval (cons 'and ; Verify that none of the files we used has changed since we used it.
(mapcar (lambda (elt)
(let ((curr (file-attributes (file-truename (car elt))))) ; Handle symlinks
(when curr (setcar (nthcdr 4 curr) 0)) ; Don't compare the access time.
(setcar (nthcdr 4 (cdr elt)) 0)
(equal (cdr elt) curr)))
Info-dir-file-attributes))))
(progn (insert Info-dir-contents) (goto-char (point-min)))
(let ((dirs (if Info-additional-directory-list
(append Info-directory-list Info-additional-directory-list)
Info-directory-list))
(dir-file-attrs ())
(case-fold-search t) ; Bind this in case the user sets it to nil.
problems ; This is set non-nil if we find a problem in some input files.
buffers buffer others nodes dirs-done)
(while dirs ; Search the directory list for the directory file.
(let ((truename (file-truename (expand-file-name (car dirs)))))
(or (member truename dirs-done)
(member (directory-file-name truename) dirs-done)
;; Try several variants of specified name. Try upcasing, appending `.info', or both.
(let* (file
(attrs (or (progn (setq file (expand-file-name "dir" truename)) (file-attributes file))
(progn (setq file (expand-file-name "DIR" truename)) (file-attributes file))
(progn (setq file (expand-file-name "dir.info" truename))
(file-attributes file))
(progn (setq file (expand-file-name "DIR.INFO" truename))
(file-attributes file))
;; Shouldn't really happen, but sometimes does, eg on Debian systems with
;; buggy packages; so may as well try it.
;; https://lists.gnu.org/archive/html/emacs-devel/2012-03/msg00005.html
(progn (setq file (expand-file-name "dir.gz" truename))
(file-attributes file)))))
(setq dirs-done (cons truename (cons (directory-file-name truename) dirs-done)))
(when attrs
(with-current-buffer (generate-new-buffer " info dir")
(unless (or buffers nomsg) (message "Composing main Info directory..."))
(condition-case nil
;; Index nodes include null bytes. DIR files should not have indices, but who knows...
(let ((inhibit-null-byte-detection t))
(insert-file-contents file)
(set (make-local-variable 'Info-dir-file-name) file)
(push (current-buffer) buffers)
(push (cons file attrs) dir-file-attrs))
(error (kill-buffer (current-buffer))))))))
(unless (cdr dirs)
(set (make-local-variable 'Info-dir-contents-directory) (file-name-as-directory (car dirs))))
(setq dirs (cdr dirs))))
(unless buffers (info-user-error "Can't find the Info directory node"))
;; Distinguish the dir file that comes with Emacs from all the others. Yes, that is really what this
;; is supposed to do. The definition of `Info-directory-list' puts it first on that list and so last in
;; `buffers' at this point.
(setq buffer (car (last buffers))
others (delq buffer buffers))
;; Insert the entire original dir file as a start; note that we've already saved its default directory
;; to use as the default directory for the whole concatenation.
(save-excursion (insert-buffer-substring buffer))
(dolist (other others) ; Look at each of the other buffers, one by one.
(let (this-buffer-nodes)
(with-current-buffer other ; In each, find all the menus.
(goto-char (point-min))
(while (re-search-forward "^\\* Menu:" nil t) ; Find each menu, and add an elt to NODES for it.
(while (and (zerop (forward-line 1)) (eolp)))
(let ((beg (point))
nodename end)
(re-search-backward "^\^_")
(search-forward "Node: ")
(setq nodename (Info-following-node-name))
(search-forward "\n\^_" nil 'move)
(beginning-of-line)
(setq end (point))
(push (list nodename other beg end) this-buffer-nodes)))
(if (assoc-string "top" this-buffer-nodes t)
(setq nodes (nconc this-buffer-nodes nodes))
(setq problems t)
(unless nomsg (message "No `top' node in %s" Info-dir-file-name))))))
(re-search-forward "^\\* Menu:") ; Add to the main menu a menu item for each other node.
(forward-line 1)
(let ((menu-items '("top"))
(end (save-excursion (search-forward "\^_" nil t) (point))))
(dolist (node nodes)
(let ((nodename (car node)))
(save-excursion
(or (member (downcase nodename) menu-items)
(re-search-forward (concat "^\\* +" (regexp-quote nodename) "::") end t)
(progn (insert "* " nodename "::" "\n") (push nodename menu-items)))))))
(dolist (node nodes) ; Now take each node of each of the other buffers and merge it into the main buffer.
(let ((case-fold-search t)
(nodename (car node)))
(goto-char (point-min))
;; Find the like-named node in the main buffer.
(if (re-search-forward (concat "^\^_.*\n.*Node: " (regexp-quote nodename) "[,\n\t]") nil t)
(progn (search-forward "\n\^_" nil 'move) (beginning-of-line) (insert "\n"))
(goto-char (point-max)) ; If none exists, add one.
(insert "\^_\nFile: dir\tNode: " nodename "\n\n* Menu:\n\n"))
;; Merge text from the other buffer's menu into the menu in the like-named node in the main buffer.
(apply 'insert-buffer-substring (cdr node))))
(Info-dir-remove-duplicates)
;; Kill all the buffers we just made, including the special one excised.
(mapc 'kill-buffer (cons buffer buffers))
(goto-char (point-min))
(unless nomsg (message "Composing main Info directory...%s"
(if problems "problems encountered, see `*Messages*'" "done")))
(set (make-local-variable 'Info-dir-contents) (buffer-string))
(set (make-local-variable 'Info-dir-file-attributes) dir-file-attrs)))
(setq default-directory Info-dir-contents-directory))
;; REPLACE ORIGINAL in `info.el':
;;
;; Handle `Info-breadcrumbs-in-mode-line-mode'.
;;
(defun Info-set-mode-line ()
"Set the Info mode line.
If `Info-breadcrumbs-in-mode-line-mode' is non-nil, insert breadcrumbs."
(if Info-breadcrumbs-in-mode-line-mode
(Info-insert-breadcrumbs-in-mode-line)
(setq mode-line-buffer-identification (nconc (propertized-buffer-identification "%b")
(list
(concat
" ("
(if (stringp Info-current-file)
(replace-regexp-in-string
"%" "%%" (file-name-nondirectory Info-current-file))
(format "*%S*" Info-current-file))
") "
(if Info-current-node
(propertize
(replace-regexp-in-string
"%" "%%" Info-current-node)
'face 'mode-line-buffer-id
'help-echo
"mouse-1: scroll forward, mouse-3: scroll back"
'mouse-face 'mode-line-highlight
'local-map Info-mode-line-node-keymap)
"")))))))
(defun Info-insert-breadcrumbs-in-mode-line ()
(let ((nodes (Info-toc-nodes Info-current-file))
(node Info-current-node)
(crumbs ())
(depth Info-breadcrumbs-depth-internal)
(text ""))
;; Get ancestors from the cached parent-children node info
(while (and (not (equal "Top" node)) (> depth 0))
(setq node (nth 1 (assoc node nodes)))
(when node (push node crumbs))
(setq depth (1- depth)))
;; Add bottom node.
(setq crumbs (nconc crumbs (list Info-current-node)))
(when crumbs
;; Add top node (and continuation if needed).
(setq crumbs (cons "Top" (if (member (pop crumbs) '(nil "Top"))
crumbs
(cons nil crumbs))))
(dolist (node crumbs)
(let ((crumbs-map (make-sparse-keymap))
(menu-map (make-sparse-keymap "Breadcrumbs in Mode Line")))
(define-key crumbs-map [mode-line mouse-3] menu-map)
(when node
(define-key menu-map [Info-prev]
`(menu-item "Previous Node" Info-prev
:visible ,(Info-check-pointer "prev[ious]*") :help "Go to the previous node"))
(define-key menu-map [Info-next]
`(menu-item "Next Node" Info-next
:visible ,(Info-check-pointer "next") :help "Go to the next node"))
(define-key menu-map [separator] '("--"))
(define-key menu-map [Info-breadcrumbs-in-mode-line-mode]
`(menu-item "Toggle Breadcrumbs" Info-breadcrumbs-in-mode-line-mode
:help "Toggle displaying breadcrumbs in the Info mode-line"
:button (:toggle . Info-breadcrumbs-in-mode-line-mode)))
(define-key menu-map [Info-set-breadcrumbs-depth]
`(menu-item "Set Breadcrumbs Depth" Info-set-breadcrumbs-depth
:help "Set depth of breadcrumbs to show in the mode-line"))
(setq node (if (equal node Info-current-node)
(propertize
(replace-regexp-in-string "%" "%%" Info-current-node)
'face 'mode-line-buffer-id
'help-echo "mouse-1: Scroll back, mouse-2: Scroll forward, mouse-3: Menu"
'mouse-face 'mode-line-highlight
'local-map
(progn
(define-key crumbs-map [mode-line mouse-1] 'Info-mouse-scroll-down)
(define-key crumbs-map [mode-line mouse-2] 'Info-mouse-scroll-up)
crumbs-map))
(propertize
node
'local-map (progn (define-key crumbs-map [mode-line mouse-1]
`(lambda () (interactive) (Info-goto-node ,node)))
(define-key crumbs-map [mode-line mouse-2]
`(lambda () (interactive) (Info-goto-node ,node)))
crumbs-map)
'mouse-face 'mode-line-highlight
'help-echo "mouse-1, mouse-2: Go to this node; mouse-3: Menu")))))
(let ((nodetext (if (not (equal node "Top"))
node
(concat (format "(%s)" (if (stringp Info-current-file)
(file-name-nondirectory Info-current-file)
;; Some legacy code can still use a symbol.
Info-current-file))
node))))
(setq text (concat text (if (equal node "Top") "" " > ") (if node nodetext "...")))))
(make-local-variable 'mode-line-format) ; Needed for Emacs 21+.
(setq mode-line-format text))))
;;; ;; REPLACE ORIGINAL in `info.el':
;;; ;;
;;; ;; BUG FIX (bug #1085, reported 2008-10-04).
;;; ;; 1. Match closing paren, if present.
;;; ;; 2. If only opening paren and CODE = t, then wrap each file name in ().
;;; ;;
;;; (defun Info-read-node-name-1 (string predicate code)
;;; "Internal function used by `Info-read-node-name'.
;;; See `completing-read' for a description of arguments and usage."
;;; (cond ((string-match "\\`(\\([^)]*\\))\\'" string) ; e.g. (emacs) or (emacs-mime)
;;; (cond ((eq code nil) string)
;;; ((eq code t) (list string))
;;; (t t)))
;;; ((string-match "\\`(\\([^)]*\\)\\'" string) ; e.g. (emacs
;;; (let ((ctwc (completion-table-with-context
;;; "("
;;; (apply-partially
;;; 'completion-table-with-terminator ")"
;;; (apply-partially 'Info-read-node-name-2
;;; Info-directory-list
;;; (mapcar 'car Info-suffix-list)))
;;; (match-string 1 string)
;;; predicate
;;; code)))
;;; (cond ((eq code nil) ctwc)
;;; ((eq code t) (mapcar (lambda (file) (concat "(" file ")")) ctwc))
;;; (t t))))
;;; ((string-match "\\`(" string) ; e.g. (emacs)Mac OS or (jlkj - just punt.
;;; (cond ((eq code nil) string)
;;; ((eq code t) nil)
;;; (t t)))
;;; ;; Otherwise use Info-read-node-completion-table - e.g. Mac OS
;;; (t (complete-with-action code Info-read-node-completion-table string predicate))))
;; REPLACE ORIGINAL in `info.el':
;;
;; 1. Added optional arg MSGP.
;; 2. Added in-progress message ("Looking...")
;; 3, Return nil if not found.
;;
(defun Info-find-emacs-command-nodes (command &optional msgp)
"Return a list of locations documenting COMMAND.
The `info-file' property of COMMAND says which Info manual to search.
If COMMAND has no property, the variable `Info-file-list-for-emacs'
defines heuristics for which Info manual to try.
The locations are of the format used in variable `Info-history', that
is, (FILENAME NODENAME BUFFERPOS\), where BUFFERPOS is the line number
of the first element of the returned list (which is treated specially
in `Info-goto-emacs-command-node'), and 0 for the other elements of
the list."
(let ((where ())
(cmd-desc (concat "^\\* +" (regexp-quote (symbol-name command))
"\\( <[0-9]+>\\)?:\\s *\\(.*\\)\\."
"\\(?:[ \t\n]+(line +\\([0-9]+\\))\\)?"))
(info-file "emacs") ;default
line-number)
;; Determine which info file this command is documented in.
(if (get command 'info-file)
(setq info-file (get command 'info-file))
;; If it doesn't say explicitly, test its name against
;; various prefixes that we know.
(let ((file-list Info-file-list-for-emacs))
(while file-list
(let* ((elt (car file-list))
(name (if (consp elt) (car elt) elt))
(file (if (consp elt) (cdr elt) elt))
(case-fold-search nil)
(regexp (concat "\\`" (regexp-quote name) "\\(\\'\\|-\\)")))
(if (string-match regexp (symbol-name command))
(setq info-file file
file-list ()))
(setq file-list (cdr file-list))))))
(when msgp (message "Looking for command `%s' in Info manual `%s'..." command (file-name-nondirectory info-file)))
(save-excursion
(condition-case nil
(progn (Info-find-node info-file "Top" (not msgp))
(or (and (search-forward "\n* menu:" nil t)
(re-search-forward "\n\\* \\(.*\\<Index\\>\\)" nil t))
(info-user-error "Info file `%s' appears to lack an index" info-file)))
(error nil)) ; Return nil: not found.
(goto-char (match-beginning 1))
;; Bind Info-history to nil, to prevent the index nodes from
;; getting into the node history.
(let ((Info-history ())
(Info-history-list ())
node
(nodes (Info-index-nodes)))
(Info-goto-node (car nodes))
(while (progn (goto-char (point-min))
(while (re-search-forward cmd-desc nil t)
(setq where (cons (list Info-current-file (match-string-no-properties 2) 0)
where)
line-number (and (match-beginning 3) (string-to-number (match-string 3)))))
(and (setq nodes (cdr nodes)
node (car nodes))))
(Info-goto-node node)))
(if (and line-number where)
(cons (list (nth 0 (car where)) (nth 1 (car where)) line-number) (cdr where))
where))))
;; REPLACES ORIGINAL in `info.el':
;;
;; 1. Uses `completing-read' in interactive spec, with `symbol-nearest-point'
;; (defined in `thingatpt+.el') or `symbol-at-point' (defined in `thingatpt.el').
;; 2. Added optional arg MSGP (interactive-p).
;; 3. Message if single node found.
;; 4. Returns `num-matches' if found; nil if not.
;; 4. Pass MSGP to `Info-find-emacs-command-nodes'.
;;
;;;###autoload
(defun Info-goto-emacs-command-node (command &optional msgp)
"Go to the Info node in the Emacs manual for command COMMAND.
The command is found by looking it up in Emacs manual's indexes,
or in another manual found via COMMAND's `info-file' property or
the variable `Info-file-list-for-emacs'.
COMMAND must be a symbol or string."
(interactive
(let ((symb (cond ((fboundp 'symbol-nearest-point) (symbol-nearest-point))
((fboundp 'symbol-at-point) (symbol-at-point))
(t nil)))
(enable-recursive-minibuffers t))
(list (intern (completing-read "Find documentation for command: "
obarray 'commandp t nil nil (symbol-name symb) t))
t)))
(unless (commandp command)
(signal 'wrong-type-argument (list 'commandp command)))
(let ((where (Info-find-emacs-command-nodes command msgp)))
(if where
(let ((num-matches (length where)))
;; Get Info running, and pop to it in another window.
(save-window-excursion (info))
(unless (eq major-mode 'Info-mode) (pop-to-buffer "*info*"))
;; Bind Info-history to nil, to prevent the last Index node visited by
;; `Info-find-emacs-command-nodes' from being pushed onto the history.
(let ((Info-history ())
(Info-history-list ()))
(Info-find-node (car (car where)) (car (cdr (car where)))))
(if (<= num-matches 1)
(when msgp (message "This info node documents command `%s'." command))
;; (car where) will be pushed onto Info-history
;; when/if they go to another node. Put the other
;; nodes that were found on the history.
(setq Info-history (nconc (cdr where) Info-history))
(when msgp
(message "Found %d other entr%s. Use %s to see %s."
(1- num-matches) (if (> num-matches 2) "ies" "y")
(substitute-command-keys "\\<Info-mode-map>\\[Info-history-back]")
(if (> num-matches 2) "them" "it"))))
num-matches) ; Return num-matches found.
(and (interactive-p) ; Return nil for unfound.
(info-user-error "No documentation found for command `%s'" command)))))
;; REPLACES ORIGINAL in `info.el':
;; 1. If key's command is not found, then `Info-search' for key sequence in text.
;; 2. Added optional arg MSGP (interactive-p).
;; 3. Message for repeating.
;;
;;;###autoload
(defun Info-goto-emacs-key-command-node (key &optional msgp)
"Go to the node in the Emacs manual describing command bound to KEY.
KEY is a string.
Interactively, if the binding is `execute-extended-command', then a
command is read.
The command is found by looking it up in Emacs manual's indexes,
or in another manual's index found via COMMAND's `info-file' property
or the variable `Info-file-list-for-emacs'.
If key's command cannot be found by looking in indexes, then
`Info-search' is used to search for the key sequence in the info text."
(interactive "kFind documentation for key: \np")
(let ((command (lookup-key global-map key))
(pp-key (key-description key)))
(when (natnump command) (setq command (key-binding key))) ; E.g. menu item.
(cond ((null command)
(when msgp (message "No doc found for key sequence `%s'." pp-key))
nil) ; RETURN nil: not found.
((and (interactive-p) (eq command 'execute-extended-command)) ; Read a new command name.
(Info-goto-emacs-command-node (read-command "Find documentation for command: ") msgp))
(t
(let ((this-file Info-current-file)
(this-node Info-current-node)
(num-cmd-matches (Info-goto-emacs-command-node command msgp)))
(cond (num-cmd-matches
;; Found key's command via a manual index.
(when msgp
(if (<= num-cmd-matches 1)
(message "This info node documents key `%s'." pp-key)
(message
(substitute-command-keys
(concat "Found %d other entr%s. Use "
"\\<Info-mode-map>`\\[Info-history-back]' to see %s."))
(1- num-cmd-matches) (if (> num-cmd-matches 2) "ies" "y")
(if (> num-cmd-matches 2) "them" "it"))))
num-cmd-matches) ; RETURN num-cmd-matches: found.
(t ;; Couldn't find key's command via a manual index.
;; Get back to where we were.
;; Would be better if there were a save-xxx-excursion-xxx
;; that would work.
(Info-goto-node (concat "(" this-file ")" this-node))
;; Would be better to now try looking for the key in indexes (e.g. Key
;; Index). Instead, just look for the key sequence in the text.
(when msgp
(message "Not found using Index. Searching for \"%s\" in text..." pp-key)
(sit-for 3))
(condition-case err
(progn
(Info-search (regexp-quote pp-key))
(when msgp
(message (substitute-command-keys
"Use \\<Info-mode-map>`\\[Info-search] RET' to search again for `%s'.")
pp-key))
t) ; RETURN t: found.
(search-failed (when msgp (message "No documentation found for key `%s'." pp-key))
nil))))))))) ; RETURN nil: not found.
;; REPLACES ORIGINAL in `info.el':
;; 1. File name in face `info-file'.
;; 2. If `Info-fontify-emphasis-flag', fontify _..._.
;; 3. If `Info-fontify-quotations-flag', fontify ... or `...' in face `info-quoted-name',
;; “...” in face `info-double-quoted-name', and "..." in face `info-string'.
;; 4. If `Info-fontify-quotations-flag' and `Info-fontify-single-quote-flag' then
;; fontify ' in face `info-single-quote'.
;; 5. If `Info-fontify-quotations-flag' and `Info-fontify-angle-bracketed-flag' then
;; fontify <...> in face `info-quoted-name'.
;;
(when (not (fboundp 'Info-breadcrumbs)) ; Emacs 23.1, not 23.2+
(defun Info-fontify-node ()
"Fontify the node."
(save-excursion
(let* ((inhibit-read-only t)
(case-fold-search t)
paragraph-markers
(not-fontified-p ; the node hasn't already been fontified
(not (let ((where (next-single-property-change (point-min) 'font-lock-face)))
(and where (not (= where (point-max)))))))
(fontify-visited-p ; visited nodes need to be re-fontified
(and Info-fontify-visited-nodes
;; Don't take time to refontify visited nodes in huge nodes
Info-fontify-maximum-menu-size
(< (- (point-max) (point-min)) Info-fontify-maximum-menu-size)))
rbeg rend)
;; Fontify emphasis: _..._
;;
;; Do this first because it can remove existing highlighting.
;;
(when info-fontify-emphasis
(goto-char (point-min))
(when (and font-lock-mode not-fontified-p)
(while (re-search-forward Info-emphasis-regexp nil t)
(let ((fn (if Info-fontify-emphasis-flag #'add-text-properties #'remove-text-properties)))
(funcall fn (match-beginning 0) (1+ (match-beginning 0))
'(invisible t front-sticky nil rear-nonsticky t))
(funcall fn (1- (match-end 0)) (match-end 0)
'(invisible t front-sticky nil rear-nonsticky t))
(funcall fn (match-beginning 1) (match-end 1) '(font-lock-face info-emphasis))))))
;; Fontify header line
(goto-char (point-min))
(when (and not-fontified-p (looking-at "^File: \\([^,: \t]+\\),?[ \t]+"))
(put-text-property (match-beginning 1) (match-end 1) 'font-lock-face 'info-file))
(goto-char (point-min))
(when (and not-fontified-p (looking-at "^\\(File: [^,: \t]+,?[ \t]+\\)?"))
(while (looking-at "[ \t]*\\([^:, \t\n]+\\):[ \t]+\\([^:,\t\n]+\\),?")
(goto-char (match-end 0))
(let* ((nbeg (match-beginning 2))
(nend (match-end 2))
(tbeg (match-beginning 1))
(tag (match-string 1)))
(if (string-equal (downcase tag) "node")
(put-text-property nbeg nend 'font-lock-face 'info-header-node)
(put-text-property nbeg nend 'font-lock-face 'info-header-xref)
(put-text-property tbeg nend 'mouse-face 'highlight)
(put-text-property tbeg nend
'help-echo
(concat "mouse-2: Go to node "
(buffer-substring nbeg nend)))
;; Always set up the text property keymap.
;; It will either be used in the buffer
;; or copied in the header line.
(put-text-property tbeg nend 'keymap
(cond
((string-equal (downcase tag) "prev") Info-prev-link-keymap)
((string-equal (downcase tag) "next") Info-next-link-keymap)
((string-equal (downcase tag) "up" ) Info-up-link-keymap))))))
(when (and Info-breadcrumbs-in-header-flag (> Info-breadcrumbs-depth 0))
(Info-insert-breadcrumbs))
;; Treat header line.
(when Info-use-header-line
(goto-char (point-min))
(let* ((header-end (line-end-position))
(header
;; If we find neither Next: nor Prev: link, show the entire
;; node header. Otherwise, don't show the File: and Node:
;; parts, to avoid wasting precious space on information that
;; is available in the mode line.
(if (re-search-forward "\\(next\\|up\\|prev[ious]*\\): " header-end t)
(progn (goto-char (match-beginning 1))
(buffer-substring (point) header-end))
(if (re-search-forward "node:[ \t]*[^ \t]+[ \t]*" header-end t)
(concat "No next, prev or up links -- "
(buffer-substring (point) header-end))
(buffer-substring (point) header-end)))))
(put-text-property (point-min) (1+ (point-min))
'header-line (replace-regexp-in-string
"%"
;; Preserve text properties on duplicated `%'.
(lambda (s) (concat s s)) header))
;; Hide the part of the first line that is in the header, if it is just part.
(cond ((and Info-breadcrumbs-in-header-flag (> Info-breadcrumbs-depth 0))
(put-text-property (point-min) (1+ header-end) 'invisible t))
((not (bobp))
;; Hide the punctuation at the end, too.
(skip-chars-backward " \t,")
(put-text-property (point) header-end 'invisible t))))))
;; Fontify ..., `...', “...”, and "..."
(goto-char (point-min))
(when Info-fontify-quotations-flag (info-fontify-quotations))
;; Fontify reference items: `-- Function:', `-- Variable:', etc.
(goto-char (point-min))
(when Info-fontify-reference-items-flag (info-fontify-reference-items))
;; Fontify titles
(goto-char (point-min))
(when (and font-lock-mode not-fontified-p)
(while (and (re-search-forward "\n\\([^ \t\n].+\\)\n\\(\\*\\*+\\|==+\\|--+\\|\\.\\.+\\)$" nil t)
;; Only consider it as an underlined title if the ASCII
;; underline has the same size as the text. A typical
;; counter example is when a continuation "..." is alone
;; on a line.
(= (string-width (match-string 1))
(string-width (match-string 2))))
(let* ((c (preceding-char))
(face (cond ((= c ?*) 'info-title-1)
((= c ?=) 'info-title-2)
((= c ?-) 'info-title-3)
(t 'info-title-4))))
(put-text-property (match-beginning 1) (match-end 1)
'font-lock-face face))
;; This is a serious problem for trying to handle multiple
;; frame types at once. We want this text to be invisible
;; on frames that can display the font above.
(when (memq (framep (selected-frame)) '(x pc w32 ns))
(add-text-properties (1- (match-beginning 2)) (match-end 2)
'(invisible t front-sticky nil rear-nonsticky t)))))
;; Fontify cross references
(goto-char (point-min))
(when (or not-fontified-p fontify-visited-p)
(while (re-search-forward
"\\(\\*Note[ \n\t]+\\)\\([^:]*\\)\\(:[ \t]*\\([^.,:(]*\\)\\(\\(([^)]\
*)\\)[^.,:]*\\)?[,:]?\n?\\)"
nil t)
(let ((start (match-beginning 0))
(next (point))
other-tag)
(when not-fontified-p
(when Info-hide-note-references
(when (and (not (eq Info-hide-note-references 'hide))
(> (line-number-at-pos) 4)) ; Skip breadcrumbs
;; *Note is often used where *note should have been
(goto-char start)
(skip-syntax-backward " ")
(when (memq (char-before) '(?\( ?\[ ?\{))
;; Check whether the paren is preceded by
;; an end of sentence
(skip-syntax-backward " ("))
(setq other-tag (cond ((save-match-data (looking-back "\\<see"))
"")
((save-match-data (looking-back "\\<in"))
"")
((memq (char-before) '(nil ?\. ?! ??))
"See ")
((save-match-data
(save-excursion (search-forward "\n\n" start t)))
"See ")
(t "see "))))
(goto-char next)
(add-text-properties (match-beginning 1)
(or (save-match-data
(let ((start1 (match-beginning 1))) ; Don't hide \n after *Note
(and (string-match "\n" (match-string 1))
(+ start1 (match-beginning 0)))))
(match-end 1))
(if other-tag
`(display ,other-tag front-sticky nil rear-nonsticky t)
'(invisible t front-sticky nil rear-nonsticky t))))
(add-text-properties (match-beginning 2) (match-end 2)
(list 'help-echo (if (or (match-end 5) (not (equal (match-string 4) "")))
(concat "mouse-2: go to " (or (match-string 5)
(match-string 4)))
"mouse-2: go to this node")
'mouse-face 'highlight)))
(when (or not-fontified-p fontify-visited-p)
(setq rbeg (match-beginning 2)
rend (match-end 2))
(put-text-property
rbeg
rend
'font-lock-face
;; Display visited nodes in a different face
(if (and Info-fontify-visited-nodes
(save-match-data
(let* ((node
(replace-regexp-in-string
"^[ \t]+" ""
(replace-regexp-in-string
"[ \t\n]+" " "
(or (match-string-no-properties 5)
(and (not (equal (match-string 4) ""))
(match-string-no-properties 4))
(match-string-no-properties 2)))))
(external-link-p (string-match "(\\([^)]+\\))\\([^)]*\\)" node))
(file (if external-link-p
(file-name-nondirectory
(match-string-no-properties 1 node))
Info-current-file))
(hl Info-history-list)
res)
(when external-link-p
(setq node (if (equal (match-string 2 node) "")
"Top"
(match-string-no-properties 2 node))))
(while hl
(if (and (string-equal node (nth 1 (car hl)))
(equal file (if (and external-link-p (stringp (caar hl)))
(file-name-nondirectory (caar hl))
(caar hl))))
(setq res (car hl)
hl nil)
(setq hl (cdr hl))))
res)))
'info-xref-visited
'info-xref))
;; For multiline ref, unfontify newline and surrounding whitespace
(save-excursion
(goto-char rbeg)
(save-match-data (while (re-search-forward "\\s-*\n\\s-*" rend t nil)
(remove-text-properties (match-beginning 0) (match-end 0)
'(font-lock-face t))))))
(when not-fontified-p
(when (memq Info-hide-note-references '(t hide))
(add-text-properties (match-beginning 3) (match-end 3)
'(invisible t front-sticky nil rear-nonsticky t))
;; Unhide the file name of the external reference in parens
(if (and (match-string 6) (not (eq Info-hide-note-references 'hide)))
(remove-text-properties (match-beginning 6) (match-end 6) '(invisible t front-sticky nil
rear-nonsticky t)))
;; Unhide newline because hidden newlines cause too long lines
(save-match-data (let ((beg3 (match-beginning 3))
(end3 (match-end 3)))
(if (and (string-match "\n[ \t]*" (match-string 3))
(not (save-match-data (save-excursion (goto-char (1+ end3))
(looking-at "[.)]*$")))))
(remove-text-properties (+ beg3 (match-beginning 0))
(+ beg3 (match-end 0))
'(invisible t front-sticky nil
rear-nonsticky t))))))
(when (and Info-refill-paragraphs Info-hide-note-references)
(push (set-marker (make-marker) start) paragraph-markers))))))
;; Refill paragraphs (experimental feature)
(when (and not-fontified-p Info-refill-paragraphs paragraph-markers)
(let ((fill-nobreak-invisible t)
(fill-individual-varying-indent nil)
(paragraph-start "\f\\|[ \t]*[-*]\\|[ \t]*$")
(paragraph-separate ".*\\.[ \t]*\n[ \t]\\|[ \t]*[-*]\\|[ \t\f]*$")
(adaptive-fill-mode nil))
(goto-char (point-max))
(dolist (m paragraph-markers)
(when (< m (point))
(goto-char m)
(beginning-of-line)
(let ((beg (point)))
(when (zerop (forward-paragraph))
(fill-individual-paragraphs beg (point) nil nil)
(goto-char beg))))
(set-marker m nil))))
;; Fontify menu items
(goto-char (point-min))
(when (and (or not-fontified-p fontify-visited-p)
(search-forward "\n* Menu:" nil t)
Info-fontify-maximum-menu-size ; Don't take time to annotate huge menus
(< (- (point-max) (point)) Info-fontify-maximum-menu-size))
(let ((n 0)
cont)
(while (re-search-forward (concat "^\\* Menu:\\|\\(?:^\\* +\\(" Info-menu-entry-name-re "\\)\\(:"
Info-node-spec-re "\\([ \t]*\\)\\)\\)")
nil t)
(when (match-beginning 1)
(when not-fontified-p
(setq n (1+ n))
(if (and (<= n 9) (zerop (% n 3))) ; visual aids to help with 1-9 keys
(put-text-property (match-beginning 0) (1+ (match-beginning 0))
'font-lock-face 'info-menu-star)))
(when not-fontified-p
(add-text-properties
(match-beginning 1) (match-end 1)
(list 'help-echo (if (and (match-end 3) (not (equal (match-string 3) "")))
(concat "mouse-2: go to " (match-string 3))
"mouse-2: go to this node")
'mouse-face 'highlight)))
(when (or not-fontified-p fontify-visited-p)
(put-text-property
(match-beginning 1) (match-end 1)
'font-lock-face
;; Display visited menu items in a different face
(if (and Info-fontify-visited-nodes
(save-match-data
(let* ((node (if (equal (match-string 3) "")
(match-string-no-properties 1)
(match-string-no-properties 3)))
(external-link-p (string-match "(\\([^)]+\\))\\([^)]*\\)" node))
(file (if external-link-p
(file-name-nondirectory
(match-string-no-properties 1 node))
Info-current-file))
(hl Info-history-list)
res)
(when external-link-p
(setq node (if (equal (match-string 2 node) "")
"Top"
(match-string-no-properties 2 node))))
(while hl
(if (and (string-equal node (nth 1 (car hl)))
(equal file (if (and external-link-p (stringp (caar hl)))
(file-name-nondirectory (caar hl))
(caar hl))))
(setq res (car hl)
hl nil)
(setq hl (cdr hl))))
res)))
'info-xref-visited
'info-xref)))
(when (and not-fontified-p
(memq Info-hide-note-references '(t hide))
(not (Info-index-node)))
(put-text-property (match-beginning 2) (1- (match-end 6)) 'invisible t)
;; Unhide the file name in parens
(if (and (match-end 4) (not (eq (char-after (match-end 4)) ?.)))
(remove-text-properties (match-beginning 4) (match-end 4) '(invisible t)))
;; We need a stretchable space like :align-to but with a minimum value.
(put-text-property (1- (match-end 6)) (match-end 6) 'display (if (>= 22
(- (match-end 1)
(match-beginning 0)))
'(space :align-to 24)
'(space :width 2)))
(setq cont (looking-at "."))
(while (and (= (forward-line 1) 0) (looking-at "\\([ \t]+\\)[^*\n]"))
(put-text-property (match-beginning 1) (1- (match-end 1)) 'invisible t)
(put-text-property (1- (match-end 1)) (match-end 1) 'display (if cont
'(space :align-to 26)
'(space :align-to 24)))
(setq cont t)))))))
;; Fontify menu headers
(goto-char (point-min))
(when (and not-fontified-p ; Add face `info-menu-header' to any header before a menu entry
(re-search-forward "^\\* Menu:" nil t))
(put-text-property (match-beginning 0) (match-end 0)
'font-lock-face 'info-menu-header)
(while (re-search-forward "\n\n\\([^*\n ].*\\)\n\n?[*]" nil t)
(put-text-property (match-beginning 1) (match-end 1)
'font-lock-face 'info-menu-header)))
(goto-char (point-min))
(when (and not-fontified-p (Info-index-node)) ; Hide index line numbers
(while (re-search-forward "[ \t\n]*(line +[0-9]+)" nil t)
(put-text-property (match-beginning 0) (match-end 0)
'invisible t)))
(goto-char (point-min))
(when not-fontified-p ; Fontify http and ftp references
(while (re-search-forward "\\(https?\\|ftp\\)://[^ \t\n\"`({<>})']+" nil t)
(add-text-properties (match-beginning 0) (match-end 0)
'(font-lock-face info-xref
mouse-face highlight
help-echo "mouse-2: go to this URL"))))
(goto-char (point-max))
(skip-chars-backward "\n") ; Hide any empty lines at the end of the node.
(when (< (1+ (point)) (point-max)) (put-text-property (1+ (point)) (point-max) 'invisible t))
(set-buffer-modified-p nil))))
)
;; REPLACES ORIGINAL in `info.el':
;;
;; 1. File name in face `info-file'.
;; 2. If `Info-fontify-emphasis-flag', fontify _..._.
;; 3. If `Info-fontify-quotations-flag', fontify ... or `...' in face `info-quoted-name',
;; “...” in face `info-double-quoted-name', and "..." in face `info-string'.
;; 4. If `Info-fontify-quotations-flag' and `Info-fontify-single-quote-flag' then
;; fontify ' in face `info-single-quote'.
;; 5. If `Info-fontify-quotations-flag' and `Info-fontify-angle-bracketed-flag' then
;; fontify <...> in face `info-quoted-name'.
;;
(when (and (fboundp 'Info-breadcrumbs) ; Emacs 23.2 through 24.1
(or (= emacs-major-version 23)
(and (= emacs-major-version 24) (= emacs-minor-version 1))))
(defun Info-fontify-node ()
"Fontify the node."
(save-excursion
(let* ((inhibit-read-only t)
(case-fold-search t)
paragraph-markers
(not-fontified-p ; the node hasn't already been fontified
(not (let ((where (next-single-property-change (point-min) 'font-lock-face)))
(and where (not (= where (point-max)))))))
(fontify-visited-p ; visited nodes need to be re-fontified
(and Info-fontify-visited-nodes
Info-fontify-maximum-menu-size ; Don't take time to refontify visited nodes in huge nodes
(< (- (point-max) (point-min)) Info-fontify-maximum-menu-size)))
rbeg rend)
;; Fontify emphasis: _..._
;;
;; Do this first because it can remove existing highlighting.
;;
(when info-fontify-emphasis
(goto-char (point-min))
(when (and font-lock-mode not-fontified-p)
(while (re-search-forward Info-emphasis-regexp nil t)
(let ((fn (if Info-fontify-emphasis-flag #'add-text-properties #'remove-text-properties)))
(funcall fn (match-beginning 0) (1+ (match-beginning 0))
'(invisible t front-sticky nil rear-nonsticky t))
(funcall fn (1- (match-end 0)) (match-end 0)
'(invisible t front-sticky nil rear-nonsticky t))
(funcall fn (match-beginning 1) (match-end 1) '(font-lock-face info-emphasis))))))
;; Fontify header line
(goto-char (point-min))
(when (and not-fontified-p (looking-at "^File: \\([^,: \t]+\\),?[ \t]+"))
(put-text-property (match-beginning 1) (match-end 1) 'font-lock-face 'info-file))
(goto-char (point-min))
(when (and not-fontified-p (looking-at "^\\(File: [^,: \t]+,?[ \t]+\\)?"))
(while (looking-at "[ \t]*\\([^:, \t\n]+\\):[ \t]+\\([^:,\t\n]+\\),?")
(goto-char (match-end 0))
(let* ((nbeg (match-beginning 2))
(nend (match-end 2))
(tbeg (match-beginning 1))
(tag (match-string 1)))
(if (string-equal (downcase tag) "node")
(put-text-property nbeg nend 'font-lock-face 'info-header-node)
(put-text-property nbeg nend 'font-lock-face 'info-header-xref)
(put-text-property tbeg nend 'mouse-face 'highlight)
(put-text-property tbeg nend
'help-echo
(concat "mouse-2: Go to node "
(buffer-substring nbeg nend)))
;; Always set up the text property keymap.
;; It will either be used in the buffer
;; or copied in the header line.
(put-text-property tbeg nend 'keymap
(cond
((string-equal (downcase tag) "prev") Info-prev-link-keymap)
((string-equal (downcase tag) "next") Info-next-link-keymap)
((string-equal (downcase tag) "up" ) Info-up-link-keymap))))))
;; Treat header line.
(when Info-use-header-line
(goto-char (point-min))
(let* ((header-end (line-end-position))
(header
;; If we find neither Next: nor Prev: link, show the entire
;; node header. Otherwise, don't show the File: and Node:
;; parts, to avoid wasting precious space on information that
;; is available in the mode line.
(if (re-search-forward "\\(next\\|up\\|prev[ious]*\\): " header-end t)
(progn (goto-char (match-beginning 1))
(buffer-substring (point) header-end))
(if (re-search-forward "node:[ \t]*[^ \t]+[ \t]*" header-end t)
(concat "No next, prev or up links -- "
(buffer-substring (point) header-end))
(buffer-substring (point) header-end)))))
(put-text-property (point-min) (1+ (point-min))
'header-line (replace-regexp-in-string
"%"
;; Preserve text properties on duplicated `%'.
(lambda (s) (concat s s)) header))
;; Hide the part of the first line that is in the header, if it is just part.
(cond ((and Info-breadcrumbs-in-header-flag (> Info-breadcrumbs-depth 0))
(let ((ov (make-overlay (point-min) (1+ header-end))))
(overlay-put ov 'display (Info-breadcrumbs))
(overlay-put ov 'evaporate t)))
((not (bobp))
;; Hide the punctuation at the end, too.
(skip-chars-backward " \t,")
(put-text-property (point) header-end 'invisible t))))))
;; Fontify ..., `...', “...”, and "..."
(goto-char (point-min))
(when Info-fontify-quotations-flag (info-fontify-quotations))
;; Fontify reference items: `-- Function:', `-- Variable:', etc.
(goto-char (point-min))
(when Info-fontify-reference-items-flag (info-fontify-reference-items))
;; Fontify titles
(goto-char (point-min))
(when (and font-lock-mode not-fontified-p)
(while (and (re-search-forward "\n\\([^ \t\n].+\\)\n\\(\\*\\*+\\|==+\\|--+\\|\\.\\.+\\)$" nil t)
;; Only consider it as an underlined title if the ASCII underline has the same size as the
;; text. A typical counter example is when a continuation "..." is alone on a line.
(= (string-width (match-string 1)) (string-width (match-string 2))))
(let* ((c (preceding-char))
(face (cond ((= c ?*) 'info-title-1)
((= c ?=) 'info-title-2)
((= c ?-) 'info-title-3)
(t 'info-title-4))))
(put-text-property (match-beginning 1) (match-end 1)
'font-lock-face face))
;; This is a serious problem for trying to handle multiple
;; frame types at once. We want this text to be invisible
;; on frames that can display the font above.
(when (memq (framep (selected-frame)) '(x pc w32 ns))
(add-text-properties (1- (match-beginning 2)) (match-end 2)
'(invisible t front-sticky nil rear-nonsticky t)))))
;; Fontify cross references
(goto-char (point-min))
(when (or not-fontified-p fontify-visited-p)
(while (re-search-forward
"\\(\\*Note[ \n\t]+\\)\\([^:]*\\)\\(:[ \t]*\\([^.,:(]*\\)\\(\\(([^)]\
*)\\)[^.,:]*\\)?[,:]?\n?\\)"
nil t)
(let ((start (match-beginning 0))
(next (point))
other-tag)
(when not-fontified-p
(when Info-hide-note-references
(when (and (not (eq Info-hide-note-references 'hide))
(> (line-number-at-pos) 4)) ; Skip breadcrumbs
;; *Note is often used where *note should have been
(goto-char start)
(skip-syntax-backward " ")
(when (memq (char-before) '(?\( ?\[ ?\{))
(skip-syntax-backward " (")) ; Check whether the paren is preceded by an end of sentence
(setq other-tag (cond ((save-match-data (looking-back "\\<see")) "")
((save-match-data (looking-back "\\<in")) "")
((memq (char-before) '(nil ?\. ?! ??)) "See ")
((save-match-data (save-excursion (search-forward "\n\n" start t)))
"See ")
(t "see "))))
(goto-char next)
(add-text-properties
(match-beginning 1)
(or (save-match-data
(let ((start1 (match-beginning 1))) ; Don't hide \n after *Note
(and (string-match "\n" (match-string 1)) (+ start1 (match-beginning 0)))))
(match-end 1))
(if other-tag
`(display ,other-tag front-sticky nil rear-nonsticky t)
'(invisible t front-sticky nil rear-nonsticky t))))
(add-text-properties
(match-beginning 2) (match-end 2)
(list 'help-echo (if (or (match-end 5) (not (equal (match-string 4) "")))
(concat "mouse-2: go to " (or (match-string 5) (match-string 4)))
"mouse-2: go to this node")
'mouse-face 'highlight)))
(when (or not-fontified-p fontify-visited-p)
(setq rbeg (match-beginning 2)
rend (match-end 2))
(put-text-property
rbeg
rend
'font-lock-face
(if (and Info-fontify-visited-nodes ; Display visited nodes in a different face
(save-match-data
(let* ((node (replace-regexp-in-string
"^[ \t]+" ""
(replace-regexp-in-string
"[ \t\n]+" " "
(or (match-string-no-properties 5)
(and (not (equal (match-string 4) ""))
(match-string-no-properties 4))
(match-string-no-properties 2)))))
(external-link-p (string-match "(\\([^)]+\\))\\([^)]*\\)" node))
(file (if external-link-p
(file-name-nondirectory
(match-string-no-properties 1 node))
Info-current-file))
(hl Info-history-list)
res)
(when external-link-p
(setq node (if (equal (match-string 2 node) "")
"Top"
(match-string-no-properties 2 node))))
(while hl
(if (and (string-equal node (nth 1 (car hl)))
(equal file (if (and external-link-p (stringp (caar hl)))
(file-name-nondirectory (caar hl))
(caar hl))))
(setq res (car hl)
hl nil)
(setq hl (cdr hl))))
res)))
'info-xref-visited
'info-xref))
(save-excursion ; For multiline ref, unfontify newline and surrounding whitespace
(goto-char rbeg)
(save-match-data
(while (re-search-forward "\\s-*\n\\s-*" rend t nil)
(remove-text-properties (match-beginning 0) (match-end 0) '(font-lock-face t))))))
(when not-fontified-p
(when (memq Info-hide-note-references '(t hide))
(add-text-properties (match-beginning 3) (match-end 3)
'(invisible t front-sticky nil rear-nonsticky t))
;; Unhide the file name of the external reference in parens
(when (and (match-string 6) (not (eq Info-hide-note-references 'hide)))
(remove-text-properties (match-beginning 6) (match-end 6)
'(invisible t front-sticky nil rear-nonsticky t)))
(save-match-data ; Unhide newline because hidden newlines cause too long lines
(let ((beg3 (match-beginning 3))
(end3 (match-end 3)))
(when (and (string-match "\n[ \t]*" (match-string 3))
(not (save-match-data (save-excursion (goto-char (1+ end3))
(looking-at "[.)]*$")))))
(remove-text-properties (+ beg3 (match-beginning 0))
(+ beg3 (match-end 0))
'(invisible t front-sticky nil rear-nonsticky t))))))
(when (and Info-refill-paragraphs Info-hide-note-references)
(push (set-marker (make-marker) start) paragraph-markers))))))
;; Refill paragraphs (experimental feature)
(when (and not-fontified-p Info-refill-paragraphs paragraph-markers)
(let ((fill-nobreak-invisible t)
(fill-individual-varying-indent nil)
(paragraph-start "\f\\|[ \t]*[-*]\\|[ \t]*$")
(paragraph-separate ".*\\.[ \t]*\n[ \t]\\|[ \t]*[-*]\\|[ \t\f]*$")
(adaptive-fill-mode nil))
(goto-char (point-max))
(dolist (m paragraph-markers)
(when (< m (point))
(goto-char m)
(beginning-of-line)
(let ((beg (point)))
(when (zerop (forward-paragraph))
(fill-individual-paragraphs beg (point) nil nil)
(goto-char beg))))
(set-marker m nil))))
;; Fontify menu items
(goto-char (point-min))
(when (and (or not-fontified-p fontify-visited-p)
(search-forward "\n* Menu:" nil t)
Info-fontify-maximum-menu-size ; Don't take time to annotate huge menus
(< (- (point-max) (point)) Info-fontify-maximum-menu-size))
(let ((n 0)
cont)
(while (re-search-forward
(concat "^\\* Menu:\\|\\(?:^\\* +\\(" Info-menu-entry-name-re "\\)\\(:"
Info-node-spec-re "\\([ \t]*\\)\\)\\)")
nil t)
(when (match-beginning 1)
(when not-fontified-p
(setq n (1+ n))
(if (and (<= n 9) (zerop (% n 3))) ; visual aids to help with 1-9 keys
(put-text-property (match-beginning 0) (1+ (match-beginning 0))
'font-lock-face 'info-menu-star)))
(when not-fontified-p
(add-text-properties
(match-beginning 1) (match-end 1)
(list 'help-echo (if (and (match-end 3) (not (equal (match-string 3) "")))
(concat "mouse-2: go to " (match-string 3))
"mouse-2: go to this node")
'mouse-face 'highlight)))
(when (or not-fontified-p fontify-visited-p)
(put-text-property
(match-beginning 1) (match-end 1)
'font-lock-face
(if (and Info-fontify-visited-nodes ; Display visited menu items in a different face
(save-match-data
(let* ((node (if (equal (match-string 3) "")
(match-string-no-properties 1)
(match-string-no-properties 3)))
(external-link-p (string-match "(\\([^)]+\\))\\([^)]*\\)" node))
(file (if external-link-p
(file-name-nondirectory
(match-string-no-properties 1 node))
Info-current-file))
(hl Info-history-list)
res)
(when external-link-p
(setq node (if (equal (match-string 2 node) "")
"Top"
(match-string-no-properties 2 node))))
(while hl
(if (and (string-equal node (nth 1 (car hl)))
(equal file (if (and external-link-p (stringp (caar hl)))
(file-name-nondirectory (caar hl))
(caar hl))))
(setq res (car hl)
hl nil)
(setq hl (cdr hl))))
res)))
'info-xref-visited
'info-xref)))
(when (and not-fontified-p
(memq Info-hide-note-references '(t hide))
(not (Info-index-node)))
(put-text-property (match-beginning 2) (1- (match-end 6)) 'invisible t)
;; Unhide the file name in parens
(if (and (match-end 4) (not (eq (char-after (match-end 4)) ?.)))
(remove-text-properties (match-beginning 4) (match-end 4) '(invisible t)))
;; We need a stretchable space like :align-to but with a minimum value.
(put-text-property (1- (match-end 6)) (match-end 6) 'display
(if (>= 22 (- (match-end 1) (match-beginning 0)))
'(space :align-to 24)
'(space :width 2)))
(setq cont (looking-at "."))
(while (and (= (forward-line 1) 0) (looking-at "\\([ \t]+\\)[^*\n]"))
(put-text-property (match-beginning 1) (1- (match-end 1)) 'invisible t)
(put-text-property (1- (match-end 1)) (match-end 1) 'display (if cont
'(space :align-to 26)
'(space :align-to 24)))
(setq cont t)))))))
;; Fontify menu headers
(goto-char (point-min))
;; Add face `info-menu-header' to any header before a menu entry
(when (and not-fontified-p (re-search-forward "^\\* Menu:" nil t))
(put-text-property (match-beginning 0) (match-end 0) 'font-lock-face 'info-menu-header)
(while (re-search-forward "\n\n\\([^*\n ].*\\)\n\n?[*]" nil t)
(put-text-property (match-beginning 1) (match-end 1) 'font-lock-face 'info-menu-header)))
(goto-char (point-min))
(when (and not-fontified-p (Info-index-node)) ; Hide index line numbers
(while (re-search-forward "[ \t\n]*(line +[0-9]+)" nil t)
(put-text-property (match-beginning 0) (match-end 0)
'invisible t)))
(goto-char (point-min))
(when not-fontified-p ; Fontify http and ftp references
(while (re-search-forward "\\(https?\\|ftp\\)://[^ \t\n\"`({<>})']+" nil t)
(add-text-properties (match-beginning 0) (match-end 0)
'(font-lock-face info-xref
mouse-face highlight
help-echo "mouse-2: go to this URL"))))
(goto-char (point-max))
(skip-chars-backward "\n") ; Hide any empty lines at the end of the node.
(when (< (1+ (point)) (point-max)) (put-text-property (1+ (point)) (point-max) 'invisible t))
(set-buffer-modified-p nil))))
)
;; REPLACES ORIGINAL in `info.el':
;;
;; 1. File name in face `info-file'.
;; 2. If `Info-fontify-emphasis-flag', fontify _..._.
;; 3. If `Info-fontify-quotations-flag', fontify ... or `...' in face `info-quoted-name',
;; “...” in face `info-double-quoted-name', and "..." in face `info-string'.
;; 4. If `Info-fontify-quotations-flag' and `Info-fontify-single-quote-flag' then
;; fontify ' in face `info-single-quote'.
;; 5. If `Info-fontify-quotations-flag' and `Info-fontify-angle-bracketed-flag' then
;; fontify <...> in face `info-quoted-name'.
;;
(when (or (> emacs-major-version 24) ; Emacs 24.2+
(and (= emacs-major-version 24) (> emacs-minor-version 1)))
(defun Info-fontify-node ()
"Fontify the node."
(save-excursion
(let* ((inhibit-read-only t)
(case-fold-search t)
(fontify-bookmarked-p (and (boundp 'Info-fontify-bookmarked-xrefs-flag)
Info-fontify-bookmarked-xrefs-flag))
(node-not-too-large (and (or fontify-bookmarked-p Info-fontify-visited-nodes)
;; Don't take time to refontify xrefs in huge nodes
Info-fontify-maximum-menu-size
(or (eq t Info-fontify-maximum-menu-size)
(< (- (point-max) (point-min)) Info-fontify-maximum-menu-size))))
(fontify-bookmarked-p (and node-not-too-large fontify-bookmarked-p))
(fontify-visited-p (and node-not-too-large Info-fontify-visited-nodes))
(not-fontified-p (not (let ((where (next-single-property-change (point-min) 'font-lock-face)))
(and where (not (= where (point-max)))))))
paragraph-markers rbeg rend)
;; Fontify emphasis: _..._
;;
;; Do this first because it can remove existing highlighting.
;;
(goto-char (point-min))
(when (and font-lock-mode not-fontified-p)
(while (re-search-forward Info-emphasis-regexp nil t)
(let ((fn (if Info-fontify-emphasis-flag #'add-text-properties #'remove-text-properties)))
(funcall fn (match-beginning 0) (1+ (match-beginning 0))
'(invisible t front-sticky nil rear-nonsticky t))
(funcall fn (1- (match-end 0)) (match-end 0)
'(invisible t front-sticky nil rear-nonsticky t))
(funcall fn (match-beginning 1) (match-end 1) '(font-lock-face info-emphasis)))))
;; Fontify header line
(goto-char (point-min))
(when (and not-fontified-p (looking-at "^File: \\([^,: \t]+\\),?[ \t]+"))
(put-text-property (match-beginning 1) (match-end 1) 'font-lock-face 'info-file))
(goto-char (point-min))
(when (and not-fontified-p (looking-at "^\\(File: [^,: \t]+,?[ \t]+\\)?"))
(while (looking-at "[ \t]*\\([^:, \t\n]+\\):[ \t]+\\([^:,\t\n]+\\),?")
(goto-char (match-end 0))
(let* ((nbeg (match-beginning 2))
(nend (match-end 2))
(tbeg (match-beginning 1))
(tag (match-string 1)))
(if (string-equal (downcase tag) "node")
(put-text-property nbeg nend 'font-lock-face 'info-header-node)
(put-text-property nbeg nend 'font-lock-face 'info-header-xref)
(put-text-property tbeg nend 'mouse-face 'highlight)
(put-text-property tbeg nend
'help-echo
(concat "mouse-2: Go to node "
(buffer-substring nbeg nend)))
;; Set up the text property keymap. Depending on
;; `Info-use-header-line', it is either used in the
;; buffer, or copied to the header line. A symbol value
;; of the `link-args' property is handled specially by
;; `Info-mouse-follow-link'.
(put-text-property tbeg nend 'keymap Info-link-keymap)
(put-text-property tbeg nend 'link-args (intern (downcase tag))))))
;; Treat header line.
(when Info-use-header-line
(goto-char (point-min))
(let* ((header-end (line-end-position))
(header
;; If we find neither Next: nor Prev: link, show the entire
;; node header. Otherwise, don't show the File: and Node:
;; parts, to avoid wasting precious space on information that
;; is available in the mode line.
(if (re-search-forward "\\(next\\|up\\|prev[ious]*\\): " header-end t)
(progn (goto-char (match-beginning 1))
(buffer-substring (point) header-end))
(if (re-search-forward "node:[ \t]*[^ \t]+[ \t]*" header-end t)
(concat "No next, prev or up links -- "
(buffer-substring (point) header-end))
(buffer-substring (point) header-end)))))
(put-text-property (point-min) (1+ (point-min))
'header-line (replace-regexp-in-string
"%"
;; Preserve text properties on duplicated `%'.
(lambda (s) (concat s s)) header))
;; Hide the part of the first line that is in the header, if it is just part.
(cond ((and Info-breadcrumbs-in-header-flag (> Info-breadcrumbs-depth 0))
(let ((ov (make-overlay (point-min) (1+ header-end))))
(overlay-put ov 'display (Info-breadcrumbs))
(overlay-put ov 'evaporate t)))
((not (bobp))
;; Hide the punctuation at the end, too.
(skip-chars-backward " \t,")
(put-text-property (point) header-end 'invisible t)
;; Hide the suffix (`.info') of the Info file name.
(beginning-of-line)
(if (re-search-forward (format "File: %s\\([^,\n\t]+\\),"
(if (stringp Info-current-file)
(file-name-nondirectory Info-current-file)
Info-current-file))
header-end t)
(put-text-property (match-beginning 1) (match-end 1) 'invisible t)))))))
;; Fontify ..., `...', “...”, and "..."
(goto-char (point-min))
(when Info-fontify-quotations-flag (info-fontify-quotations))
;; Fontify reference items: `-- Function:', `-- Variable:', etc.
(goto-char (point-min))
(when Info-fontify-reference-items-flag (info-fontify-reference-items))
;; Fontify titles
(goto-char (point-min))
(when (and font-lock-mode not-fontified-p)
(while (and (re-search-forward "\n\\([^ \t\n].+\\)\n\\(\\*\\*+\\|==+\\|--+\\|\\.\\.+\\)$" nil t)
;; Only consider it as an underlined title if the ASCII underline has the same size as
;; the text. A typical counter example is when a continuation "..." is alone on a line.
(= (string-width (match-string 1)) (string-width (match-string 2))))
(let* ((c (preceding-char))
(face (cond ((= c ?*) 'info-title-1)
((= c ?=) 'info-title-2)
((= c ?-) 'info-title-3)
(t 'info-title-4))))
(put-text-property (match-beginning 1) (match-end 1)
'font-lock-face face))
;; This is a serious problem for trying to handle multiple
;; frame types at once. We want this text to be invisible
;; on frames that can display the font above.
(when (memq (framep (selected-frame)) '(x pc w32 ns))
(add-text-properties (1- (match-beginning 2)) (match-end 2)
'(invisible t front-sticky nil rear-nonsticky t)))))
;; Fontify cross references
(goto-char (point-min))
(when (or not-fontified-p fontify-bookmarked-p fontify-visited-p)
(while (re-search-forward
"\\(\\*Note[ \n\t]+\\)\\([^:]*\\)\\(:[ \t]*\\([^.,:(]*\\)\\(\\(([^)]\
*)\\)[^.,:]*\\)?[,:]?\n?\\)"
nil t)
(let ((start (match-beginning 0))
(next (point))
other-tag)
(when not-fontified-p
(when Info-hide-note-references
(when (and (not (eq Info-hide-note-references 'hide))
(> (line-number-at-pos) 4)) ; Skip breadcrumbs
;; *Note is often used where *note should have been
(goto-char start)
(skip-syntax-backward " ")
(when (memq (char-before) '(?\( ?\[ ?\{))
(skip-syntax-backward " (")) ; Check whether the paren is preceded by an end of sentence
(setq other-tag (cond ((save-match-data (looking-back "\\<see")) "")
((save-match-data (looking-back "\\<in")) "")
((memq (char-before) '(nil ?\. ?! ??)) "See ")
((save-match-data (save-excursion (search-forward "\n\n" start t)))
"See ")
(t "see "))))
(goto-char next)
(add-text-properties
(match-beginning 1)
(or (save-match-data
(let ((start1 (match-beginning 1))) ; Don't hide \n after *Note
(and (string-match "\n" (match-string 1)) (+ start1 (match-beginning 0)))))
(match-end 1))
(if other-tag
`(display ,other-tag front-sticky nil rear-nonsticky t)
'(invisible t front-sticky nil rear-nonsticky t))))
(add-text-properties
(match-beginning 2) (match-end 2)
(list 'help-echo (if (or (match-end 5) (not (equal (match-string 4) "")))
(concat "mouse-2: go to " (or (match-string 5) (match-string 4)))
"mouse-2: go to this node")
'mouse-face 'highlight)))
(when (or not-fontified-p fontify-bookmarked-p fontify-visited-p)
(setq rbeg (match-beginning 2)
rend (match-end 2))
(let (node)
(put-text-property
rbeg
rend
'font-lock-face
(if (and (or Info-fontify-visited-nodes fontify-bookmarked-p)
(save-match-data
(setq node (replace-regexp-in-string
"^[ \t]+" ""
(replace-regexp-in-string
"[ \t\n]+" " "
(or (match-string-no-properties 5)
(and (not (equal (match-string 4) ""))
(match-string-no-properties 4))
(match-string-no-properties 2)))))
(let* ((external-link-p (string-match "(\\([^)]+\\))\\([^)]*\\)" node))
(file (if external-link-p
(file-name-nondirectory
(match-string-no-properties 1 node))
Info-current-file))
(hl Info-history-list)
res)
(when external-link-p
(setq node (if (equal (match-string 2 node) "")
"Top"
(match-string-no-properties 2 node))))
(or (and fontify-bookmarked-p (Info-bookmark-name-for-node node))
(progn
(while hl
(if (and (string-equal node (nth 1 (car hl)))
(equal file (if (and external-link-p (stringp (caar hl)))
(file-name-nondirectory (caar hl))
(caar hl))))
(setq res (car hl)
hl nil)
(setq hl (cdr hl))))
res)))))
(let ((bmk (and fontify-bookmarked-p (Info-bookmark-for-node node 'LOCALP))))
(if bmk
(or (bmkp-get-tag-value bmk "bmkp-info-face") 'info-xref-bookmarked)
'info-xref-visited))
'info-xref)))
(save-excursion ; For multiline ref, unfontify newline and surrounding whitespace
(goto-char rbeg)
(save-match-data (while (re-search-forward "\\s-*\n\\s-*" rend t nil)
(remove-text-properties (match-beginning 0) (match-end 0)
'(font-lock-face t))))))
(when not-fontified-p
(when (memq Info-hide-note-references '(t hide))
(add-text-properties (match-beginning 3) (match-end 3)
'(invisible t front-sticky nil rear-nonsticky t))
;; Unhide the file name of the external reference in parens
(when (and (match-string 6) (not (eq Info-hide-note-references 'hide)))
(remove-text-properties (match-beginning 6) (match-end 6)
'(invisible t front-sticky nil rear-nonsticky t)))
(save-match-data ; Unhide newline because hidden newlines cause too long lines
(let ((beg3 (match-beginning 3))
(end3 (match-end 3)))
(when (and (string-match "\n[ \t]*" (match-string 3))
(not (save-match-data (save-excursion (goto-char (1+ end3))
(looking-at "[.)]*$")))))
(remove-text-properties (+ beg3 (match-beginning 0)) (+ beg3 (match-end 0))
'(invisible t front-sticky nil rear-nonsticky t))))))
(when (and Info-refill-paragraphs Info-hide-note-references)
(push (set-marker (make-marker) start) paragraph-markers))))))
;; Refill paragraphs (experimental feature)
(when (and not-fontified-p Info-refill-paragraphs paragraph-markers)
(let ((fill-nobreak-invisible t)
(fill-individual-varying-indent nil)
(paragraph-start "\f\\|[ \t]*[-*]\\|[ \t]*$")
(paragraph-separate ".*\\.[ \t]*\n[ \t]\\|[ \t]*[-*]\\|[ \t\f]*$")
(adaptive-fill-mode nil))
(goto-char (point-max))
(dolist (m paragraph-markers)
(when (< m (point))
(goto-char m)
(beginning-of-line)
(let ((beg (point)))
(when (zerop (forward-paragraph))
(fill-individual-paragraphs beg (point) nil nil)
(goto-char beg))))
(set-marker m nil))))
;; Fontify menu items
(goto-char (point-min))
(when (and (or not-fontified-p fontify-bookmarked-p fontify-visited-p)
(search-forward "\n* Menu:" nil t)
Info-fontify-maximum-menu-size ; Don't take time to annotate huge menus
(or (eq t Info-fontify-maximum-menu-size)
(< (- (point-max) (point)) Info-fontify-maximum-menu-size)))
(let ((n 0)
cont)
(while (re-search-forward (concat "^\\* Menu:\\|\\(?:^\\* +\\(" Info-menu-entry-name-re "\\)\\(:"
Info-node-spec-re "\\([ \t]*\\)\\)\\)")
nil t)
(when (match-beginning 1)
(when not-fontified-p
(setq n (1+ n))
(when (and (<= n 9) (zerop (% n 3))) ; visual aids to help with 1-9 keys
(put-text-property (match-beginning 0) (1+ (match-beginning 0))
'font-lock-face 'info-menu-star)))
(when not-fontified-p
(add-text-properties (match-beginning 1) (match-end 1)
(list 'help-echo (if (and (match-end 3)
(not (equal (match-string 3) "")))
(concat "mouse-2: go to " (match-string 3))
"mouse-2: go to this node")
'mouse-face 'highlight)))
(when (or not-fontified-p fontify-bookmarked-p fontify-visited-p)
(let (node)
(put-text-property
(match-beginning 1) (match-end 1)
'font-lock-face
(if (and (or Info-fontify-visited-nodes fontify-bookmarked-p)
(save-match-data
(setq node (if (equal (match-string 3) "")
(match-string-no-properties 1)
(match-string-no-properties 3)))
(let* ((external-link-p (string-match "(\\([^)]+\\))\\([^)]*\\)" node))
(file (if external-link-p
(file-name-nondirectory
(match-string-no-properties 1 node))
Info-current-file))
(hl Info-history-list)
res)
(when external-link-p
(setq node (if (equal (match-string 2 node) "")
"Top"
(match-string-no-properties 2 node))))
(or (and fontify-bookmarked-p (Info-bookmark-name-for-node node))
(progn
(while hl
(if (and (string-equal node (nth 1 (car hl)))
(equal file (if (and external-link-p (stringp (caar hl)))
(file-name-nondirectory (caar hl))
(caar hl))))
(setq res (car hl)
hl nil)
(setq hl (cdr hl))))
res)))))
(let ((bmk (and fontify-bookmarked-p (Info-bookmark-for-node node 'LOCALP))))
(if bmk
(or (bmkp-get-tag-value bmk "bmkp-info-face") 'info-xref-bookmarked)
'info-xref-visited))
'info-xref))))
(when (and not-fontified-p
(memq Info-hide-note-references '(t hide))
(not (Info-index-node)))
(put-text-property (match-beginning 2) (1- (match-end 6)) 'invisible t)
;; Unhide the file name in parens
(if (and (match-end 4) (not (eq (char-after (match-end 4)) ?.)))
(remove-text-properties (match-beginning 4) (match-end 4)
'(invisible t)))
;; We need a stretchable space like :align-to but with a minimum value.
(put-text-property (1- (match-end 6)) (match-end 6) 'display (if (>= 22
(- (match-end 1)
(match-beginning 0)))
'(space :align-to 24)
'(space :width 2)))
(setq cont (looking-at "."))
(while (and (= (forward-line 1) 0) (looking-at "\\([ \t]+\\)[^*\n]"))
(put-text-property (match-beginning 1) (1- (match-end 1)) 'invisible t)
(put-text-property (1- (match-end 1)) (match-end 1) 'display (if cont
'(space :align-to 26)
'(space :align-to 24)))
(setq cont t)))))))
;; Fontify menu headers
(goto-char (point-min))
(when (and not-fontified-p ; Add face `info-menu-header' to any header before a menu entry
(re-search-forward "^\\* Menu:" nil t))
(put-text-property (match-beginning 0) (match-end 0) 'font-lock-face 'info-menu-header)
(while (re-search-forward "\n\n\\([^*\n ].*\\)\n\n?[*]" nil t)
(put-text-property (match-beginning 1) (match-end 1) 'font-lock-face 'info-menu-header)))
(goto-char (point-min))
(when (and not-fontified-p (Info-index-node)) ; Hide index line numbers
(while (re-search-forward "[ \t\n]*(line +[0-9]+)" nil t)
(put-text-property (match-beginning 0) (match-end 0) 'invisible t)))
(goto-char (point-min))
(when not-fontified-p ; Fontify http and ftp references
(while (re-search-forward "\\(https?\\|ftp\\)://[^ \t\n\"`({<>})']+" nil t)
(add-text-properties (match-beginning 0) (match-end 0) '(font-lock-face info-xref
mouse-face highlight
help-echo "mouse-2: go to this URL"))))
(goto-char (point-max))
(skip-chars-backward "\n") ; Hide any empty lines at the end of the node.
(when (< (1+ (point)) (point-max)) (put-text-property (1+ (point)) (point-max) 'invisible t))
(set-buffer-modified-p nil))))
)
;;;###autoload (autoload 'Info-set-breadcrumbs-depth "info+")
(defun Info-set-breadcrumbs-depth ()
"Set current breadcrumbs depth.
Update breadcrumbs display in mode line accordingly.
You are prompted for the depth value."
(interactive)
(setq Info-breadcrumbs-depth-internal (read-number "New breadcrumbs depth: "
Info-breadcrumbs-depth-internal))
(when Info-breadcrumbs-in-mode-line-mode (Info-insert-breadcrumbs-in-mode-line)))
(defun info-fontify-quotations ()
"Fontify ..., `...', “...”, \"...\", and possibly <...> and single '.
If `Info-fontify-angle-bracketed-flag' then fontify <...> also.
If `Info-fontify-single-quote-flag' then fontify singleton ' also.
..., `...', and <...>\t- use face `info-quoted-name'.
... uses face `info-double-quoted-name'.
\"...\"\t- uses face `info-string'.
'\t- uses face `info-single-quote'."
(let ((regexp (if Info-fontify-angle-bracketed-flag info-quoted+<>-regexp info-quotation-regexp))
(property 'font-lock-face))
(while (condition-case nil (re-search-forward regexp nil t) (error nil))
(cond ((and (eq (aref (match-string 0) 0) ?`) ; Single-quote wrapped backslashes: `\', `\\', `\\\', etc.
(goto-char (match-beginning 0))
(save-match-data (looking-at "\\(`\\\\+'\\)")))
(put-text-property (1+ (match-beginning 0)) (1- (match-end 0)) property 'info-quoted-name)
(goto-char (match-end 0)))
((and (eq (aref (match-string 0) 0) ?) ; Single-quote wrapped backslashes:
(goto-char (match-beginning 0)) ; \, \\, \\\, etc.
(save-match-data (looking-at "\\(\\\\+\\)")))
(put-text-property (1+ (match-beginning 0)) (1- (match-end 0)) property 'info-quoted-name)
(goto-char (match-end 0)))
((and (memq (aref (match-string 0) 0) '(?` ?)) ; ..., `...'
(goto-char (match-beginning 0)) ; If ` or is preceded by \, then skip it
(< (save-excursion (skip-chars-backward "\\\\")) 0))
(goto-char (1+ (match-beginning 0))))
((and Info-fontify-angle-bracketed-flag
(eq ?< (aref (match-string 0) 0)) ; <...>: If < is preceded by \, then skip it
(goto-char (match-beginning 0))
(< (save-excursion (skip-chars-backward "\\\\")) 0))
(goto-char (1+ (match-beginning 0))))
((memq (aref (match-string 0) 0) '(?` ?)) ; ..., `...'
(put-text-property (1+ (match-beginning 0)) (1- (match-end 0)) property 'info-quoted-name)
(goto-char (match-end 0)) (forward-char 1))
((and Info-fontify-angle-bracketed-flag
(eq ?< (aref (match-string 0) 0))) ; <...>
(put-text-property (1+ (match-beginning 0)) (1- (match-end 0)) property 'info-quoted-name)
(goto-char (match-end 0)) (forward-char 1))
((eq (aref (match-string 0) 0) ?“) ; “...”
(put-text-property (1+ (match-beginning 0)) (1- (match-end 0)) property 'info-double-quoted-name)
(goto-char (match-end 0)) (forward-char 1))
;; Don't try to handle strings correctly. Check only the first " for being escaped.
;; The second " ends the match, even if it is escaped (odd number of \s before it).
((and (goto-char (match-beginning 0)) ; "...": If " preceded by \, then skip it
(< (save-excursion (skip-chars-backward "\\\\")) 0))
(goto-char (1+ (match-beginning 0))))
((and Info-fontify-single-quote-flag
(string= "'" (buffer-substring (match-beginning 0) (match-end 0)))) ; Single ': 'foo
(put-text-property (match-beginning 0) (match-end 0)
property 'info-single-quote)
(goto-char (match-end 0)) (forward-char 1))
((and (not (string= "'" (buffer-substring (match-beginning 0) (match-end 0)))) ; "..."
(not (string= "" (buffer-substring (match-beginning 0) (match-end 0)))))
(put-text-property (match-beginning 0) (match-end 0) property 'info-string)
(goto-char (match-end 0)) (forward-char 1))
(t
(goto-char (match-end 0)) (forward-char 1))))))
(defun info-fontify-reference-items ()
"Fontify reference items such as \"Function:\" in Info buffer."
(while (re-search-forward "^ --? \\(Command:\\|Constant:\\|Function:\\|Macro:\\|Special Form:\\|\
Syntax class:\\|User Option:\\|Variable:\\)\\(.*\\)\\(\n \\(.*\\)\\)*"
nil t)
(let ((symb (intern (match-string 1))))
(put-text-property (match-beginning 1) (match-end 1)
'font-lock-face (case symb
('Constant: 'info-constant-ref-item)
('Command: 'info-command-ref-item)
('Function: 'info-function-ref-item)
('Macro: 'info-macro-ref-item)
('Special\ Form: 'info-special-form-ref-item)
('Syntax\ class: 'info-syntax-class-item)
('User\ Option: 'info-user-option-ref-item)
('Variable: 'info-variable-ref-item)))
(put-text-property (match-beginning 2) (match-end 2)
'font-lock-face 'info-reference-item)
(when (match-beginning 4)
(put-text-property (match-beginning 4) (match-end 4)
'font-lock-face 'info-reference-item)))))
(defun Info-search-beg () ;; `isearchp-reg-beg' is defined in library `isearch+.el' for Emacs 24.3+.
"`isearchp-reg-beg', if defined and non-nil; else `point-min'."
(or (and (boundp 'isearchp-reg-beg) isearchp-reg-beg) (point-min)))
(defun Info-search-end () ;; `isearchp-reg-end' is defined in library `isearch+.el' for Emacs 24.3+.
"`isearchp-reg-end', if defined and non-nil; else `point-max'."
(or (and (boundp 'isearchp-reg-end) isearchp-reg-end) (point-max)))
(defun Info-isearch-search-p ()
"Return non-nil if isearch in Info searches through multiple nodes.
\(Returns nil if search is restricted to the active region.)"
(and Info-isearch-search (or (not (boundp 'isearchp-reg-beg)) (not isearchp-reg-beg))))
;; REPLACES ORIGINAL in `info.el':
;;
;; Use `Info-isearch-search-p', not var `Info-isearch-search'.
;;
(defun Info-isearch-wrap ()
(if (not (Info-isearch-search-p))
(goto-char (if isearch-forward (Info-search-beg) (Info-search-end)))
(if (not Info-isearch-initial-node)
(setq Info-isearch-initial-node Info-current-node
isearch-wrapped nil)
(if isearch-forward (Info-top-node) (Info-final-node))
(goto-char (if isearch-forward (Info-search-beg) (Info-search-end))))))
;; REPLACES ORIGINAL in `info.el':
;;
;; Use `Info-isearch-search-p', not var `Info-isearch-search'.
;;
(when (= emacs-major-version 23) ; Emacs 23 only
(defun Info-isearch-search () ; Use `Info-isearch-search-p', not var `Info-isearch-search'.
(if (Info-isearch-search-p)
(lambda (string &optional bound noerror count)
(if isearch-word
(Info-search (concat "\\b" (replace-regexp-in-string "\\W+" "\\W+"
(replace-regexp-in-string
"^\\W+\\|\\W+$" "" string)
nil t)
(and (or isearch-nonincremental ; Lax version of word search
(eq (length string) (length (isearch-string-state
(car isearch-cmds)))))
"\\b"))
bound noerror count (unless isearch-forward 'backward))
(Info-search (if isearch-regexp string (regexp-quote string))
bound noerror count (unless isearch-forward 'backward)))
(point))
(let ((isearch-search-fun-function nil)) (isearch-search-fun)))))
;; REPLACES ORIGINAL in `info.el':
;;
;; Use `Info-isearch-search-p', not var `Info-isearch-search'.
;;
(when (> emacs-major-version 23) ; Emacs 24+
(unless (boundp 'isearch-lax-whitespace) ; Emacs 24.1, 24.2.
(defvar isearch-lax-whitespace t
"If non-nil, a space will match a sequence of whitespace chars.
When you enter a space or spaces in ordinary incremental search, it
will match any sequence matched by the regexp defined by the variable
`search-whitespace-regexp'. If the value is nil, each space you type
matches literally, against one space. You can toggle the value of this
variable by the command `isearch-toggle-lax-whitespace'.")
(defvar isearch-regexp-lax-whitespace nil
"If non-nil, a space will match a sequence of whitespace chars.
When you enter a space or spaces in regexp incremental search, it
will match any sequence matched by the regexp defined by the variable
`search-whitespace-regexp'. If the value is nil, each space you type
matches literally, against one space. You can toggle the value of this
variable by the command `isearch-toggle-lax-whitespace'.")
)
(defun Info-isearch-search () ; Use `Info-isearch-search-p', not var `Info-isearch-search'.
(if (Info-isearch-search-p)
(lambda (string &optional bound noerror count)
(let ((Info-search-whitespace-regexp (if (if isearch-regexp
isearch-regexp-lax-whitespace
isearch-lax-whitespace)
search-whitespace-regexp)))
(Info-search (cond (isearch-word ; Lax version of word search
(let ((lax (not (or isearch-nonincremental
(eq (length string) (length (isearch--state-string
(car isearch-cmds))))))))
(if (functionp isearch-word)
(funcall isearch-word string lax)
(word-search-regexp string lax))))
(isearch-regexp string)
(t (regexp-quote string)))
bound noerror count (unless isearch-forward 'backward)))
(point))
(isearch-search-fun-default)))
)
;; REPLACES ORIGINAL in `info.el':
;;
;; 1. Fit frame if `one-window-p'.
;; 2. Highlight the found regexp if `search-highlight'.
;; 3. Respect restriction to active region.
;;
;;;###autoload (autoload 'Info-search "info+")
(defun Info-search (regexp &optional bound _noerror _count direction)
"Search for REGEXP, starting from point, and select node of search hit.
If DIRECTION is `backward', search backward.
Fits frame if `one-window-p'.
Highlights current location of found regexp if `search-highlight'.
If called interactively then this is a non-incremental search. In
that case, the search-hit highlighting remains, after the search is
over. To remove the highlighting, just start an incremental search:
`\\[isearch-forward]'."
(interactive
(list (let ((prompt (if Info-search-history
(format "Regexp search%s (default `%s'): "
(if case-fold-search "" " case-sensitively")
(car Info-search-history))
(format "Regexp search%s: "
(if case-fold-search "" " case-sensitively")))))
(if (fboundp 'icicle-read-string-completing)
(icicle-read-string-completing prompt nil nil 'Info-search-history)
(read-string prompt nil 'Info-search-history)))))
(deactivate-mark)
(when (equal regexp "")
(setq regexp (car Info-search-history)))
(when regexp
(prog1
(let (found beg-found give-up
(backward (eq direction 'backward))
(onode Info-current-node)
(ofile Info-current-file)
(opoint (point))
(opoint-min (Info-search-beg))
(opoint-max (Info-search-end))
(ostart (window-start))
(osubfile Info-current-subfile))
(setq Info-search-case-fold case-fold-search) ; `Info-search-case-fold' is free here.
(save-excursion
(save-restriction
(widen)
(when backward
(narrow-to-region (save-excursion ; Hide Info file header for backward search
(goto-char (Info-search-beg))
(search-forward "\n\^_")
(1- (point)))
(Info-search-end)))
(while (and (not give-up) (or (null found)
(not (funcall isearch-filter-predicate beg-found found))))
(let ((search-spaces-regexp Info-search-whitespace-regexp))
(if (if backward
(re-search-backward regexp bound t)
(re-search-forward regexp bound t))
(setq found (point)
beg-found (if backward (match-end 0) (match-beginning 0)))
(setq give-up t))))))
(when (and isearch-mode
(Info-isearch-search-p)
(not Info-isearch-initial-node) ; `Info-isearch-initial-node' is free here.
(not bound)
(or give-up (and found (not (and (> found opoint-min) (< found opoint-max))))))
(signal 'search-failed (list regexp "end of node")))
(when give-up ; If no subfiles, give error now.
(if (null Info-current-subfile)
(let ((search-spaces-regexp Info-search-whitespace-regexp))
(if backward (re-search-backward regexp) (re-search-forward regexp)))
(setq found nil)))
(when (and bound (not found)) (signal 'search-failed (list regexp)))
(unless (or found bound)
(unwind-protect
(let ((list ())) ; Try other subfiles.
(with-current-buffer (marker-buffer Info-tag-table-marker)
(goto-char (Info-search-beg))
(search-forward "\n\^_\nIndirect:")
(save-restriction
(narrow-to-region (point) (progn (search-forward "\n\^_") (1- (point))))
(goto-char (Info-search-beg))
(search-forward (concat "\n" osubfile ": ")) ; Find the subfile we just searched.
(forward-line (if backward 0 1)) ; Skip that one.
(if backward (forward-char -1))
;; Make a list of all following subfiles.
;; Each elt has the form (VIRT-POSITION . SUBFILENAME).
(while (not (if backward (bobp) (eobp)))
(if backward
(re-search-backward "\\(^.*\\): [0-9]+$")
(re-search-forward "\\(^.*\\): [0-9]+$"))
(goto-char (+ (match-end 1) 2))
(setq list (cons (cons (+ (Info-search-beg) (read (current-buffer)))
(match-string-no-properties 1))
list))
(goto-char (if backward (1- (match-beginning 0)) (1+ (match-end 0)))))
(setq list (nreverse list)))) ; Put in forward order
(while list
(when (interactive-p) (message "Searching subfile %s..." (cdr (car list))))
(Info-read-subfile (car (car list)))
(when backward
;; Hide Info file header for backward search
(narrow-to-region (save-excursion (goto-char (Info-search-beg))
(search-forward "\n\^_")
(1- (point)))
(Info-search-end))
(goto-char (Info-search-end)))
(setq list (cdr list)
give-up nil
found nil)
(while (and (not give-up) (or (null found)
(not (funcall isearch-filter-predicate beg-found found))))
(let ((search-spaces-regexp Info-search-whitespace-regexp))
(if (if backward
(re-search-backward regexp nil t)
(re-search-forward regexp nil t))
(setq found (point)
beg-found (if backward (match-end 0) (match-beginning 0)))
(setq give-up t))))
(when give-up (setq found nil))
(when found (setq list ())))
(if found (message "") (signal 'search-failed (list regexp))))
(unless found
(Info-read-subfile osubfile)
(goto-char opoint)
(Info-select-node)
(set-window-start (selected-window) ostart))))
(if (and (string= osubfile Info-current-subfile) (> found opoint-min) (< found opoint-max))
(goto-char found) ; Search landed in the same node
(widen)
(goto-char found)
(save-match-data (Info-select-node)))
(when search-highlight (isearch-highlight (match-beginning 0) (match-end 0))) ; Highlight regexp.
;; Use `string-equal', not `equal', to ignore text props.
(or (and (string-equal onode Info-current-node) (equal ofile Info-current-file))
(and isearch-mode isearch-wrapped (eq opoint (if isearch-forward opoint-min opoint-max)))
(setq Info-history (cons (list ofile onode opoint) Info-history)))
(when (and (one-window-p t) (not (window-minibuffer-p)) ; Fit the frame, if appropriate.
(fboundp 'fit-frame) ; Defined in `fit-frame.el'.
Info-fit-frame-flag)
(fit-frame)))
(when (and (interactive-p) (not isearch-mode))
(message (substitute-command-keys
"Use \\<Info-mode-map>`\\[Info-search] RET' to search again for `%s'.")
regexp)))))
;; REPLACES ORIGINAL in `info.el':
;; Added optional arg FORK.
;;
(defun Info-next (&optional fork) ; `n'
"Go to the next node of this node.
If FORK is non-nil (interactively with a prefix arg), show the node in
a new Info buffer.
If FORK is a string, it is the name to use for the new buffer."
(interactive "P")
(unless (derived-mode-p 'Info-mode) (Info--pop-to-buffer-same-window "*info*"))
(Info-goto-node (Info-extract-pointer "next") fork))
;; REPLACES ORIGINAL in `info.el':
;; Added optional arg FORK.
;;
(defun Info-prev (&optional fork) ; `p'
"Go to the previous node of this node.
If FORK is non-nil (interactively with a prefix arg), show the node in
a new Info buffer.
If FORK is a string, it is the name to use for the new buffer."
(interactive "P")
(unless (derived-mode-p 'Info-mode) (Info--pop-to-buffer-same-window "*info*"))
(Info-goto-node (Info-extract-pointer "prev[ious]*" "previous") fork))
;; REPLACES ORIGINAL in `info.el':
;; Added optional arg FORK.
;;
(defun Info-up (&optional same-file fork) ; `u', `^'
"Go to the superior node of this node.
If SAME-FILE is non-nil, do not move to a different Info file.
If FORK is non-nil (interactively with a prefix arg), show the node in
a new Info buffer.
If FORK is a string, it is the name to use for the new buffer."
(interactive "i\nP")
(unless (derived-mode-p 'Info-mode) (Info--pop-to-buffer-same-window "*info*"))
(let ((old-node Info-current-node)
(old-file Info-current-file)
(node (Info-extract-pointer "up"))
pp)
(when (and same-file (string-match "^(" node)) (error "Up node is in another Info file"))
(Info-goto-node node fork)
(setq pp (point))
(goto-char (point-min))
(cond ((and (stringp old-file)
(search-forward "\n* Menu:" nil t)
(re-search-forward
(if (string-equal old-node "Top")
(concat "\n\\*[^:]+: +(" (file-name-nondirectory old-file) ")")
(concat "\n\\* +\\(" (regexp-quote old-node)
":\\|[^:]+: +" (regexp-quote old-node) "\\)"))
nil t))
(beginning-of-line)
(if (looking-at "^\\* ") (forward-char 2)))
(t
(goto-char pp)
(Info-restore-point Info-history)))))
;; REPLACES ORIGINAL in `info.el':
;; Added optional arg FORK.
;;
(defun Info-history-back (&optional fork) ; `l'
"Go back in the history to the last node visited.
If FORK is non-nil (interactively with a prefix arg), show the node in
a new Info buffer.
If FORK is a string, it is the name to use for the new buffer."
(interactive "P")
(unless Info-history
(funcall (if (fboundp 'user-error) #'user-error #'error)
"This is the first Info node you have visited"))
(let ((history-forward (cons (list Info-current-file Info-current-node (point))
Info-history-forward))
filename nodename opoint)
(setq filename (caar Info-history)
nodename (car (cdar Info-history))
opoint (cadr (cdar Info-history))
Info-history (cdr Info-history))
(when fork
(set-buffer (clone-buffer (concat "*info-" (if (stringp fork) fork nodename) "*") t)))
(Info-find-node filename nodename)
(setq Info-history (cdr Info-history)
Info-history-forward history-forward)
(goto-char opoint)))
;; REPLACES ORIGINAL in `info.el':
;; Added optional arg FORK.
;;
(defun Info-history-forward (&optional fork) ; `r'
"Go forward in the history of visited nodes.
If FORK is non-nil (interactively with a prefix arg), show the node in
a new Info buffer.
If FORK is a string, it is the name to use for the new buffer."
(interactive "P")
(unless Info-history-forward
(funcall (if (fboundp 'user-error) #'user-error #'error)
"This is the last Info node you have visited"))
(let ((history-forward (cdr Info-history-forward))
filename nodename opoint)
(setq filename (caar Info-history-forward)
nodename (car (cdar Info-history-forward))
opoint (cadr (cdar Info-history-forward)))
(when fork
(set-buffer (clone-buffer (concat "*info-" (if (stringp fork) fork nodename) "*") t)))
(Info-find-node filename nodename)
(setq Info-history-forward history-forward)
(goto-char opoint)))
;; REPLACES ORIGINAL in `info.el':
;; Added optional arg FORK.
;;
(defun Info-directory (&optional fork) ; `d'
"Go to the Info directory node.
If FORK is non-nil (interactively with a prefix arg), show the node in
a new Info buffer.
If FORK is a string, it is the name to use for the new buffer."
(interactive "P")
(when fork (set-buffer (clone-buffer nil t)))
(Info-find-node "dir" "top"))
;; REPLACES ORIGINAL in `info.el':
;; Added optional arg FORK.
;;
(defun Info-forward-node (&optional not-down not-up no-error fork) ; `]'
"Go forward one node, considering all nodes as forming one sequence.
If FORK is non-nil (interactively with a prefix arg), show the node in
a new Info buffer.
If FORK is a string, it is the name to use for the new buffer."
(interactive "i\ni\ni\nP")
(goto-char (point-min))
(forward-line 1)
(let ((case-fold-search t))
;; Three possibilities, in order of priority:
;; 1. Next node is in a menu in this node (but not in an index)
;; 2. Next node is next at same level
;; 3. Next node is up and next
(cond ((and (not not-down)
(save-excursion (search-forward "\n* menu:" nil t))
(not (Info-index-node)))
(Info-goto-node (Info-extract-menu-counting 1) fork)
t)
((save-excursion (search-backward "next:" nil t))
(Info-next fork)
t)
((and (not not-up)
(save-excursion (search-backward "up:" nil t))
;; Use `string-equal', not `equal', to ignore text properties.
(not (string-equal (downcase (Info-extract-pointer "up")) "top")))
(let ((old-node Info-current-node))
(Info-up)
(let ((old-history Info-history)
success)
(unwind-protect
(setq success (Info-forward-node t nil no-error fork))
(or success (Info-goto-node old-node fork)))
(and (boundp 'Info-history-skip-intermediate-nodes) ; Emacs 24+
Info-history-skip-intermediate-nodes
(setq Info-history old-history)))))
(no-error nil)
(t (user-error "No pointer forward from this node")))))
;; REPLACES ORIGINAL in `info.el':
;; Added optional arg FORK.
;;
(defun Info-backward-node (&optional fork) ; `['
"Go backward one node, considering all nodes as forming one sequence.
If FORK is non-nil (interactively with a prefix arg), show the node in
a new Info buffer.
If FORK is a string, it is the name to use for the new buffer."
(interactive "P")
(let ((prevnode (Info-extract-pointer "prev[ious]*" t))
(upnode (Info-extract-pointer "up" t))
(case-fold-search t))
(cond ((and upnode (string-match "(" upnode)) (user-error "First node in file"))
((and upnode (or (null prevnode)
;; Use `string-equal', not `equal', to ignore text properties.
(string-equal (downcase prevnode) (downcase upnode))))
(Info-up fork))
(prevnode
;; If we move back at the same level, go down to find the last subnode*.
(Info-prev fork)
(let ((old-history Info-history)
node)
(while (and (not (Info-index-node))
(save-excursion (search-forward "\n* Menu:" nil t)))
(Info-goto-node (setq node (Info-extract-menu-counting nil))))
;;; (when fork (Info-goto-node node fork))
(and (boundp 'Info-history-skip-intermediate-nodes) ; Emacs 24+
Info-history-skip-intermediate-nodes
(setq Info-history old-history))))
(t (user-error "No pointer backward from this node")))))
;; REPLACES ORIGINAL in `info.el':
;; Added optional arg FORK.
;;
;;;###autoload (autoload 'Info-mouse-follow-nearest-node "info+")
(defun Info-mouse-follow-nearest-node (click &optional fork)
"\\<Info-mode-map>Follow a node reference near point.
Like \\[Info-menu], \\[Info-follow-reference], \\[Info-next], \\[Info-prev] or \\[Info-up] \
command, depending on where you click.
At end of the node's text, moves to the next node, or up if none.
With a prefix argument, open the node in a separate window."
(interactive "e\nP")
(mouse-set-point click)
(and (not (Info-try-follow-nearest-node fork))
(save-excursion (forward-line 1) (eobp))
(Info-next-preorder)))
;; REPLACES ORIGINAL in `info.el':
;; Use `Info-mode-syntax-table' (bug #3312).
;; Doc string changed: displays all bindings.
;;
(defun Info-mode ()
"Provides commands for browsing through the Info documentation tree.
Documentation in Info is divided into \"nodes\", each of which discusses
one topic and contains hyperlink references to other nodes that discuss
related topics. Info has commands to follow the references.
The most important commands to know are: \
\\<Info-mode-map>\
`\\[Info-exit]', `\\[Info-mouse-follow-nearest-node]', `\\[Info-history-back]', and `\\[Info-search]'.
Help commands
-------------
\\[describe-mode]\tDisplay this help.
\\[Info-help]\tThe Info tutorial. Learn about Info while using it.
Selecting other nodes (basic)
-----------------------------
\\[Info-mouse-follow-nearest-node]\tFollow a node reference you click.
\tThis works with menu items, cross references, \"Next\",
\t\"Previous\" and \"Up\" references.
\tAt end of node's text, goes to \"Next\", or \"Up\" if no \"Next\".
\\[Info-follow-nearest-node]\tLike `\\[Info-mouse-follow-nearest-node]', \
except cursor location, not mouse location.
\\[Info-history-back]\tGo back to the last node you were at. (chronological)
\\[Info-history-forward]\tGo forward to where you were before using \\[Info-history-back].
\\[Info-history]\tGo to menu of visited nodes.
\\[Info-toc]\tGo to table of contents of the current Info file.
\\[Info-index]\tLook up a topic in this file's Index and move to its node.
\\[Info-index-next]\t(comma) Go to the next match from a previous \
`\\[Info-index]' command.
Structural navigation commands
------------------------------
\\[Info-menu]\tGo to a menu item's node. Completion available for its name.
1\tGo to first menu item's node.
2, 3, 4, 5, 6, 7, 8, 9\tGo to second...ninth menu item's node.
\\[Info-next]\tGo to this node's \"Next\" node.
\\[Info-prev]\tGo to this node's \"Previous\" node. (*not* chronological)
\\[Info-up]\tGo \"Up\" from this node to its parent node.
\\[Info-directory]\tGo to the Info directory (root) node.
\\[Info-follow-reference]\tFollow a cross reference. Prompts for name.
Moving within a node
--------------------
\\[Info-scroll-up]\tNormally, scroll forward a full screen.
\tIf node's menu appears below cursor, go to first menu item.
\tIf node's menu appears above cursor, go to parent node.
\\[Info-scroll-down]\tNormally, scroll backward. If beginning of buffer is \
already
\tvisible, go to previous menu entry, or up if there is none.
\\[beginning-of-buffer]\tGo to beginning of node.
\\[Info-next-reference]\tMove cursor to next cross-reference or menu item in \
this node.
\\[Info-prev-reference]\tMove cursor to previous cross-reference or menu item.
Other navigation commands
-------------------------
\\[Info-exit]\tQuit Info.
\\[Info-goto-node]\tGo to a node with a given name.
\tYou may include a filename as well, as \"(FILENAME)NODENAME\".
\\[universal-argument] \\[info]\tGo to a new Info file. (Completion \
available.)
\\[universal-argument] N \\[info]\tOpen Info with number in buffer name: *info*<N>.
\\[Info-top-node]\tGo to first node (\"Top\") of current Info file.
\\[Info-final-node]\tGo to final node of current Info file.
\\[Info-forward-node]\tGo forward a node, considering all nodes as one \
sequence.
\\[Info-backward-node]\tGo backward a node, considering all nodes as one \
sequence.
Other commands
--------------
\\[Info-save-current-node]\tSave current node name for use by `\\[Info-virtual-book]'.
\\[Info-virtual-book]\tOpen a virtual Info book of nodes saved using `\\[Info-save-current-node]'.
\\[isearch-forward]\tIsearch this Info manual for a literal string.
\\[isearch-forward-regexp]\tIsearch this Info manual for a regexp.
\\[Info-search]\tSearch this Info manual for a regexp
\\[Info-search-case-sensitively]\tLike `\\[Info-search]', but case-sensitive.
\\[info-apropos]\tLook for a string in the indexes of all manuals.
\\[Info-copy-current-node-name]\tPut name of current info node in the kill ring.
\\[clone-buffer]\tSelect a new cloned Info buffer in another window.
\\[Info-merge-subnodes]\tIntegrate current node with nodes referred to \
in its Menu.
\tDisplay the result outside of Info. `\\[universal-argument]': Recursively.
User options you can customize
------------------------------
`Info-fontify-quotations-flag' -
Fontify quoted names (... or `...') and strings (\"...\").
Toggle with \\[Info-toggle-fontify-quotations].
`Info-fontify-angle-bracketed-flag' -
Fontify angle-bracketd names (<...>).
Toggle with \\[Info-toggle-fontify-angle-bracketed].
`Info-fontify-single-quote-flag' - Fontify single quotes (').
Toggle with \\[Info-toggle-fontify-single-quote].
`Info-saved-nodes' - Node names you can visit using `\\[Info-virtual-book]'.
`Info-subtree-separator' - See `Info-merge-subnodes'.
Faces you can customize
-----------------------
`info-file' - Face used for file heading labels
`info-string' - Face used for strings (e.g. \"toto\")
`info-double-quoted-name'
- Face used for curly double-quoted names (e.g. toto)
`info-quoted-name' - Face used for quoted names (e.g. toto or `toto')
`info-single-quote' - Face used for isolated single-quote (e.g. 'foo)
These are all of the current Info Mode bindings:
\\{Info-mode-map}"
(kill-all-local-variables)
(setq major-mode 'Info-mode
mode-name "Info"
tab-width 8)
(use-local-map Info-mode-map)
(add-hook 'activate-menubar-hook 'Info-menu-update nil t)
(set-syntax-table Info-mode-syntax-table)
(setq local-abbrev-table text-mode-abbrev-table
case-fold-search t
buffer-read-only t)
(make-local-variable 'Info-current-file)
(make-local-variable 'Info-current-subfile)
(make-local-variable 'Info-current-node)
(make-local-variable 'Info-tag-table-marker)
(setq Info-tag-table-marker (make-marker))
(make-local-variable 'Info-tag-table-buffer)
(setq Info-tag-table-buffer nil)
(make-local-variable 'Info-history)
(make-local-variable 'Info-history-forward)
(make-local-variable 'Info-index-alternatives)
(when Info-use-header-line ; do not override global header lines
(setq header-line-format '(:eval (get-text-property (point-min) 'header-line))))
(set (make-local-variable 'tool-bar-map) info-tool-bar-map)
;; This is for the sake of the invisible text we use handling titles.
(make-local-variable 'line-move-ignore-invisible)
(setq line-move-ignore-invisible t)
(make-local-variable 'desktop-save-buffer)
(make-local-variable 'widen-automatically)
(setq widen-automatically nil) ; `widen-automatically' is free here.
(setq desktop-save-buffer 'Info-desktop-buffer-misc-data)
(add-hook 'kill-buffer-hook 'Info-kill-buffer nil t)
(add-hook 'clone-buffer-hook 'Info-clone-buffer nil t)
(add-hook 'change-major-mode-hook 'font-lock-defontify nil t)
(add-hook 'isearch-mode-hook 'Info-isearch-start nil t)
;; The `Info-*' variables are free here.
(set (make-local-variable 'isearch-search-fun-function) 'Info-isearch-search)
(set (make-local-variable 'isearch-wrap-function) 'Info-isearch-wrap)
(set (make-local-variable 'isearch-push-state-function) 'Info-isearch-push-state)
(set (make-local-variable 'isearch-filter-predicate) 'Info-isearch-filter)
(unless (or (> emacs-major-version 24) (and (= emacs-major-version 24) (> emacs-minor-version 2)))
(set (make-local-variable 'search-whitespace-regexp) Info-search-whitespace-regexp))
(set (make-local-variable 'revert-buffer-function) 'Info-revert-buffer-function)
(Info-set-mode-line)
(set (make-local-variable 'bookmark-make-record-function) 'Info-bookmark-make-record)
(run-mode-hooks 'Info-mode-hook))
;; REPLACES ORIGINAL in `info.el':
;;
;; Use completion for inputting the manual name, for all Emacs versions 23+.
;;
;;;###autoload (autoload 'info-display-manual "info+")
(defun info-display-manual (manual)
"Display an Info buffer displaying MANUAL.
If there is an existing Info buffer for MANUAL, display it.
Otherwise, visit the manual in a new Info buffer.
With a prefix arg (Emacs 24.4+), completion candidates are limited to
currently visited manuals."
(interactive
(let ((manuals (and (fboundp 'info--manual-names)
(condition-case nil
(info--manual-names current-prefix-arg) ; Arg was added in Emacs 25.
(error nil)))))
(unless manuals
(condition-case nil
(with-temp-buffer
(Info-mode)
(Info-directory)
(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 (list manual)))))
(error nil)))
(list (completing-read "Display manual: " manuals nil t))))
(let ((blist (buffer-list))
(manual-re (concat "\\(/\\|\\`\\)" manual "\\(\\.\\|\\'\\)"))
(case-fold-search t)
found)
(dolist (buffer blist)
(with-current-buffer buffer
(when (and (eq major-mode 'Info-mode)
(stringp Info-current-file)
(string-match manual-re Info-current-file))
(setq found buffer
blist ()))))
(if found
(Info--pop-to-buffer-same-window found)
(info-initialize)
(info (Info-find-file manual)))))
;;(@* "Non-Interactive Functions")
;;; Non-Interactive Functions ---------------------------------------
(defun Info-display-node-default-header ()
"Insert node name as header."
;; `infop-node-name' is free here - bound in `Info-merge-subnodes'.
(insert (if (fboundp 'concat-w-faces)
(concat-w-faces (list 'info-title-1 infop-node-name)) ; FREE: INFOP-NODE-NAME
infop-node-name)
"\n")
(goto-char (point-min))
(center-line 2))
(defun Info-node-name-at-point ()
"Return the Info node named at point, or nil if none."
(save-match-data
(let ((name (cond ((Info-get-token (point) "\\*note[ \n\t]+" "\\*note[ \n\t]+\\([^:]*\\):\\(:\\|[ \n\t]*(\\)?"))
((Info-get-token (point) "\\* +" "\\* +\\([^:]*\\)::")) ; Menu item: node name
((Info-get-token (point) "\\* +" "\\* +\\(.*\\): ") ; menu item: node name or index entry
(beginning-of-line)
(forward-char 2)
(Info-extract-menu-node-name nil (Info-index-node)))
((Info-get-token (point) "Up: " "Up: \\([^,\n\t]*\\)"))
((Info-get-token (point) "Next: " "Next: \\([^,\n\t]*\\)"))
((Info-get-token (point) "File: " "File: \\([^,\n\t]*\\)"))
((Info-get-token (point) "Prev: " "Prev: \\([^,\n\t]*\\)")))))
(and name (replace-regexp-in-string "\n+" " " name)))))
(when (require 'bookmark+ nil t)
(defun Info-bookmark-for-node (&optional node localp)
"Return Info bookmark for NODE, or nil if none.
Non-nil NODE can have the form `NODE' or `(MANUAL) NODE'.
If NODE is nil then read the node name. If optional arg LOCALP is
non-nil then read the node name only from the current manual."
(when (and node (stringp Info-current-file) (not (string-match-p "(\\([^)]+\\)) \\([^)]*\\)" node)))
(setq node (concat "(" (file-name-sans-extension (file-name-nondirectory Info-current-file)) ") " node)))
(unless node (setq node (Info-read-bookmarked-node-name localp)))
(bmkp-get-bookmark-in-alist node t (bmkp-info-alist-only)))
(defun Info-bookmark-name-for-node (node)
"Return the name of an Info bookmark for NODE, or nil if none.
Non-nil NODE can have the form `NODE' or `(MANUAL) NODE'.
The name of the bookmark must be the default name that
`Info-bookmark-make-record' would use. This is normally the full node
name, `(MANUAL) NODE', where MANUAL is the lowercase name of the Info
manual. For example, node `Modes' in the Emacs manual has full
name `(emacs) Modes', and the bookmark must have that same name."
(let ((bmk (Info-bookmark-for-node node)))
(and bmk (bmkp-bookmark-name-from-record bmk))))
(defun Info-bookmark-named-at-point ()
"Return Info bookmark for node named at point, or nil if none.
See `Info-bookmark-name-for-node' for the form of the bookmark name."
(let ((node (Info-node-name-at-point)))
(and node
(let* ((file (and (stringp Info-current-file)
(file-name-sans-extension (file-name-nondirectory Info-current-file))))
(bname (if file (concat "(" file ") " node) node)))
(bmkp-get-bookmark-in-alist bname t (bmkp-info-alist-only))))))
(defun Info-bookmark-name-at-point ()
"Return name of Info bookmark for node named at point, or nil if none.
See `Info-bookmark-name-for-node' for the form of the bookmark name."
(let ((bmk (Info-bookmark-named-at-point)))
(and bmk (bmkp-bookmark-name-from-record bmk))))
(define-key Info-mode-map (kbd "C-h C-b") 'Info-describe-bookmark)
)
(when (fboundp 'advice-add) ; Emacs 24.4+
(defun Info-save-history-list ()
"Save `Info-history-list' to `Info-saved-history-file'."
(when (and Info-persist-history-mode
(not (string= "" Info-saved-history-file))
(file-writable-p Info-saved-history-file)
(not (file-directory-p Info-saved-history-file)))
(let* ((ibuf (catch 'Info-save-history-list
(dolist (buf (buffer-list))
(with-current-buffer buf
(when (derived-mode-p 'Info-mode) (throw 'Info-save-history-list buf))))
nil))
(hist (and ibuf (with-current-buffer ibuf Info-history-list))))
(with-temp-file Info-saved-history-file
(print Info-history-list (current-buffer))))))
(defun Info-restore-history-list ()
"Restore `Info-history-list' from `Info-saved-history-file'."
(when (and Info-persist-history-mode
(not (string= "" Info-saved-history-file))
(file-readable-p Info-saved-history-file))
(let ((buf (let ((enable-local-variables ()))
(find-file-noselect Info-saved-history-file)))
hist)
(unwind-protect
(with-current-buffer buf
(goto-char (point-min))
(setq hist (ignore-errors (read (current-buffer)))))
(kill-buffer buf))
(when hist
(setq Info-history-list hist)
(when Info-fontify-visited-nodes (Info-fontify-node))))))
)
(if (fboundp 'pop-to-buffer-same-window)
(defalias 'Info--pop-to-buffer-same-window 'pop-to-buffer-same-window)
(defalias 'Info--pop-to-buffer-same-window 'switch-to-buffer))
;;; ;; Not currently used.
;;; (defun Info-display-node-time-header ()
;;; "Insert current time and node name as header."
;;; ;; `infop-node-name' is free here - bound in `Info-merge-subnodes'.
;;; (insert (current-time-string) " " infop-node-name) ; FREE here: INFOP-NODE-NAME
;;; (beginning-of-buffer)
;;; (center-line))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; info+.el ends here