;; Info package for Emacs -- could use a "create node" feature. ;; Copyright (C) 1985 Richard M. Stallman. ;; This file is part of GNU Emacs. ;; GNU Emacs is distributed in the hope that it will be useful, ;; but without any warranty. No author or distributor ;; accepts responsibility to anyone for the consequences of using it ;; or for whether it serves any particular purpose or works at all, ;; unless he says so in writing. ;; Everyone is granted permission to copy, modify and redistribute ;; GNU Emacs, but only under the conditions described in the ;; document "GNU Emacs copying permission notice". An exact copy ;; of the document is supposed to have been given to you along with ;; GNU Emacs so that you can know how you may redistribute it all. ;; It should be in a file named COPYING. Among other things, the ;; copyright notice and this notice must be preserved on all copies. (defvar Info-history nil "List of info nodes user has visited. Each element of list is a list (FILENAME NODENAME BUFFERPOS).") (defvar Info-enable-edit nil "Non-nil means the C-r command in Info can edit the current node.") (defvar Info-enable-active-nodes t "Non-nil allows Info to execute Lisp code associated with nodes. The Lisp code is executed when the node is selected.") (defvar Info-directory nil "Default directory for Info documentation files.") (defvar Info-current-file nil "Info file that Info is now looking at, or nil.") (defvar Info-current-node nil "Name of node that Info is now looking at, or nil.") (defvar Info-tag-table-marker (make-marker) "Marker pointing at beginning of current Info file's tag table. Marker points nowhere if file has no tag table.") (defun info () "Create a buffer for Info, the documentation browser program." (interactive) (if (get-buffer "*info*") (switch-to-buffer "*info*") (Info-directory))) ;; Go to an info node specified as separate filename and nodename. (defun Info-find-node (filename nodename) ;; Convert filename to lower case if not found as specified. ;; Expand it. (if filename (let (temp) (setq temp (expand-file-name filename Info-directory)) (if (file-exists-p temp) (setq filename temp) (setq temp (expand-file-name (downcase filename) Info-directory)) (if (file-exists-p temp) (setq filename temp) (error (concat "Info file " filename " does not exist")))))) ;; Record the node we are leaving. (if Info-current-file (setq Info-history (cons (list Info-current-file Info-current-node (dot)) Info-history))) ;; Go into info buffer. (switch-to-buffer "*info*") (Info-mode) (widen) ;; Switch files if necessary (or (null filename) (equal buffer-file-name filename) (progn (erase-buffer) (insert-file-contents filename t) ;; See whether file has a tag table. Record the location if yes. (move-marker Info-tag-table-marker nil) (goto-char (dot-max)) (forward-line -8) (if (search-forward "\^_\nEnd tag table\n" nil t) (progn (search-backward "\nTag table:\n") (move-marker Info-tag-table-marker (match-end 0)))))) (if (equal nodename "*") (progn (setq Info-current-file buffer-file-name Info-current-node nodename) (Info-set-mode-line)) ;; Search file for a suitable node. ;; First get advice from tag table if file has one. (if (marker-position Info-tag-table-marker) (progn (goto-char Info-tag-table-marker) (if (search-forward (concat " " nodename "\177") nil t) (goto-char (max (dot-min) (- (read (current-buffer)) 1000))) (goto-char (dot-min)))) (goto-char (dot-min))) (let ((regexp (concat "Node: *" (regexp-quote nodename) " *[,\t\n]"))) (catch 'foo (while (search-forward "\n\^_" nil t) (forward-line 1) (let ((beg (dot))) (forward-line 1) (if (re-search-backward regexp beg t) (throw 'foo t)))) (error "No such node"))) (Info-select-node)) (goto-char (dot-min))) ;; Select the info node that dot is in. (defun Info-select-node () (save-excursion ;; Find beginning of node. (search-backward "\n\^_") (forward-line 2) ;; Get nodename spelled as it is in the node. (re-search-forward "Node: *") (setq Info-current-node (buffer-substring (dot) (progn (skip-chars-forward "^,\t\n") (dot)))) (setq Info-current-file buffer-file-name) (Info-set-mode-line) ;; Find the end of it, and narrow. (beginning-of-line) (let (active-expression) (narrow-to-region (dot) (if (re-search-forward "\n\^_\\|\n\f" nil t) (prog1 (1- (dot)) (if (search-forward "execute: " nil t) (setq active-expression (read (current-buffer))))) (dot-max))) (if Info-enable-active-nodes (eval active-expression))))) (defun Info-set-mode-line () (setq mode-line-format (concat "--%1*%1*-Info: (" (if buffer-file-name (file-name-nondirectory buffer-file-name) "") ")" (or Info-current-node "") " %M %[(%m)%]----%3p-%-"))) ;; Go to an info node specified with a filename-and-nodename string ;; of the sort that is found in pointers in nodes. (defun Info-goto-node (nodename) "Go to info node named NAME. Give just NODENAME or (FILENAME)NODENAME." (interactive "sGoto node: ") (let (filename temp) (setq temp (string-match "[^ ]" nodename)) (if temp (setq nodename (substring nodename temp)) (setq nodename "")) (while (and (> (length nodename) 0) (= (aref nodename (1- (length nodename))) ? )) (setq nodename (substring nodename 0 -1))) (if (= (aref nodename 0) ?\() (setq filename (substring nodename 1 (string-match ")" nodename)) nodename (substring nodename (1+ (string-match ")" nodename))))) (setq temp (string-match "[^ ]" nodename)) (if temp (setq nodename (substring nodename temp)) (setq nodename "")) (if filename (progn (setq temp (string-match "[^ ]" filename)) (if temp (setq filename (substring filename temp)) (setq filename "")) (while (and (> (length filename) 0) (= (aref filename (1- (length filename))) ? )) (setq filename (substring filename 0 -1))))) (if (equal nodename "") (setq nodename "Top")) (Info-find-node filename nodename))) (defun Info-tagify () "Create or update tag table of current info file." (interactive) ;; Save and restore dot and restrictions. ;; save-restrictions would not work ;; because it records the old max relative to the end. ;; We record it relative to the beginning. (let ((omin (dot-min)) (omax (dot-max)) (odot (dot))) (unwind-protect (progn (widen) (goto-char (dot-min)) (let ((regexp "Node:[^,\n\t]*[,\t\n]") list) (while (search-forward "\n\^_" nil t) (forward-line 1) (let ((beg (dot))) (forward-line 1) (if (re-search-backward regexp beg t) (setq list (cons (list (buffer-substring beg (1- (match-end 0))) beg) list))))) (goto-char (dot-max)) (forward-line -8) (if (search-forward "\^_\nEnd tag table\n" nil t) (let ((end (dot))) (search-backward "\nTag table:\n") (beginning-of-line) (delete-region (dot) end))) (goto-char (dot-max)) (insert "\^_\f\nTag table:\n") (move-marker Info-tag-table-marker (dot)) (while list (insert (car (car list)) ?\177) (princ (car (cdr (car list))) (current-buffer)) (insert ?\n) (setq list (cdr list))) (insert "\^_\nEnd tag table\n"))) (goto-char odot) (narrow-to-region omin omax)))) (defun Info-validate () "Check that every node pointer points to an existing node." (interactive) (save-excursion (save-restriction (widen) (goto-char (dot-min)) (let ((allnodes '(("*"))) (regexp "Node:[ \t]*\\([^,\n\t]*\\)[,\t\n]") lossages) (while (search-forward "\n\^_" nil t) (forward-line 1) (let ((beg (dot))) (forward-line 1) (if (re-search-backward regexp beg t) (setq allnodes (cons (cons (buffer-substring (match-beginning 1) (progn (goto-char (match-end 1)) (skip-chars-backward " \t") (dot))) (progn (end-of-line) (and (re-search-backward "prev[ious]*:" beg t) (progn (goto-char (match-end 0)) (Info-following-node-name))))) allnodes))))) (goto-char (dot-min)) (while (search-forward "\n\^_" nil t) (forward-line 1) (let ((beg (dot)) thisnode next) (forward-line 1) (if (re-search-backward regexp beg t) (save-restriction (search-forward "\n\^_" nil 'move) (narrow-to-region beg (dot)) (setq thisnode (buffer-substring (match-beginning 1) (progn (goto-char (match-end 1)) (skip-chars-backward " \t") (dot)))) (end-of-line) (and (search-backward "next:" nil t) (setq next (Info-validate-node-name "Next")) (or (equal (cdr (assoc next allnodes)) thisnode) (setq lossages (cons (list thisnode "Previous-pointer in Next" next) lossages)))) (end-of-line) (if (search-backward "prev[ious]*:" nil t) (Info-validate-node-name "Previous")) (end-of-line) (if (search-backward "up:" nil t) (Info-validate-node-name "Up")) (if (search-forward "\n* Menu:" nil t) (while (search-forward "\n* " nil t) (Info-validate-node-name (concat "menu item " (buffer-substring (dot) (save-excursion (skip-chars-forward "^:") (dot)))) (Info-extract-menu-node-name)))) (goto-char (dot-min)) (while (search-forward "*note " nil t) (Info-validate-node-name (concat "footnote " (buffer-substring (dot) (save-excursion (skip-chars-forward "^:") (dot)))) (Info-extract-menu-node-name "Bad format footnote"))))))) (if lossages (with-output-to-temp-buffer " *problems in info file*" (while lossages (princ "In node ") (princ (car (car lossages))) (princ ", invalid ") (princ (nth 1 (car lossages))) (princ ": ") (princ (nth 2 (car lossages))) (terpri) (setq lossages (cdr lossages)))) (message "File appears valid")))))) (defun Info-validate-node-name (kind &optional name) (if name nil (goto-char (match-end 0)) (skip-chars-forward " \t") (if (= (following-char) ?\() nil (setq name (buffer-substring (dot) (progn (skip-chars-forward "^,\t\n") (skip-chars-backward " ") (dot)))))) (or (null name) (assoc name allnodes) (setq lossages (cons (list thisnode kind name) lossages))) name) (defvar Info-last-search nil "Default regexp for Info S command to search for.") (defun Info-search (regexp) "Search for REGEXP, starting from dot, and select node it's found in." (interactive "sSearch (regexp): ") (if (equal regexp "") (setq regexp Info-last-search) (setq Info-last-search regexp)) (let (found) (save-excursion (save-restriction (widen) (re-search-forward regexp) (setq found (dot)))) (if found (progn (widen) (goto-char found) (Info-select-node))))) (defun Info-extract-pointer (name) (save-excursion (goto-char (dot-min)) (forward-line 1) (if (re-search-backward (concat name ":") nil t) nil (error (concat "Node has no " (capitalize name)))) (goto-char (match-end 0)) (Info-following-node-name))) (defun Info-following-node-name (&optional allowedchars) (skip-chars-forward " \t") (buffer-substring (dot) (progn (skip-chars-forward (or allowedchars "^,\t\n")) (skip-chars-backward " ") (dot)))) (defun Info-next () "Go to the next node of this node." (interactive) (Info-goto-node (Info-extract-pointer "next"))) (defun Info-prev () "Go to the previous node of this node." (interactive) (Info-goto-node (Info-extract-pointer "prev[ious]*"))) (defun Info-up () "Go to the superior node of this node." (interactive) (Info-goto-node (Info-extract-pointer "up"))) (defun Info-last () "Go back to the last node visited." (interactive) (or Info-history (error "This is the first Info node you looked at")) (let (filename nodename odot) (setq filename (car (car Info-history))) (setq nodename (car (cdr (car Info-history)))) (setq odot (car (cdr (cdr (car Info-history))))) (setq Info-history (cdr Info-history)) (Info-find-node filename nodename) (setq Info-history (cdr Info-history)) (goto-char odot))) (defun Info-directory () "Go to the Info directory node." (interactive) (Info-find-node "dir" "top")) (defun Info-footnote (footnotename) "Follow footnote pointer of footnote name (or abbreviation) NAME." (interactive "sFootnote name: ") (let (target beg) (save-excursion (goto-char (dot-min)) (or (search-forward (concat "*note " footnotename) nil t) (error "No such footnote")) (goto-char (+ (match-beginning 0) 6)) (setq target (Info-extract-menu-node-name "Bad format footnote"))) (Info-goto-node target))) (defun Info-extract-menu-node-name (&optional errmessage) (skip-chars-forward " \t") (let ((beg (dot))) (skip-chars-forward "^:\n") (or (looking-at ":") (error (or errmessage "Bad format menu item"))) (forward-char 1) (if (looking-at ":") (buffer-substring beg (1- (dot))) (Info-following-node-name "^.,\t\n")))) (defun Info-menu-item-seqence (list) (while list (Info-menu-item (car list)) (setq list (cdr list)))) (defun Info-menu (menu-item) "Go to node for menu item named (or abbreviated) NAME." (interactive "sMenu item: ") (Info-goto-node (Info-extract-menu-item menu-item))) (defun Info-extract-menu-item (menu-item) (save-excursion (goto-char (dot-min)) (or (search-forward "\n* menu:" nil t) (error "No menu in this node")) (or (search-forward (concat "\n* " menu-item) nil t) (error "No such item in menu")) (beginning-of-line) (forward-char 2) (Info-extract-menu-node-name))) (defun Info-extract-menu-counting (count) (save-excursion (goto-char (dot-min)) (or (search-forward "\n* menu:" nil t) (error "No menu in this node")) (or (search-forward "\n* " nil t count) (error "Too few items in menu")) (Info-extract-menu-node-name))) (defun Info-first-menu-item () "Go to the node of the first menu item." (interactive) (Info-goto-node (Info-extract-menu-counting 1))) (defun Info-second-menu-item () "Go to the node of the second menu item." (interactive) (Info-goto-node (Info-extract-menu-counting 2))) (defun Info-third-menu-item () "Go to the node of the third menu item." (interactive) (Info-goto-node (Info-extract-menu-counting 3))) (defun Info-fourth-menu-item () "Go to the node of the fourth menu item." (interactive) (Info-goto-node (Info-extract-menu-counting 4))) (defun Info-fifth-menu-item () "Go to the node of the fifth menu item." (interactive) (Info-goto-node (Info-extract-menu-counting 5))) (defun Info-exit () "Exit Info by selecting some other buffer." (interactive) (switch-to-buffer (other-buffer (current-buffer)))) (defun Info-undefined () "Make command be undefined in Info." (interactive) (ding)) (defun Info-help () "Enter the Info tutorial." (interactive) (Info-find-node "dir" (if (< (window-height) 23) "Help-Small-Screen" "Help"))) (defun Info-summary () "Display a brief summary of all Info commands." (interactive) (save-window-excursion (switch-to-buffer "*Help*") (erase-buffer) (insert (documentation 'Info-mod)) (goto-char (dot-min)) (setq unread-command-char (read-char)) (if (= unread-command-char ? ) (read-char)))) (defvar Info-mode-map nil "Keymap containing Info commands.") (setq Info-mode-map (make-keymap)) (suppress-keymap Info-mode-map) (define-key Info-mode-map " " 'scroll-up) (define-key Info-mode-map "1" 'Info-first-menu-item) (define-key Info-mode-map "2" 'Info-second-menu-item) (define-key Info-mode-map "3" 'Info-third-menu-item) (define-key Info-mode-map "4" 'Info-fourth-menu-item) (define-key Info-mode-map "5" 'Info-fifth-menu-item) (define-key Info-mode-map "6" 'undefined) (define-key Info-mode-map "7" 'undefined) (define-key Info-mode-map "8" 'undefined) (define-key Info-mode-map "9" 'undefined) (define-key Info-mode-map "0" 'undefined) (define-key Info-mode-map "?" 'Info-summary) (define-key Info-mode-map "b" 'beginning-of-buffer) (define-key Info-mode-map "e" 'Info-edit) (define-key Info-mode-map "f" 'Info-footnote) (define-key Info-mode-map "g" 'Info-goto-node) (define-key Info-mode-map "h" 'Info-help) (define-key Info-mode-map "l" 'Info-last) (define-key Info-mode-map "m" 'Info-menu) (define-key Info-mode-map "n" 'Info-next) (define-key Info-mode-map "p" 'Info-prev) (define-key Info-mode-map "q" 'Info-exit) (define-key Info-mode-map "s" 'Info-search) (define-key Info-mode-map "u" 'Info-up) (defun Info-mode () "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 references to other nodes which discuss related topics. Info has commands to follow the references and show you other nodes. h Invoke the Info tutorial. Selecting other nodes: n Move to the \"next\" node from the selected node. p Move to the \"previous\" node from the selected node. u Move \"up\" from this node. m Pick menu item specified by name (or abbreviation). Picking a menu item causes another node to be selected. f Move to text of a footnote. Give the footnote name (or abbreviation) as an argument. l Move to the last node you were at. Moving within a node: Space scroll forward a page. b Go to beginning of node. Advanced commands: q Quit Info: reselect previously selected buffer. e Edit contents of selected node. 1 Pick first item in node's menu. 2, 3, 4, 5 Pick second ... fifth item in node's menu. g Move to node specified by name. You may include a filename as well, as (FILENAME)NODENAME. s Search through this Info file for specified regexp, and select the node in which the next occurrence is found." (kill-all-local-variables) (setq major-mode 'Info-mode) (setq mode-name "Info") (use-local-map Info-mode-map) (set-syntax-table text-mode-syntax-table) (setq local-abbrev-table text-mode-abbrev-table) (setq case-fold-search t) (Info-set-mode-line)) (defvar Info-edit-map (append text-mode-map nil) "Local keymap used within `e' command of Info.") (define-key Info-edit-map "\^z" 'Info-cease-edit) (defun Info-edit-mode () "Major mode for editing the contents of an Info node. The editing commands are the same as in Text mode, except for C-z to return to Info." ) (defun Info-edit () "Edit the contents of this Info node. Allowed only if variable Info-enable-edit is non-nil." (interactive) (or Info-enable-edit (error "Editing info nodes is not enabled")) (use-local-map Info-edit-map) (setq major-mode 'Info-edit-mode) (setq mode-name "Info Edit") (setq mode-line-format default-mode-line-format) ;; Make mode line update. (set-buffer-modified-p (buffer-modified-p))) (defun Info-cease-edit () "Finish editing Info node; switch back to Info proper." (interactive) ;; Do this first, so nothing has changed if user C-g's at query. (and (buffer-modified-p) (y-or-n-p "Save the file? ") (save-buffer)) (use-local-map Info-mode-map) (setq major-mode 'Info-mode) (setq mode-name "Info") (Info-set-mode-line) ;; Make mode line update. (set-buffer-modified-p (buffer-modified-p)))