From 1a9461d069cbf51a16b453f283b1367e05c875b9 Mon Sep 17 00:00:00 2001 From: Carsten Dominik Date: Mon, 16 Aug 1999 07:42:41 +0000 Subject: [PATCH] Initial revision --- lisp/textmodes/reftex-auc.el | 190 ++++ lisp/textmodes/reftex-cite.el | 928 ++++++++++++++++++ lisp/textmodes/reftex-global.el | 295 ++++++ lisp/textmodes/reftex-index.el | 1219 ++++++++++++++++++++++++ lisp/textmodes/reftex-parse.el | 987 ++++++++++++++++++++ lisp/textmodes/reftex-ref.el | 776 ++++++++++++++++ lisp/textmodes/reftex-sel.el | 671 ++++++++++++++ lisp/textmodes/reftex-toc.el | 583 ++++++++++++ lisp/textmodes/reftex-vars.el | 1547 +++++++++++++++++++++++++++++++ lisp/textmodes/reftex-vcr.el | 452 +++++++++ 10 files changed, 7648 insertions(+) create mode 100644 lisp/textmodes/reftex-auc.el create mode 100644 lisp/textmodes/reftex-cite.el create mode 100644 lisp/textmodes/reftex-global.el create mode 100644 lisp/textmodes/reftex-index.el create mode 100644 lisp/textmodes/reftex-parse.el create mode 100644 lisp/textmodes/reftex-ref.el create mode 100644 lisp/textmodes/reftex-sel.el create mode 100644 lisp/textmodes/reftex-toc.el create mode 100644 lisp/textmodes/reftex-vars.el create mode 100644 lisp/textmodes/reftex-vcr.el diff --git a/lisp/textmodes/reftex-auc.el b/lisp/textmodes/reftex-auc.el new file mode 100644 index 00000000000..92a74658ab1 --- /dev/null +++ b/lisp/textmodes/reftex-auc.el @@ -0,0 +1,190 @@ +;;; reftex-auc.el - RefTeX's interface to AUC TeX +;;; Version: 4.5 +;;; +;;; See main file reftex.el for licensing information + +(provide 'reftex-auc) +(require 'reftex) +;;; + +(defun reftex-plug-flag (which) + ;; Tell if a certain flag is set in reftex-plug-into-AUCTeX + (or (eq t reftex-plug-into-AUCTeX) + (and (listp reftex-plug-into-AUCTeX) + (nth which reftex-plug-into-AUCTeX)))) + +(defun reftex-arg-label (optional &optional prompt definition) + "Use `reftex-label', `reftex-reference' or AUCTeX's code to insert label arg. +What is being used depends upon `reftex-plug-into-AUCTeX'." + (let (label) + (cond + ((and definition (reftex-plug-flag 1)) + ;; Create a new label, with a temporary brace for `reftex-what-macro' + (unwind-protect + (progn (insert "{") (setq label (or (reftex-label nil t) ""))) + (delete-backward-char 1))) + ((and (not definition) (reftex-plug-flag 2)) + ;; Reference a label with RefTeX + (setq label (reftex-reference nil t))) + (t + ;; AUCTeX's default mechanism + (setq label (completing-read (TeX-argument-prompt optional prompt "Key") + (LaTeX-label-list))))) + (if (and definition (not (string-equal "" label))) + (LaTeX-add-labels label)) + (TeX-argument-insert label optional))) + +(defun reftex-arg-cite (optional &optional prompt definition) + "Use `reftex-citation' or AUCTeX's code to insert a cite-key macro argument. +What is being used depends upon `reftex-plug-into-AUCTeX'." + (let (items) + (cond + ((and (not definition) (reftex-plug-flag 3)) + (setq items (list (or (reftex-citation t) "")))) + (t + (setq prompt (concat (if optional "(Optional) " "") + (if prompt prompt "Add key") + ": (default none) ")) + (setq items (multi-prompt "," t prompt (LaTeX-bibitem-list))))) + (apply 'LaTeX-add-bibitems items) + (TeX-argument-insert (mapconcat 'identity items ",") optional))) + + +(defun reftex-arg-index-tag (optional &optional prompt &rest args) + "Prompt for an index tag with completion. +This is the name of an index, not the entry." + (let (tag taglist) + (setq prompt (concat (if optional "(Optional) " "") + (if prompt prompt "Index tag") + ": (default none) ")) + (if (and reftex-support-index (reftex-plug-flag 4)) + ;; Use RefTeX completion + (progn + (reftex-access-scan-info nil) + (setq taglist + (cdr (assoc 'index-tags + (symbol-value reftex-docstruct-symbol))) + tag (completing-read prompt (mapcar 'list taglist)))) + ;; Just ask like AUCTeX does. + (setq tag (read-string prompt))) + (TeX-argument-insert tag optional))) + +(defun reftex-arg-index (optional &optional prompt &rest args) + "Prompt for an index entry completing with known entries. +Completion is specific for just one index, if the macro or a tag +argument identify one of multiple indices." + (let* (tag key) + (if (and reftex-support-index (reftex-plug-flag 4)) + (progn + (reftex-access-scan-info nil) + (setq tag (reftex-what-index-tag) + key (reftex-index-complete-key (or tag "idx")))) + (setq key (completing-read (TeX-argument-prompt optional prompt "Key") + (LaTeX-index-entry-list)))) + (unless (string-equal "" key) + (LaTeX-add-index-entries key)) + (TeX-argument-insert key optional))) + +(defun reftex-what-index-tag () + ;; Look backward to find out what index the macro at point belongs to + (let ((macro (save-excursion + (and (re-search-backward "\\\\[a-zA-Z*]+" nil t) + (match-string 0)))) + tag entry) + (when (and macro + (setq entry (assoc macro reftex-index-macro-alist))) + (setq tag (nth 1 entry)) + (cond + ((stringp tag) tag) + ((integerp tag) + (save-excursion + (goto-char (match-end 1)) + (or (reftex-nth-arg tag (nth 6 entry)) "idx"))) + (t "idx"))))) + +(defvar LaTeX-label-function) +(defun reftex-plug-into-AUCTeX () + ;; Replace AUCTeX functions with RefTeX functions. + ;; Which functions are replaced is controlled by the variable + ;; `reftex-plug-into-AUCTeX'. + + (if (reftex-plug-flag 0) + (setq LaTeX-label-function 'reftex-label) + (setq LaTeX-label-function nil)) + + (and (or (reftex-plug-flag 1) (reftex-plug-flag 2)) + (fboundp 'TeX-arg-label) + (fset 'TeX-arg-label 'reftex-arg-label)) + + (and (reftex-plug-flag 3) + (fboundp 'TeX-arg-cite) + (fset 'TeX-arg-cite 'reftex-arg-cite)) + + (and (reftex-plug-flag 4) + (fboundp 'TeX-arg-index-tag) + (fset 'TeX-arg-index-tag 'reftex-arg-index-tag)) + (and (reftex-plug-flag 4) + (fboundp 'TeX-arg-index) + (fset 'TeX-arg-index 'reftex-arg-index))) + +(defun reftex-toggle-plug-into-AUCTeX () + "Toggle Interface between AUCTeX and RefTeX on and off." + (interactive) + (unless (and (featurep 'tex-site) (featurep 'latex)) + (error "AUCTeX's LaTeX mode does not seem to be loaded")) + (setq reftex-plug-into-AUCTeX (not reftex-plug-into-AUCTeX)) + (reftex-plug-into-AUCTeX) + (if reftex-plug-into-AUCTeX + (message "RefTeX has been plugged into AUCTeX.") + (message "RefTeX no longer interacts with AUCTeX."))) + +(defun reftex-add-label-environments (entry-list) + "Add label environment descriptions to `reftex-label-alist-style'. +The format of ENTRY-LIST is exactly like `reftex-label-alist'. See there +for details. +This function makes it possible to support RefTeX from AUCTeX style files. +The entries in ENTRY-LIST will be processed after the user settings in +`reftex-label-alist', and before the defaults (specified in +`reftex-default-label-alist-entries'). Any changes made to +`reftex-label-alist-style' will raise a flag to the effect that +the label information is recompiled on next use." + (unless reftex-docstruct-symbol + (reftex-tie-multifile-symbols)) + (when (and reftex-docstruct-symbol + (symbolp reftex-docstruct-symbol)) + (let ((list (get reftex-docstruct-symbol 'reftex-label-alist-style)) + entry changed) + (while entry-list + (setq entry (pop entry-list)) + (unless (member entry list) + (setq reftex-tables-dirty t + changed t) + (push entry list))) + (when changed + (put reftex-docstruct-symbol 'reftex-label-alist-style list))))) +(defalias 'reftex-add-to-label-alist 'reftex-add-label-environments) + +(defun reftex-add-section-levels (entry-list) + "Add entries to the value of `reftex-section-levels'. +The added values are kept local to the current document. The format +of ENTRY-LIST is a list of cons cells (\"MACRONAME\" . LEVEL). See +`reftex-section-levels' for an example." + (unless reftex-docstruct-symbol + (reftex-tie-multifile-symbols)) + (when (and reftex-docstruct-symbol + (symbolp reftex-docstruct-symbol)) + (let ((list (get reftex-docstruct-symbol 'reftex-section-levels)) + entry changed) + (while entry-list + (setq entry (pop entry-list)) + (unless (member entry list) + (setq reftex-tables-dirty t + changed t) + (push entry list))) + (when changed + (put reftex-docstruct-symbol 'reftex-section-levels list))))) + +(defun reftex-notice-new-section () + (reftex-notice-new 1 'force)) + +;;; reftex-auc.el ends here diff --git a/lisp/textmodes/reftex-cite.el b/lisp/textmodes/reftex-cite.el new file mode 100644 index 00000000000..afe000785e3 --- /dev/null +++ b/lisp/textmodes/reftex-cite.el @@ -0,0 +1,928 @@ +;;; reftex-cite.el - Creating citations with RefTeX +;;; Version: 4.5 +;;; +;;; See main file reftex.el for licensing information + +(provide 'reftex-cite) +(require 'reftex) +;;; + +;; Variables and constants + +;; The history list of regular expressions used for citations +(defvar reftex-cite-regexp-hist nil) + +;; Prompt and help string for citation selection +(defconst reftex-citation-prompt + "Select: [n]ext [p]revious [r]estrict [ ]full_entry [q]uit RET [?]Help+more") + +(defconst reftex-citation-help + " n / p Go to next/previous entry (Cursor motion works as well). + g / r Start over with new regexp / Refine with additional regexp. + SPC Show full database entry in other window. + f Toggle follow mode: Other window will follow with full db entry. + . Show insertion point. + q Quit without inserting \\cite macro into buffer. + TAB Enter citation key with completion. + RET Accept current entry (also on mouse-2) and create \\cite macro. + m / u Mark/Unmark the entry. + a / A Put all (marked) entries into one/many \\cite commands.") + +;; Find bibtex files + +(defun reftex-default-bibliography () + ;; Return the expanded value of `reftex-default-bibliography'. + ;; The expanded value is cached. + (unless (eq (get 'reftex-default-bibliography :reftex-raw) + reftex-default-bibliography) + (put 'reftex-default-bibliography :reftex-expanded + (reftex-locate-bibliography-files + default-directory reftex-default-bibliography)) + (put 'reftex-default-bibliography :reftex-raw + reftex-default-bibliography)) + (get 'reftex-default-bibliography :reftex-expanded)) + +(defun reftex-get-bibfile-list () + ;; Return list of bibfiles for current document. + ;; When using the chapterbib or bibunits package you should either + ;; use the same database files everywhere, or separate parts using + ;; different databases into different files (included into the mater file). + ;; Then this function will return the applicable database files. + + ;; Ensure access to scanning info + (reftex-access-scan-info) + (or + ;; Try inside this file (and its includes) + (cdr (reftex-last-assoc-before-elt + 'bib (list 'eof (buffer-file-name)) + (member (list 'bof (buffer-file-name)) + (symbol-value reftex-docstruct-symbol)))) + ;; Try after the beginning of this file + (cdr (assq 'bib (member (list 'bof (buffer-file-name)) + (symbol-value reftex-docstruct-symbol)))) + ;; Anywhere in the entire document + (cdr (assq 'bib (symbol-value reftex-docstruct-symbol))) + (error "\\bibliography statement missing or .bib files not found"))) + +;; Find a certain reference in any of the BibTeX files. + +(defun reftex-pop-to-bibtex-entry (key file-list &optional mark-to-kill + highlight item return) + ;; Find BibTeX KEY in any file in FILE-LIST in another window. + ;; If MARK-TO-KILL is non-nil, mark new buffer to kill. + ;; If HIGHLIGHT is non-nil, highlight the match. + ;; If ITEM in non-nil, search for bibitem instead of database entry. + ;; If RETURN is non-nil, just return the entry. + + (let* ((re + (if item + (concat "\\\\bibitem\\(\\[[^]]*\\]\\)?{" (regexp-quote key) "}") + (concat "@[a-zA-Z]+[ \t\n\r]*[{(][ \t\n\r]*" (regexp-quote key) + "[, \t\r\n}]"))) + (buffer-conf (current-buffer)) + file buf) + + (catch 'exit + (while file-list + (setq file (car file-list) + file-list (cdr file-list)) + (unless (setq buf (reftex-get-file-buffer-force file mark-to-kill)) + (error "No such file %s" file)) + (set-buffer buf) + (widen) + (goto-char (point-min)) + (when (re-search-forward re nil t) + (goto-char (match-beginning 0)) + (when return + ;; Just return the relevant entry + (if item (goto-char (match-end 0))) + (setq return (buffer-substring + (point) (reftex-end-of-bib-entry item))) + (set-buffer buffer-conf) + (throw 'exit return)) + (switch-to-buffer-other-window buf) + (recenter 0) + (if highlight + (reftex-highlight 0 (match-beginning 0) (match-end 0))) + (throw 'exit (selected-window)))) + (set-buffer buffer-conf) + (if item + (error "No \\bibitem with citation key %s" key) + (error "No BibTeX entry with citation key %s" key))))) + +(defun reftex-end-of-bib-entry (item) + (save-excursion + (condition-case nil + (if item + (progn (end-of-line) + (re-search-forward + "\\\\bibitem\\|\\end{thebibliography}") + (1- (match-beginning 0))) + (progn (forward-list 1) (point))) + (error (min (point-max) (+ 300 (point))))))) + +;; Parse bibtex buffers + +(defun reftex-extract-bib-entries (buffers) + ;; Extract bib entries which match regexps from BUFFERS. + ;; BUFFERS is a list of buffers or file names. + ;; Return list with entries." + (let* (re-list first-re rest-re + (buffer-list (if (listp buffers) buffers (list buffers))) + found-list entry buffer1 buffer alist + key-point start-point end-point) + + ;; Read a regexp, completing on known citation keys. + (setq re-list + (split-string + (completing-read + "RegExp [ && RegExp...]: " + (if reftex-mode + (if (fboundp 'LaTeX-bibitem-list) + (LaTeX-bibitem-list) + (cdr (assoc 'bibview-cache + (symbol-value reftex-docstruct-symbol)))) + nil) + nil nil nil 'reftex-cite-regexp-hist) + "[ \t]*&&[ \t]*")) + + (setq first-re (car re-list) ; We'll use the first re to find things, + rest-re (cdr re-list)) ; the others to narrow down. + (if (string-match "\\`[ \t]*\\'" (or first-re "")) + (error "Empty regular expression")) + + (save-excursion + (save-window-excursion + + ;; Walk through all bibtex files + (while buffer-list + (setq buffer (car buffer-list) + buffer-list (cdr buffer-list)) + (if (and (bufferp buffer) + (buffer-live-p buffer)) + (setq buffer1 buffer) + (setq buffer1 (reftex-get-file-buffer-force + buffer (not reftex-keep-temporary-buffers)))) + (if (not buffer1) + (message "No such BibTeX file %s (ignored)" buffer) + (message "Scanning bibliography database %s" buffer1)) + + (set-buffer buffer1) + (save-excursion + (goto-char (point-min)) + (while (re-search-forward first-re nil t) + (catch 'search-again + (setq key-point (point)) + (unless (re-search-backward + "\\(\\`\\|[\n\r]\\)[ \t]*@\\([a-zA-Z]+\\)[ \t\n\r]*[{(]" nil t) + (throw 'search-again nil)) + (setq start-point (point)) + (goto-char (match-end 0)) + (condition-case nil + (up-list 1) + (error (goto-char key-point) + (throw 'search-again nil))) + (setq end-point (point)) + + ;; Ignore @string, @comment and @c entries or things + ;; outside entries + (when (or (string= (downcase (match-string 2)) "string") + (string= (downcase (match-string 2)) "comment") + (string= (downcase (match-string 2)) "c") + (< (point) key-point)) ; this means match not in {} + (goto-char key-point) + (throw 'search-again nil)) + + ;; Well, we have got a match + (setq entry (concat + (buffer-substring start-point (point)) "\n")) + + ;; Check if other regexp match as well + (setq re-list rest-re) + (while re-list + (unless (string-match (car re-list) entry) + ;; nope - move on + (throw 'search-again nil)) + (pop re-list)) + + (setq alist (reftex-parse-bibtex-entry + nil start-point end-point)) + (push (cons "&entry" entry) alist) + + ;; check for crossref entries + (if (assoc "crossref" alist) + (setq alist + (append + alist (reftex-get-crossref-alist alist)))) + + ;; format the entry + (push (cons "&formatted" (reftex-format-bib-entry alist)) + alist) + + ;; make key the first element + (push (reftex-get-bib-field "&key" alist) alist) + + ;; add it to the list + (push alist found-list)))) + (reftex-kill-temporary-buffers)))) + (setq found-list (nreverse found-list)) + + ;; Sorting + (cond + ((eq 'author reftex-sort-bibtex-matches) + (sort found-list 'reftex-bib-sort-author)) + ((eq 'year reftex-sort-bibtex-matches) + (sort found-list 'reftex-bib-sort-year)) + ((eq 'reverse-year reftex-sort-bibtex-matches) + (sort found-list 'reftex-bib-sort-year-reverse)) + (t found-list)))) + +(defun reftex-bib-sort-author (e1 e2) + (let ((al1 (reftex-get-bib-names "author" e1)) + (al2 (reftex-get-bib-names "author" e2))) + (while (and al1 al2 (string= (car al1) (car al2))) + (pop al1) + (pop al2)) + (if (and (stringp (car al1)) + (stringp (car al2))) + (string< (car al1) (car al2)) + (not (stringp (car al1)))))) + +(defun reftex-bib-sort-year (e1 e2) + (< (string-to-int (cdr (assoc "year" e1))) + (string-to-int (cdr (assoc "year" e2))))) + +(defun reftex-bib-sort-year-reverse (e1 e2) + (> (string-to-int (or (cdr (assoc "year" e1)) "0")) + (string-to-int (or (cdr (assoc "year" e2)) "0")))) + +(defun reftex-get-crossref-alist (entry) + ;; return the alist from a crossref entry + (let ((crkey (cdr (assoc "crossref" entry))) + start) + (save-excursion + (save-restriction + (widen) + (if (re-search-forward + (concat "@\\w+[{(][ \t\n\r]*" (regexp-quote crkey) + "[ \t\n\r]*,") nil t) + (progn + (setq start (match-beginning 0)) + (condition-case nil + (up-list 1) + (error nil)) + (reftex-parse-bibtex-entry nil start (point))) + nil))))) + +;; Parse the thebibliography environment +(defun reftex-extract-bib-entries-from-thebibliography (file) + ;; Extract bib-entries from the \begin{thebibliography} environment. + ;; Parsing is not as good as for the BibTeX database stuff. + ;; The environment should be located in file FILE. + + (let* (start end buf entries re re-list) + (unless file + (error "Need file name to find thebibliography environment")) + (setq buf (reftex-get-file-buffer-force + file (not reftex-keep-temporary-buffers))) + (unless buf + (error "No such file %s" file)) + (message "Scanning thebibliography environment in %s" file) + + (save-excursion + (set-buffer buf) + (save-restriction + (widen) + (goto-char (point-min)) + (if (re-search-forward + "\\(\\`\\|[\n\r]\\)[ \t]*\\\\begin{thebibliography}" nil t) + (progn + (beginning-of-line 2) + (setq start (point)))) + (if (re-search-forward + "\\(\\`\\|[\n\r]\\)[ \t]*\\\\end{thebibliography}" nil t) + (progn + (beginning-of-line 1) + (setq end (point)))) + (when (and start end) + (setq entries + (mapcar 'reftex-parse-bibitem + (delete "" + (split-string + (buffer-substring-no-properties start end) + "[ \t\n\r]*\\\\bibitem\\(\\[[^]]*]\\)*"))))))) + (unless entries + (error "No bibitems found")) + + (setq re-list (split-string + (read-string "RegExp [ && RegExp...]: " + nil 'reftex-cite-regexp-hist) + "[ \t]*&&[ \t]*")) + (if (string-match "\\`[ \t]*\\'" (car re-list)) + (error "Empty regular expression")) + + (while (and (setq re (pop re-list)) entries) + (setq entries + (delq nil (mapcar + (lambda (x) + (if (string-match re (cdr (assoc "&entry" x))) + x nil)) + entries)))) + (setq entries + (mapcar + (lambda (x) + (push (cons "&formatted" (reftex-format-bibitem x)) x) + (push (reftex-get-bib-field "&key" x) x) + x) + entries)) + + entries)) + +;; Parse and format individual entries + +(defun reftex-get-bib-names (field entry) + ;; Return a list with the author or editor names in ENTRY + (let ((names (reftex-get-bib-field field entry))) + (if (equal "" names) + (setq names (reftex-get-bib-field "editor" entry))) + (while (string-match "\\band\\b[ \t]*" names) + (setq names (replace-match "\n" nil t names))) + (while (string-match "[\\.a-zA-Z\\-]+\\.[ \t]*\\|,.*\\|[{}]+" names) + (setq names (replace-match "" nil t names))) + (while (string-match "^[ \t]+\\|[ \t]+$" names) + (setq names (replace-match "" nil t names))) + (while (string-match "[ \t][ \t]+" names) + (setq names (replace-match " " nil t names))) + (split-string names "\n"))) + +(defun reftex-parse-bibtex-entry (entry &optional from to) + (let (alist key start field) + (save-excursion + (save-restriction + (if entry + (progn + (set-buffer (get-buffer-create " *RefTeX-scratch*")) + (fundamental-mode) + (erase-buffer) + (insert entry)) + (widen) + (narrow-to-region from to)) + (goto-char (point-min)) + + (if (re-search-forward + "@\\(\\w+\\)[ \t\n\r]*[{(][ \t\n\r]*\\([^ \t\n\r,]+\\)" nil t) + (setq alist + (list + (cons "&type" (downcase (reftex-match-string 1))) + (cons "&key" (reftex-match-string 2))))) + (while (re-search-forward "\\(\\w+\\)[ \t\n\r]*=[ \t\n\r]*" nil t) + (setq key (downcase (reftex-match-string 1))) + (cond + ((= (following-char) ?{) + (forward-char 1) + (setq start (point)) + (condition-case nil + (up-list 1) + (error nil))) + ((= (following-char) ?\") + (forward-char 1) + (setq start (point)) + (while (and (search-forward "\"" nil t) + (= ?\\ (char-after (- (point) 2)))))) + (t + (setq start (point)) + (re-search-forward "[ \t]*[\n\r,}]" nil 1))) + (setq field (buffer-substring-no-properties start (1- (point)))) + ;; remove extra whitespace + (while (string-match "[\n\t\r]\\|[ \t][ \t]+" field) + (setq field (replace-match " " nil t field))) + ;; remove leading garbage + (if (string-match "^[ \t{]+" field) + (setq field (replace-match "" nil t field))) + ;; remove trailing garbage + (if (string-match "[ \t}]+$" field) + (setq field (replace-match "" nil t field))) + (push (cons key field) alist)))) + alist)) + +(defun reftex-get-bib-field (fieldname entry &optional format) + ;; Extract the field FIELDNAME from an ENTRY + (let ((cell (assoc fieldname entry))) + (if cell + (if format + (format format (cdr cell)) + (cdr cell)) + ""))) + +(defun reftex-format-bib-entry (entry) + ;; Format a BibTeX ENTRY so that it is nice to look at + (let* + ((auth-list (reftex-get-bib-names "author" entry)) + (authors (mapconcat 'identity auth-list ", ")) + (year (reftex-get-bib-field "year" entry)) + (title (reftex-get-bib-field "title" entry)) + (type (reftex-get-bib-field "&type" entry)) + (key (reftex-get-bib-field "&key" entry)) + (extra + (cond + ((equal type "article") + (concat (reftex-get-bib-field "journal" entry) " " + (reftex-get-bib-field "volume" entry) ", " + (reftex-get-bib-field "pages" entry))) + ((equal type "book") + (concat "book (" (reftex-get-bib-field "publisher" entry) ")")) + ((equal type "phdthesis") + (concat "PhD: " (reftex-get-bib-field "school" entry))) + ((equal type "mastersthesis") + (concat "Master: " (reftex-get-bib-field "school" entry))) + ((equal type "inbook") + (concat "Chap: " (reftex-get-bib-field "chapter" entry) + ", pp. " (reftex-get-bib-field "pages" entry))) + ((or (equal type "conference") + (equal type "incollection") + (equal type "inproceedings")) + (reftex-get-bib-field "booktitle" entry "in: %s")) + (t "")))) + (setq authors (reftex-truncate authors 30 t t)) + (when (reftex-use-fonts) + (put-text-property 0 (length key) 'face + (reftex-verified-face reftex-label-face + 'font-lock-constant-face + 'font-lock-reference-face) + key) + (put-text-property 0 (length authors) 'face reftex-bib-author-face + authors) + (put-text-property 0 (length year) 'face reftex-bib-year-face + year) + (put-text-property 0 (length title) 'face reftex-bib-title-face + title) + (put-text-property 0 (length extra) 'face reftex-bib-extra-face + extra)) + (concat key "\n " authors " " year " " extra "\n " title "\n\n"))) + +(defun reftex-parse-bibitem (item) + ;; Parse a \bibitem entry + (let ((key "") (text "")) + (when (string-match "\\`{\\([^}]+\\)}\\([\001-\255]*\\)" item) + (setq key (match-string 1 item) + text (match-string 2 item))) + ;; Clean up the text a little bit + (while (string-match "[\n\r\t]\\|[ \t][ \t]+" text) + (setq text (replace-match " " nil t text))) + (if (string-match "\\`[ \t]+" text) + (setq text (replace-match "" nil t text))) + (list + (cons "&key" key) + (cons "&text" text) + (cons "&entry" (concat key " " text))))) + +(defun reftex-format-bibitem (item) + ;; Format a \bibitem entry so that it is (relatively) nice to look at. + (let ((text (reftex-get-bib-field "&text" item)) + (key (reftex-get-bib-field "&key" item)) + (lines nil)) + + ;; Wrap the text into several lines. + (while (and (> (length text) 70) + (string-match " " (substring text 60))) + (push (substring text 0 (+ 60 (match-beginning 0))) lines) + (setq text (substring text (+ 61 (match-beginning 0))))) + (push text lines) + (setq text (mapconcat 'identity (nreverse lines) "\n ")) + + (when (reftex-use-fonts) + (put-text-property 0 (length text) 'face reftex-bib-author-face text)) + (concat key "\n " text "\n\n"))) + +;; Make a citation + +;;;###autoload +(defun reftex-citation (&optional no-insert) + "Make a citation using BibTeX database files. +After prompting for a regular expression, scans the buffers with +bibtex entries (taken from the \\bibliography command) and offers the +matching entries for selection. The selected entry is formated according +to `reftex-cite-format' and inserted into the buffer. + +If NO-INSERT is non-nil, nothing is inserted, only the selected key returned. + +When called with one or two `C-u' prefixes, first rescans the document. +When called with a numeric prefix, make that many citations. When +called with point inside the braces of a `\cite' command, it will +add another key, ignoring the value of `reftex-cite-format'. + +The regular expression uses an expanded syntax: && is interpreted as `and'. +Thus, `aaaa&&bbb' matches entries which contain both `aaaa' and `bbb'. +While entering the regexp, completion on knows citation keys is possible. +`=' is a good regular expression to match all entries in all files." + + (interactive) + + ;; check for recursive edit + (reftex-check-recursive-edit) + + ;; This function may also be called outside reftex-mode. + ;; Thus look for the scanning info only if in reftex-mode. + + (when reftex-mode + (reftex-access-scan-info current-prefix-arg)) + + ;; Call reftex-do-citation, but protected + (unwind-protect + (reftex-do-citation current-prefix-arg no-insert) + (reftex-kill-temporary-buffers))) + +(defun reftex-do-citation (&optional arg no-insert) + ;; This really does the work of reftex-citation. + + (let* ((format (reftex-figure-out-cite-format arg no-insert)) + (docstruct-symbol reftex-docstruct-symbol) + (selected-entries (reftex-offer-bib-menu)) + (insert-entries selected-entries) + entry string cite-view) + + (unless selected-entries (error "Quit")) + + (if (stringp selected-entries) + ;; Nonexistent entry + (setq selected-entries nil + insert-entries (list (list selected-entries + (cons "&key" selected-entries)))) + ;; It makes sense to compute the cite-view strings. + (setq cite-view t)) + + (when (eq (car selected-entries) 'concat) + ;; All keys go into a single command - we need to trick a little + (pop selected-entries) + (let ((concat-keys (mapconcat 'car selected-entries ","))) + (setq insert-entries + (list (list concat-keys (cons "&key" concat-keys)))))) + + (unless no-insert + + ;; We shall insert this into the buffer... + (message "Formatting...") + + (while (setq entry (pop insert-entries)) + ;; Format the citation and insert it + (setq string (if reftex-format-cite-function + (funcall reftex-format-cite-function + (reftex-get-bib-field "&key" entry) + format) + (reftex-format-citation entry format))) + (insert string)) + + ;; Reposition cursor? + (when (string-match "\\?" string) + (search-backward "?") + (delete-char 1)) + + ;; Tell AUCTeX + (when (and reftex-mode + (fboundp 'LaTeX-add-bibitems) + reftex-plug-into-AUCTeX) + (apply 'LaTeX-add-bibitems (mapcar 'car selected-entries))) + + ;; Produce the cite-view strings + (when (and reftex-mode reftex-cache-cite-echo cite-view) + (mapcar (lambda (entry) + (reftex-make-cite-echo-string entry docstruct-symbol)) + selected-entries)) + + (message "")) + + (set-marker reftex-select-return-marker nil) + (reftex-kill-buffer "*RefTeX Select*") + + ;; Check if the prefix arg was numeric, and call recursively + (when (integerp arg) + (if (> arg 1) + (progn + (skip-chars-backward "}") + (decf arg) + (reftex-do-citation arg)) + (forward-char 1))) + + ;; Return the citation key + (car (car selected-entries)))) + +(defun reftex-figure-out-cite-format (arg no-insert) + ;; Check if there is already a cite command at point and change cite format + ;; in order to only add another reference in the same cite command. + (let ((macro (car (reftex-what-macro 1))) + (cite-format-value (reftex-get-cite-format)) + key format) + (cond + (no-insert + ;; Format does not really matter because nothing will be inserted. + (setq format "%l")) + + ((and (stringp macro) + (string-match "\\`\\\\cite\\|cite\\'" macro)) + ;; We are already inside a cite macro + (if (or (not arg) (not (listp arg))) + (setq format + (concat + (if (member (preceding-char) '(?\{ ?,)) "" ",") + "%l" + (if (member (following-char) '(?\} ?,)) "" ","))) + (setq format "%l"))) + (t + ;; Figure out the correct format + (setq format + (if (and (symbolp cite-format-value) + (assq cite-format-value reftex-cite-format-builtin)) + (nth 2 (assq cite-format-value reftex-cite-format-builtin)) + cite-format-value)) + (when (listp format) + (setq key + (reftex-select-with-char + "" (concat "SELECT A CITATION FORMAT\n\n" + (mapconcat + (lambda (x) + (format "[%c] %s %s" (car x) + (if (> (car x) 31) " " "") + (cdr x))) + format "\n")))) + (if (assq key format) + (setq format (cdr (assq key format))) + (error "No citation format associated with key `%c'" key))))) + format)) + +(defvar reftex-select-bib-map) +(defun reftex-offer-bib-menu () + ;; Offer bib menu and return list of selected items + + (let (found-list rtn key data selected-entries) + (while + (not + (catch 'done + ;; Scan bibtex files + (setq found-list + (cond + ((assq 'bib (symbol-value reftex-docstruct-symbol)) + ;; using BibTeX database files. + (reftex-extract-bib-entries (reftex-get-bibfile-list))) + ((assq 'thebib (symbol-value reftex-docstruct-symbol)) + ;; using thebibliography environment. + (reftex-extract-bib-entries-from-thebibliography + (cdr (assq 'thebib (symbol-value reftex-docstruct-symbol))))) + (reftex-default-bibliography + (message "Using default bibliography") + (reftex-extract-bib-entries (reftex-default-bibliography))) + (t (error "No valid bibliography in this document, and no default available")))) + + (unless found-list + (error "Sorry, no matches found")) + + ;; Remember where we came from + (setq reftex-call-back-to-this-buffer (current-buffer)) + (set-marker reftex-select-return-marker (point)) + + ;; Offer selection + (save-window-excursion + (delete-other-windows) + (let ((default-major-mode 'reftex-select-bib-mode)) + (reftex-kill-buffer "*RefTeX Select*") + (switch-to-buffer-other-window "*RefTeX Select*") + (unless (eq major-mode 'reftex-select-bib-mode) + (reftex-select-bib-mode)) + (let ((buffer-read-only nil)) + (erase-buffer) + (reftex-insert-bib-matches found-list))) + (setq buffer-read-only t) + (if (= 0 (buffer-size)) + (error "No matches found")) + (setq truncate-lines t) + (goto-char 1) + (while t + (setq rtn + (reftex-select-item + reftex-citation-prompt + reftex-citation-help + reftex-select-bib-map + nil + 'reftex-bibtex-selection-callback nil)) + (setq key (car rtn) + data (nth 1 rtn)) + (unless key (throw 'done t)) + (cond + ((eq key ?g) + ;; Start over + (throw 'done nil)) + ((eq key ?r) + ;; Restrict with new regular expression + (setq found-list (reftex-restrict-bib-matches found-list)) + (let ((buffer-read-only nil)) + (erase-buffer) + (reftex-insert-bib-matches found-list)) + (goto-char 1)) + ((eq key ?A) + ;; Take all (marked) + (setq selected-entries + (if reftex-select-marked + (mapcar 'car (nreverse reftex-select-marked)) + found-list)) + (throw 'done t)) + ((eq key ?a) + ;; Take all (marked), and push the symbol 'concat + (setq selected-entries + (cons 'concat + (if reftex-select-marked + (mapcar 'car (nreverse reftex-select-marked)) + found-list))) + (throw 'done t)) + ((or (eq key ?\C-m) + (eq key 'return)) + ;; Take selected + (setq selected-entries + (if reftex-select-marked + (cons 'concat + (mapcar 'car (nreverse reftex-select-marked))) + (if data (list data) nil))) + (throw 'done t)) + ((stringp key) + ;; Got this one with completion + (setq selected-entries key) + (throw 'done t)) + (t + (ding)))))))) + selected-entries)) + +(defun reftex-restrict-bib-matches (found-list) + ;; Limit FOUND-LIST with more regular expressions + (let ((re-list (split-string (read-string + "RegExp [ && RegExp...]: " + nil 'reftex-cite-regexp-hist) + "[ \t]*&&[ \t]*")) + (found-list-r found-list) + re) + (while (setq re (pop re-list)) + (setq found-list-r + (delq nil + (mapcar + (lambda (x) + (if (string-match + re (cdr (assoc "&entry" x))) + x + nil)) + found-list-r)))) + (if found-list-r + found-list-r + (ding) + found-list))) + +(defun reftex-insert-bib-matches (list) + ;; Insert the bib matches and number them correctly + (let ((mouse-face + (if (memq reftex-highlight-selection '(mouse both)) + reftex-mouse-selected-face + nil)) + tmp len) + (mapcar + (lambda (x) + (setq tmp (cdr (assoc "&formatted" x)) + len (length tmp)) + (put-text-property 0 len :data x tmp) + (put-text-property 0 (1- len) 'mouse-face mouse-face tmp) + (insert tmp)) + list)) + (run-hooks 'reftex-display-copied-context-hook)) + +(defun reftex-format-names (namelist n) + (let (last (len (length namelist))) + (cond + ((< len 1) "") + ((= 1 len) (car namelist)) + ((> len n) (concat (car namelist) (nth 2 reftex-cite-punctuation))) + (t + (setq n (min len n) + last (nth (1- n) namelist)) + (setcdr (nthcdr (- n 2) namelist) nil) + (concat + (mapconcat 'identity namelist (nth 0 reftex-cite-punctuation)) + (nth 1 reftex-cite-punctuation) + last))))) + +(defun reftex-format-citation (entry format) + ;; Format a citation from the info in the BibTeX ENTRY + + (unless (stringp format) (setq format "\\cite{%l}")) + + (if (and reftex-comment-citations + (string-match "%l" reftex-cite-comment-format)) + (error "reftex-cite-comment-format contains illegal %%l")) + + (while (string-match + "\\(\\`\\|[^%]\\)\\(\\(%\\([0-9]*\\)\\([a-zA-Z]\\)\\)[.,;: ]*\\)" + format) + (let ((n (string-to-int (match-string 4 format))) + (l (string-to-char (match-string 5 format))) + rpl b e) + (save-match-data + (setq rpl + (cond + ((= l ?l) (concat + (reftex-get-bib-field "&key" entry) + (if reftex-comment-citations + reftex-cite-comment-format + ""))) + ((= l ?a) (reftex-format-names + (reftex-get-bib-names "author" entry) + (or n 2))) + ((= l ?A) (car (reftex-get-bib-names "author" entry))) + ((= l ?b) (reftex-get-bib-field "booktitle" entry "in: %s")) + ((= l ?B) (reftex-abbreviate-title + (reftex-get-bib-field "booktitle" entry "in: %s"))) + ((= l ?c) (reftex-get-bib-field "chapter" entry)) + ((= l ?d) (reftex-get-bib-field "edition" entry)) + ((= l ?e) (reftex-format-names + (reftex-get-bib-names "editor" entry) + (or n 2))) + ((= l ?E) (car (reftex-get-bib-names "editor" entry))) + ((= l ?h) (reftex-get-bib-field "howpublished" entry)) + ((= l ?i) (reftex-get-bib-field "institution" entry)) + ((= l ?j) (reftex-get-bib-field "journal" entry)) + ((= l ?k) (reftex-get-bib-field "key" entry)) + ((= l ?m) (reftex-get-bib-field "month" entry)) + ((= l ?n) (reftex-get-bib-field "number" entry)) + ((= l ?o) (reftex-get-bib-field "organization" entry)) + ((= l ?p) (reftex-get-bib-field "pages" entry)) + ((= l ?P) (car (split-string + (reftex-get-bib-field "pages" entry) + "[- .]+"))) + ((= l ?s) (reftex-get-bib-field "school" entry)) + ((= l ?u) (reftex-get-bib-field "publisher" entry)) + ((= l ?r) (reftex-get-bib-field "address" entry)) + ((= l ?t) (reftex-get-bib-field "title" entry)) + ((= l ?T) (reftex-abbreviate-title + (reftex-get-bib-field "title" entry))) + ((= l ?v) (reftex-get-bib-field "volume" entry)) + ((= l ?y) (reftex-get-bib-field "year" entry))))) + + (if (string= rpl "") + (setq b (match-beginning 2) e (match-end 2)) + (setq b (match-beginning 3) e (match-end 3))) + (setq format (concat (substring format 0 b) rpl (substring format e))))) + (while (string-match "%%" format) + (setq format (replace-match "%" t t format))) + (while (string-match "[ ,.;:]*%<" format) + (setq format (replace-match "" t t format))) + format) + +(defun reftex-make-cite-echo-string (entry docstruct-symbol) + ;; Format a bibtex entry for the echo area and cache the result. + (let* ((key (reftex-get-bib-field "&key" entry)) + (string + (let* ((reftex-cite-punctuation '(" " " & " " etal."))) + (reftex-format-citation entry reftex-cite-view-format))) + (cache (assq 'bibview-cache (symbol-value docstruct-symbol))) + (cache-entry (assoc key (cdr cache)))) + (unless cache + ;; This docstruct has no cache - make one. + (set docstruct-symbol (cons (cons 'bibview-cache nil) + (symbol-value docstruct-symbol)))) + (when reftex-cache-cite-echo + (setq key (copy-sequence key)) + (set-text-properties 0 (length key) nil key) + (set-text-properties 0 (length string) nil string) + (if cache-entry + (unless (string= (cdr cache-entry) string) + (setcdr cache-entry string) + (put reftex-docstruct-symbol 'modified t)) + (push (cons key string) (cdr cache)) + (put reftex-docstruct-symbol 'modified t))) + string)) + +(defun reftex-bibtex-selection-callback (data ignore no-revisit) + ;; Callback function to be called from the BibTeX selection, in + ;; order to display context. This function is relatively slow and not + ;; recommended for follow mode. It works OK for individual lookups. + (let ((win (selected-window)) + (key (reftex-get-bib-field "&key" data)) + bibfile-list item tmp) + + (catch 'exit + (save-excursion + (set-buffer reftex-call-back-to-this-buffer) + (cond + ((assq 'bib (symbol-value reftex-docstruct-symbol)) + (setq bibfile-list (reftex-get-bibfile-list))) + ((setq tmp (assq 'thebib (symbol-value reftex-docstruct-symbol))) + (setq bibfile-list (list (cdr tmp)) + item t)) + (reftex-default-bibliography + (setq bibfile-list (reftex-default-bibliography))) + (t (ding) (throw 'exit)))) + + (when no-revisit + (setq bibfile-list (reftex-visited-files bibfile-list))) + + (condition-case nil + (reftex-pop-to-bibtex-entry + key bibfile-list (not reftex-keep-temporary-buffers) t item) + (error (ding)))) + + (select-window win))) + +;;; reftex-cite.el ends here diff --git a/lisp/textmodes/reftex-global.el b/lisp/textmodes/reftex-global.el new file mode 100644 index 00000000000..72cf27a4fad --- /dev/null +++ b/lisp/textmodes/reftex-global.el @@ -0,0 +1,295 @@ +;;; reftex-global.el - Operations on entire documents with RefTeX +;;; Version: 4.5 +;;; +;;; See main file reftex.el for licensing information + +(provide 'reftex-global) +(require 'reftex) +;;; + +(defun reftex-create-tags-file () + "Create TAGS file by running `etags' on the current document. +The TAGS file is also immediately visited with `visit-tags-table'." + (interactive) + (reftex-access-scan-info current-prefix-arg) + (let* ((master (reftex-TeX-master-file)) + (files (reftex-all-document-files)) + (cmd (format "etags %s" (mapconcat 'identity files " ")))) + (save-excursion + (set-buffer (reftex-get-buffer-visiting master)) + (message "Running etags to create TAGS file...") + (shell-command cmd) + (visit-tags-table "TAGS")))) + +;; History of grep commands. +(defvar reftex-grep-history nil) +(defvar reftex-grep-command "grep -n " + "Last grep command used in \\[reftex-grep-document]; default for next grep.") + +(defun reftex-grep-document (grep-cmd) + "Run grep query through all files related to this document. +With prefix arg, force to rescan document. +No active TAGS table is required." + + (interactive + (list (read-from-minibuffer "Run grep on document (like this): " + reftex-grep-command nil nil + 'reftex-grep-history))) + (reftex-access-scan-info current-prefix-arg) + (let* ((files (reftex-all-document-files t)) + (cmd (format + "%s %s" grep-cmd + (mapconcat 'identity files " ")))) + (grep cmd))) + +(defun reftex-search-document (&optional regexp) + "Regexp search through all files of the current document. +Starts always in the master file. Stops when a match is found. +To continue searching for next match, use command \\[tags-loop-continue]. +No active TAGS table is required." + (interactive) + (let ((default (reftex-this-word))) + (unless regexp + (setq regexp (read-string (format "Search regexp in document [%s]: " + default)))) + (if (string= regexp "") (setq regexp (regexp-quote default))) + + (reftex-access-scan-info current-prefix-arg) + (tags-search regexp (list 'reftex-all-document-files)))) + +(defun reftex-query-replace-document (&optional from to delimited) + "Run a query-replace-regexp of FROM with TO over the entire document. +Third arg DELIMITED (prefix arg) means replace only word-delimited matches. +If you exit (\\[keyboard-quit] or ESC), you can resume the query replace +with the command \\[tags-loop-continue]. +No active TAGS table is required." + (interactive) + (let ((default (reftex-this-word))) + (unless from + (setq from (read-string (format "Replace regexp in document [%s]: " + default))) + (if (string= from "") (setq from (regexp-quote default)))) + (unless to + (setq to (read-string (format "Replace regexp %s with: " from)))) + (reftex-access-scan-info current-prefix-arg) + (tags-query-replace from to (or delimited current-prefix-arg) + (list 'reftex-all-document-files)))) + +(defun reftex-find-duplicate-labels () + "Produce a list of all duplicate labels in the document." + + (interactive) + + ;; Rescan the document to make sure + (reftex-access-scan-info t) + + (let ((master (reftex-TeX-master-file)) + (cnt 0) + (dlist + (mapcar + (lambda (x) + (let (x1) + (cond + ((memq (car x) + '(toc bof eof bib thebib label-numbers xr xr-doc + master-dir file-error bibview-cache appendix + is-multi index)) + nil) + (t + (setq x1 (reftex-all-assoc-string + (car x) (symbol-value reftex-docstruct-symbol))) + (if (< 1 (length x1)) + (append (list (car x)) + (mapcar (lambda(x) + (abbreviate-file-name (nth 3 x))) + x1)) + (list nil)))))) + (reftex-uniquify-by-car (symbol-value reftex-docstruct-symbol))))) + + (setq dlist (reftex-uniquify-by-car dlist)) + (if (null dlist) (error "No duplicate labels in document")) + (switch-to-buffer-other-window "*Duplicate Labels*") + (set (make-local-variable 'TeX-master) master) + (erase-buffer) + (insert " MULTIPLE LABELS IN CURRENT DOCUMENT:\n") + (insert + " Move point to label and type `r' to run a query-replace on the label\n" + " and its references. Type `q' to exit this buffer.\n\n") + (insert " LABEL FILE\n") + (insert " -------------------------------------------------------------\n") + (use-local-map (make-sparse-keymap)) + (local-set-key [?q] (lambda () "Kill this buffer." (interactive) + (kill-buffer (current-buffer)) (delete-window))) + (local-set-key [?r] 'reftex-change-label) + (while dlist + (when (and (car (car dlist)) + (cdr (car dlist))) + (incf cnt) + (insert (mapconcat 'identity (car dlist) "\n ") "\n")) + (pop dlist)) + (goto-char (point-min)) + (when (= cnt 0) + (kill-buffer (current-buffer)) + (delete-window) + (message "Document does not contain duplicate labels.")))) + +(defun reftex-change-label (&optional from to) + "Query replace FROM with TO in all \\label and \\ref commands. +Works on the entire multifile document. +If you exit (\\[keyboard-quit] or ESC), you can resume the query replace +with the command \\[tags-loop-continue]. +No active TAGS table is required." + (interactive) + (let ((default (reftex-this-word "-a-zA-Z0-9_*.:"))) + (unless from + (setq from (read-string (format "Replace label globally [%s]: " + default)))) + (if (string= from "") (setq from default)) + (unless to + (setq to (read-string (format "Replace label %s with: " + from)))) + (reftex-query-replace-document + (concat "\\\\\\(label\\|[a-z]*ref\\){" (regexp-quote from) "}") + (format "\\\\\\1{%s}" to)))) + +(defun reftex-renumber-simple-labels () + "Renumber all simple labels in the document to make them sequentially. +Simple labels are the ones created by RefTeX, consisting only of the +prefix and a number. After the command completes, all these labels will +have sequential numbers throughout the document. Any references to +the labels will be changed as well. For this, RefTeX looks at the +arguments of any macros which either start or end in the string `ref'. +This command should be used with care, in particular in multifile +documents. You should not use it if another document refers to this +one with the `xr' package." + (interactive) + ;; Resan the entire document + (reftex-access-scan-info 1) + ;; Get some insurance + (if (and (reftex-is-multi) + (not (yes-or-no-p "Replacing all simple labels in multiple files is risky. Continue? "))) + (error "Abort")) + ;; Make the translation list + (let* ((re-core (concat "\\(" + (mapconcat 'cdr reftex-typekey-to-prefix-alist "\\|") + "\\)")) + (label-re (concat "\\`" re-core "\\([0-9]+\\)\\'")) + (search-re (concat "[{,]\\(" re-core "\\([0-9]+\\)\\)[,}]")) + (error-fmt "Undefined label or reference %s. Ignore and continue? ") + (label-numbers-alist (mapcar (lambda (x) (cons (cdr x) 0)) + reftex-typekey-to-prefix-alist)) + (files (reftex-all-document-files)) + (list (symbol-value reftex-docstruct-symbol)) + translate-alist n entry label new-label nr-cell changed-sequence) + + (while (setq entry (pop list)) + (when (and (stringp (car entry)) + (string-match label-re (car entry))) + (setq label (car entry) + nr-cell (assoc (match-string 1 (car entry)) + label-numbers-alist)) + (if (assoc label translate-alist) + (error "Duplicate label %s" label)) + (setq new-label (concat (match-string 1 (car entry)) + (incf (cdr nr-cell)))) + (push (cons label new-label) translate-alist) + (or (string= label new-label) (setq changed-sequence t)))) + + (unless changed-sequence + (error "Simple labels are already in correct sequence")) + + ;; Save all document buffers before this operation + (reftex-save-all-document-buffers) + + ;; First test to check for erros + (setq n (reftex-translate + files search-re translate-alist error-fmt 'test)) + + ;; Now the real thing. + (if (yes-or-no-p + (format "Replace %d items at %d places in %d files? " + (length translate-alist) n (length files))) + (progn + (let ((inhibit-quit t)) ;; Do not disturb... + (reftex-translate + files search-re translate-alist error-fmt nil) + (setq quit-flag nil)) + (if (and (reftex-is-multi) + (yes-or-no-p "Save entire document? ")) + (reftex-save-all-document-buffers)) + ;; Rescan again... + (reftex-access-scan-info 1) + (message "Done replacing simple labels.")) + (message "No replacements done")))) + +(defun reftex-translate (files search-re translate-alist error-fmt test) + ;; In FILES, look for SEARCH-RE and replace match 1 of it with + ;; its association in TRANSLATE-ALSIT. + ;; If we do not find an association and TEST is non-nil, query + ;; to ignore the problematic string. + ;; If TEST is nil, it is ignored without query. + ;; Return the number of replacements. + (let ((n 0) file label match-data buf macro pos cell) + (while (setq file (pop files)) + (setq buf (reftex-get-file-buffer-force file)) + (unless buf + (error "No such file %s" file)) + (set-buffer buf) + (save-excursion + (save-restriction + (widen) + (goto-char (point-min)) + (while (re-search-forward search-re nil t) + (backward-char) + (save-excursion + (setq label (reftex-match-string 1) + cell (assoc label translate-alist) + match-data (match-data) + macro (reftex-what-macro 1) + pos (cdr macro)) + (goto-char (or pos (point))) + (when (and macro + (or (looking-at "\\\\ref") + (looking-at "\\\\[a-zA-Z]*ref\\(range\\)?[^a-zA-Z]") + (looking-at "\\\\ref[a-zA-Z]*[^a-zA-Z]") + (looking-at (format + reftex-find-label-regexp-format + (regexp-quote label))))) + ;; OK, we should replace it. + (set-match-data match-data) + (cond + ((and test (not cell)) + ;; We've got a problem + (unwind-protect + (progn + (reftex-highlight 1 (match-beginning 0) (match-end 0)) + (ding) + (or (y-or-n-p (format error-fmt label)) + (error "Abort"))) + (reftex-unhighlight 1))) + ((and test cell) + (incf n)) + ((and (not test) cell) + ;; Replace + (goto-char (match-beginning 1)) + (delete-region (match-beginning 1) (match-end 1)) + (insert (cdr cell))) + (t nil)))))))) + n)) + +(defun reftex-save-all-document-buffers () + "Save all documents associated with the current document. +The function is useful after a global action like replacing or renumbering +labels." + (interactive) + (let ((files (reftex-all-document-files)) + file buffer) + (save-excursion + (while (setq file (pop files)) + (setq buffer (reftex-get-buffer-visiting file)) + (when buffer + (set-buffer buffer) + (save-buffer)))))) + + +;;; reftex-global.el ends here diff --git a/lisp/textmodes/reftex-index.el b/lisp/textmodes/reftex-index.el new file mode 100644 index 00000000000..05c07bcaba0 --- /dev/null +++ b/lisp/textmodes/reftex-index.el @@ -0,0 +1,1219 @@ +;;; reftex-index.el - Index support with RefTeX +;;; Version: 4.5 +;;; +;;; See main file reftex.el for licensing information + +(provide 'reftex-index) +(require 'reftex) +;;; + +(defvar mark-active) +(defvar zmacs-regions) +(defun reftex-index-selection-or-word (&optional arg) + "Put selection or the word near point into the default index macro. +This uses the information in `reftex-index-default-macro' to make an index +entry. The phrase indexed is the current selection or the word near point. +When called with one `C-u' prefix, let the user have a chance to edit the +index entry. When called with 2 `C-u' as prefix, also ask for the index +macro and other stuff. +When called inside TeX math mode as determined by the `texmathp.el' library +which is part of AUCTeX, the string is first processed with the +`reftex-index-math-format', which see." + (interactive "P") + (let* ((use-default (not (equal arg '(16)))) ; check for double prefix + ;; check if we have an active selection + (active (if (boundp 'zmacs-regions) + (and zmacs-regions (region-exists-p)) ; XEmacs + (and transient-mark-mode mark-active))) ; Emacs + (beg (if active + (region-beginning) + (save-excursion + (skip-syntax-backward "w\\") (point)))) + (end (if active + (region-end) + (save-excursion + (skip-syntax-forward "w\\") (point)))) + (sel (buffer-substring beg end)) + (mathp (condition-case nil (texmathp) (error nil))) + (current-prefix-arg nil) ; we want to call reftex-index without prefix. + key def-char def-tag full-entry repeat-word) + + (if (equal sel "") + ;; Nothing selecte, no word, so use full reftex-index command + (reftex-index) + ;; OK, we have something to index here. + ;; Add the dollars when necessary + (setq key (if mathp + (format reftex-index-math-format sel) + sel)) + ;; Get info from `reftex-index-default-macro' + (setq def-char (if use-default (car reftex-index-default-macro))) + (setq def-tag (if use-default (nth 1 reftex-index-default-macro))) + ;; Does the user want to edit the entry? + (setq full-entry (if arg + (reftex-index-complete-key + def-tag nil (cons key 0)) + key)) + ;; Do we neet to repeat the word outside the macro? + (setq repeat-word (if use-default + (nth 2 reftex-index-default-macro) + (y-or-n-p "Repeat phrase outside macro? "))) + ;; Delete what is in the buffer and make the index entry + (delete-region beg end) + (reftex-index def-char full-entry def-tag (if repeat-word sel nil))))) + +(defun reftex-index (&optional char key tag postfix no-insert) + "Query for an index macro and insert it along with its argments. +The index macros available are those defined in `reftex-index-macro' or +by a call to `reftex-add-index-macros', typically from an AUCTeX style file. +RefteX provides completion for the index tag and the index key, and +will prompt for other arguments." + + (interactive) + + ;; Ensure access to scanning info + (reftex-ensure-index-support t) + (reftex-access-scan-info current-prefix-arg) + + ;; Find out which macro we are going to use + (let* ((char (or char + (reftex-select-with-char reftex-query-index-macro-prompt + reftex-query-index-macro-help))) + (macro (nth 1 (assoc char reftex-key-to-index-macro-alist))) + (entry (or (assoc macro reftex-index-macro-alist) + (error "No index macro associated with %c" char))) + (ntag (nth 1 entry)) + (tag (or tag (nth 1 entry))) + (nargs (nth 4 entry)) + (nindex (nth 5 entry)) + (opt-args (nth 6 entry)) + opt tag1 value) + + ;; Get the supported arguments + (if (stringp tag) + (setq tag1 tag) + (setq tag1 (or (reftex-index-complete-tag tag opt-args) ""))) + (setq key (or key + (reftex-index-complete-key + (if (string= tag1 "") "idx" tag1) + (member nindex opt-args)))) + + ;; Insert the macro and ask for any additional args + (insert macro) + (loop for i from 1 to nargs do + (setq opt (member i opt-args) + value (cond ((= nindex i) key) + ((equal ntag i) tag1) + (t (read-string (concat "Macro arg nr. " + (int-to-string i) + (if opt " (optional)" "") + ": "))))) + (unless (and opt (string= value "")) + (insert (if opt "[" "{") value (if opt "]" "}")))) + (and (stringp postfix) (insert postfix)) + (and key reftex-plug-into-AUCTeX (fboundp 'LaTeX-add-index-entries) + (LaTeX-add-index-entries key)) + (reftex-index-update-taglist tag1) + (reftex-notice-new))) + +(defun reftex-default-index () + (cond ((null reftex-index-default-tag) nil) + ((stringp reftex-index-default-tag) reftex-index-default-tag) + (t (or (get reftex-docstruct-symbol 'default-index-tag) + "idx")))) + +(defun reftex-update-default-index (tag &optional tag-list) + (if (and (not (equal tag "")) + (stringp tag) + (eq reftex-index-default-tag 'last) + (or (null tag-list) + (member tag tag-list))) + (put reftex-docstruct-symbol 'default-index-tag tag))) + +(defun reftex-index-complete-tag (&optional itag opt-args) + ;; Ask the user for a tag, completing on known tags. + ;; ITAG is the argument number which contains the tag. + ;; OPT-ARGS is a list of optional argument indices, as given by + ;; `reftex-parse-args'. + (let* ((opt (and (integerp itag) (member itag opt-args))) + (index-tags (cdr (assq 'index-tags + (symbol-value reftex-docstruct-symbol)))) + (default (reftex-default-index)) + (prompt (concat "Index tag" + (if default (format " (default: %s)" default) "") + (if opt " (optional)" "") ": ")) + (tag (completing-read prompt (mapcar 'list index-tags)))) + (if (and default (equal tag "")) (setq tag default)) + (reftex-update-default-index tag) + tag)) + +(defun reftex-index-select-tag () + ;; Have the user select an index tag. + ;; FIXME: should we cache tag-alist, prompt and help? + (let* ((index-tags (cdr (assoc 'index-tags + (symbol-value reftex-docstruct-symbol)))) + (default (reftex-default-index))) + (cond + ((null index-tags) + (error "No index tags available")) + + ((= (length index-tags) 1) + ;; Just one index, use it + (car index-tags)) + + ((> (length index-tags) 1) + ;; Several indices, ask. + (let* ((tags (copy-sequence index-tags)) + (cnt 0) + tag-alist i val len tag prompt help rpl) + ;; Move idx and glo up in the list to ensure ?i and ?g shortcuts + (if (member "glo" tags) + (setq tags (cons "glo" (delete "glo" tags)))) + (if (member "idx" tags) + (setq tags (cons "idx" (delete "idx" tags)))) + ;; Find unique shortcuts for each index. + (while (setq tag (pop tags)) + (setq len (length tag) + i -1 + val nil) + (catch 'exit + (while (and (< (incf i) len) (null val)) + (unless (assq (aref tag i) tag-alist) + (push (list (aref tag i) + tag + (concat (substring tag 0 i) + "[" (substring tag i (incf i)) "]" + (substring tag i))) + tag-alist) + (throw 'exit t))) + (push (list (+ ?0 (incf cnt)) tag + (concat "[" (int-to-string cnt) "]:" tag)) + tag-alist))) + (setq tag-alist (nreverse tag-alist)) + ;; Compute Prompt and Help strings + (setq prompt + (concat + (format "Select Index%s: " + (if default (format " (Default <%s>)" default) "")) + (mapconcat (lambda(x) (nth 2 x)) tag-alist " "))) + (setq help + (concat "Select an Index\n===============\n" + (if default + (format "[^M] %s (the default)\n" default) + "") + (mapconcat (lambda(x) + (apply 'format "[%c] %s" x)) + tag-alist "\n"))) + ;; Query the user for an index-tag + (setq rpl (reftex-select-with-char prompt help 3 t)) + (message "") + (if (and default (equal rpl ?\C-m)) + default + (if (assq rpl tag-alist) + (progn + (reftex-update-default-index (nth 1 (assq rpl tag-alist))) + (nth 1 (assq rpl tag-alist))) + (error "No index tag associated with %c" rpl))))) + (t (error "This should not happen (reftex-index-select-tag)"))))) + +(defun reftex-index-complete-key (&optional tag optional initial) + ;; Read an index key, with completion. + ;; Restrict completion table on index tag TAG. + ;; OPTIONAL indicates if the arg is optional. + (let* ((table (reftex-sublist-nth + (symbol-value reftex-docstruct-symbol) 6 + (lambda(x) (and (eq (car x) 'index) + (string= (nth 1 x) (or tag "")))) + t)) + (prompt (concat "Index key" (if optional " (optional)" "") ": ")) + (key (completing-read prompt table nil nil initial))) + key)) + +(defun reftex-index-update-taglist (newtag) + ;; add NEWTAG to the list of available index tags. + (let ((cell (assoc 'index-tags (symbol-value reftex-docstruct-symbol)))) + (and newtag (cdr cell) (not (member newtag (cdr cell))) + (push newtag (cdr cell))))) + +(defvar reftex-last-index-file) +(defun reftex-index-globally (&optional data call-file) + "Index a word with a global search and replace. +This works very much like `reftex-query-replace-document', but the +defaults for the search and replace strings are derived from +local context. +When there is an index entry, we try to index similar words. The word +to search for is either a word in direct contact with the index macro +(like `\\index{WORD}WORD' or `WORD\\index{WORD}') or the index key. +The replacement text is the index macro with all its arguments and the +attached word. +When there is no index entry at point, we search for the word near point +and propose to index it like this: `\\index{word}word'. +You get a chance to edit the search and replacement strings. +DATA can be a docstruct entry describing an index entry, and then the +defaults will be derived from it. +CALL-FILE may be the file from where to call the global search command." + (interactive) + (let* ((call-file (cond (call-file call-file) + (reftex-mode (buffer-file-name)) + ((eq major-mode 'reftex-index-mode) + reftex-last-index-file) + (t (error "Need a call file here")))) + (pos (point)) + (data (cond + (data data) + ((and reftex-mode + (save-excursion + (forward-char 20) + (re-search-backward reftex-everything-regexp nil t) + (< (count-lines (min pos (point)) (max pos (point))) + 2))) + (reftex-index-info (buffer-file-name))) + (t nil))) + (ksep (car reftex-index-special-chars)) + (words-include-escapes t) + (case-replace nil) + (case-fold-search t) + word rpl start analyze-list pre key attr actual post) + + ;; Find the word and construct the replacement string + (if (and data (eq (car data) 'index)) + ;; OK, we have an index entry + (progn + (setq analyze-list (reftex-index-analyze-entry data) + pre (car analyze-list) + key (nth 1 analyze-list) + attr (nth 2 analyze-list) + actual (nth 3 analyze-list) + post (nth 4 analyze-list)) + (when (string-match (concat "\\<\\(\\sw+\\)" reftex-index-re) pre) + (setq word (match-string 1 pre) + pre (concat "<<<1>>>" (substring pre (match-end 1))) + rpl (concat pre key attr actual post))) + (when (string-match "}\\(\\sw+\\)\\>[^}]*\\'" post) + (setq word (match-string 1 post) + post (concat (substring post 0 (match-beginning 1)) + "<<<1>>>") + rpl (concat pre key attr actual post))) + (when (and (not word) key) + (if (string-match (concat ".*" (regexp-quote ksep)) key) + (setq word (substring key (match-end 0))) + (setq word key)) + (setq rpl (concat pre key attr actual post)))) + ;; No index entry, just use local word. + (setq word (save-excursion + (buffer-substring-no-properties + (progn (skip-syntax-backward "w") (point)) + (progn (skip-syntax-forward "w") (point)))) + rpl (concat "\\index{" word "}<<<1>>>"))) + ;; Quote what is necessary + (setq word (regexp-quote (downcase word))) + (setq start 0) + (while (setq start (string-match "\\\\" rpl start)) + (setq rpl (replace-match "\\\\" t t rpl) + start (+ 2 start))) + ;; We used <<<1>>> instead of \1 to avoid the quoting. Fix this now. + (if (string-match "<<<1>>>" rpl) + (setq rpl (replace-match "\\1" t t rpl))) + + ;; Give the user a chance to edit the strings + (setq word (read-string "Search: " + (if word (format "\\<\\(%s\\)\\>" word))) + rpl (read-string "Replace with: " rpl)) + + ;; Execute the command + (save-excursion + (switch-to-buffer (get-file-buffer call-file)) + (condition-case nil + (reftex-query-replace-document word rpl) + (error nil))))) + +(defvar reftex-index-map (make-sparse-keymap) + "Keymap used for *Index* buffers.") + +(defvar reftex-index-menu) + +(defvar reftex-last-index-file nil + "Stores the file name from which `reftex-display-index' was called.") +(defvar reftex-index-tag nil + "Stores the tag of the index in an index buffer.") + +(defvar reftex-index-return-marker (make-marker) + "Marker which makes it possible to return from index to old position.") + +(defvar reftex-index-restriction-indicator nil) +(defvar reftex-index-restriction-data nil) + +(defun reftex-index-mode () + "Major mode for managing Index buffers for LaTeX files. +This buffer was created with RefTeX. +Press `?' for a summary of important key bindings, or check the menu. + +Here are all local bindings. + +\\{reftex-index-map}" + (interactive) + (kill-all-local-variables) + (setq major-mode 'reftex-index-mode + mode-name "RefTeX Index") + (use-local-map reftex-index-map) + (set (make-local-variable 'revert-buffer-function) 'reftex-index-revert) + (set (make-local-variable 'reftex-index-restriction-data) nil) + (set (make-local-variable 'reftex-index-restriction-indicator) nil) + (setq mode-line-format + (list "---- " 'mode-line-buffer-identification + " " 'global-mode-string + " R<" 'reftex-index-restriction-indicator ">" + " -%-")) + (setq truncate-lines t) + (make-local-hook 'post-command-hook) + (make-local-hook 'pre-command-hook) + (make-local-variable 'reftex-last-follow-point) + (easy-menu-add reftex-index-menu reftex-index-map) + (add-hook 'post-command-hook 'reftex-index-post-command-hook nil t) + (add-hook 'pre-command-hook 'reftex-index-pre-command-hook nil t) + (run-hooks 'reftex-index-mode-hook)) + +(defconst reftex-index-help +" AVAILABLE KEYS IN INDEX BUFFER + ============================== +! A..Z Goto the section of entries starting with this letter. +n / p next-entry / previous-entry +SPC / TAB Show/Goto the corresponding entry in the LaTeX document. +RET Goto the entry and hide the *Index* window (also on mouse-2). +q / k Hide/Kill *Index* buffer. +C-c = Switch to the TOC buffer. +f / c Toggle follow mode / Toggle display of [c]ontext. +g Refresh *Index* buffer. +r / C-u r Reparse the LaTeX document / Reparse entire LaTeX document. +s Switch to a different index (for documents with multiple indices). +e / C-k Edit/Kill the entry. +* | @ Edit specific part of entry: [*]key [|]attribute [@]visual + With prefix: kill that part. +( ) Toggle entry's beginning/end of page range property. +_ ^ Add/Remove parent key (to make this item a subitem). +& Index the same word everywhere in the document. +} / { Restrict Index to a single document section / Widen. +< / > When restricted, move restriction to previous/next section.") + +(defun reftex-index-show-entry (data &optional no-revisit) + ;; Find an index entry associated with DATA and display it highlighted + ;; in another window. NO-REVISIT means we are not allowed to visit + ;; files for this. + ;; Note: This function just looks for the nearest match of the + ;; context string and may fail if the entry moved and an identical + ;; entry is close to the old position. Frequent rescans make this + ;; safer. + (let* ((file (nth 3 data)) + (literal (nth 2 data)) + (pos (nth 4 data)) + (re (regexp-quote literal)) + (match + (cond + ((or (not no-revisit) + (reftex-get-buffer-visiting file)) + (switch-to-buffer-other-window + (reftex-get-file-buffer-force file nil)) + (goto-char (or pos (point-min))) + (or (looking-at re) + (reftex-nearest-match re (length literal)))) + (t (message reftex-no-follow-message) nil)))) + (when match + (goto-char (match-beginning 0)) + (recenter '(4)) + (reftex-highlight 0 (match-beginning 0) (match-end 0) (current-buffer))) + match)) + +(defun reftex-display-index (&optional tag overriding-restriction + &rest locations) + "Display a buffer with an index compiled from the current document. +When the document has multiple indices, first prompts for the correct one. +When index support is turned off, offer to turn it on. +With one or two `C-u' prefixes, rescan document first. +With prefix 2, restrict index to current document section. +With prefix 3, restrict index to region." + + (interactive) + + ;; Ensure access to scanning info and rescan buffer if prefix are is '(4). + (let ((current-prefix-arg current-prefix-arg)) + (reftex-ensure-index-support t) + (reftex-access-scan-info current-prefix-arg)) + + (set-marker reftex-index-return-marker (point)) + (setq reftex-last-follow-point 1) + + ;; Determine the correct index to process + (let* ((docstruct (symbol-value reftex-docstruct-symbol)) + (docstruct-symbol reftex-docstruct-symbol) + (index-tag (or tag (reftex-index-select-tag))) + (master (reftex-TeX-master-file)) + (calling-file (buffer-file-name)) + (restriction + (or overriding-restriction + (and (interactive-p) + (reftex-get-restriction current-prefix-arg docstruct)))) + (locations + ;; See if we are on an index macro as initial position + (or locations + (let* ((what-macro (reftex-what-macro-safe 1)) + (macro (car what-macro)) + (here-I-am (when (member macro reftex-macros-with-index) + (save-excursion + (goto-char (+ (cdr what-macro) + (length macro))) + (reftex-move-over-touching-args) + (reftex-where-am-I))))) + (if (eq (car (car here-I-am)) 'index) + (list (car here-I-am)))))) + buffer-name) + + (setq buffer-name (reftex-make-index-buffer-name index-tag)) + + ;; Goto the buffer and put it into the correct mode + + (when (or restriction current-prefix-arg) + (reftex-kill-buffer buffer-name)) + + (if (get-buffer-window buffer-name) + (select-window (get-buffer-window buffer-name)) + (let ((default-major-mode 'reftex-index-mode)) + (switch-to-buffer buffer-name))) + + (or (eq major-mode 'reftex-index-mode) (reftex-index-mode)) + + ;; If the buffer is currently restricted, empty it to force update. + (when reftex-index-restriction-data + (reftex-erase-buffer)) + (set (make-local-variable 'reftex-last-index-file) calling-file) + (set (make-local-variable 'reftex-index-tag) index-tag) + (set (make-local-variable 'reftex-docstruct-symbol) docstruct-symbol) + (if restriction + (setq reftex-index-restriction-indicator (car restriction) + reftex-index-restriction-data (cdr restriction)) + (if (interactive-p) + (setq reftex-index-restriction-indicator nil + reftex-index-restriction-data nil))) + (when (= (buffer-size) 0) + ;; buffer is empty - fill it + (message "Building %s buffer..." buffer-name) + + (setq buffer-read-only nil) + (insert (format +"INDEX <%s> on %s +Restriction: <%s> +SPC=view TAB=goto RET=goto+hide [e]dit [q]uit [r]escan [f]ollow [?]Help +------------------------------------------------------------------------------ +" index-tag (abbreviate-file-name master) +(if (eq (car (car reftex-index-restriction-data)) 'toc) + (nth 2 (car reftex-index-restriction-data)) + reftex-index-restriction-indicator))) + + (if (reftex-use-fonts) + (put-text-property 1 (point) 'face reftex-index-header-face)) + (put-text-property 1 (point) 'intangible t) + + (reftex-insert-index docstruct index-tag) + (goto-char (point-min)) + (run-hooks 'reftex-display-copied-context-hook) + (message "Building %s buffer...done." buffer-name) + (setq buffer-read-only t)) + (and locations (apply 'reftex-find-start-point (point) locations)) + (if reftex-index-restriction-indicator + (message "Index restricted: <%s>" reftex-index-restriction-indicator)))) + +(defun reftex-insert-index (docstruct tag &optional update-one remark) + ;; Insert an index into the current buffer. Entries are from the + ;; DOCSTRUCT. + ;; TAG is the subindex to process. + ;; UPDATE-ONE: When non-nil, delete the entry at point and replace + ;; it with whatever the DOCSTRUCT contains. + ;; REMARK can be a note to add to the entry. + (let* ((all docstruct) + (indent " ") + (context reftex-index-include-context) + (context-indent (concat indent " ")) + (section-chars (mapcar 'identity reftex-index-section-letters)) + (this-section-char 0) + (font (reftex-use-fonts)) + (bor (car reftex-index-restriction-data)) + (eor (nth 1 reftex-index-restriction-data)) + (mouse-face + (if (memq reftex-highlight-selection '(mouse both)) + reftex-mouse-selected-face + nil)) + (index-face (reftex-verified-face reftex-label-face + 'font-lock-constant-face + 'font-lock-reference-face)) + sublist cell from to first-char) + + ;; Make the sublist and sort it + (when bor + (setq all (or (memq bor all) all))) + + (while (setq cell (pop all)) + (if (eq cell eor) + (setq all nil) + (and (eq (car cell) 'index) + (equal (nth 1 cell) tag) + (push cell sublist)))) + (setq sublist (sort (nreverse sublist) + (lambda (a b) (string< (nth 8 a) (nth 8 b))))) + + (when update-one + ;; Delete the entry at place + (and (bolp) (forward-char 1)) + (delete-region (previous-single-property-change (1+ (point)) :data) + (or (next-single-property-change (point) :data) + (point-max)))) + + ;; Walk through the list and insert all entries + (while (setq cell (pop sublist)) + (unless update-one + (setq first-char (upcase (string-to-char (nth 6 cell)))) + (when (and (not (equal first-char this-section-char)) + (member first-char section-chars)) + ;; There is a new initial letter, so start a new section + (reftex-index-insert-new-letter first-char font) + (setq section-chars (delete first-char section-chars) + this-section-char first-char)) + (when (= this-section-char 0) + (setq this-section-char ?!) + (reftex-index-insert-new-letter this-section-char font))) + + (setq from (point)) + (insert indent (nth 7 cell)) + (when font + (setq to (point)) + (put-text-property + (- (point) (length (nth 7 cell))) to + 'face index-face) + (goto-char to)) + + (when (or remark (nth 9 cell)) + (and (< (current-column) 40) + ;; FIXME: maybe this is too slow? + (insert (make-string (max (- 40 (current-column)) 0) ?\ ))) + (and (nth 9 cell) (insert " " (substring (nth 5 cell) (nth 9 cell)))) + (and remark (insert " " remark))) + + (insert "\n") + (setq to (point)) + + (when context + (insert context-indent (nth 2 cell) "\n") + (setq to (point))) + (put-text-property from to :data cell) + (when mouse-face + (put-text-property from (1- to) + 'mouse-face mouse-face)) + (goto-char to)))) + + +(defun reftex-index-insert-new-letter (letter &optional font) + ;; Start a new section in the index + (let ((from (point))) + (insert "\n" letter letter letter + "-----------------------------------------------------------------") + (when font + (put-text-property from (point) 'face reftex-index-section-face)) + (insert "\n"))) + +(defun reftex-get-restriction (arg docstruct) + ;; Interprete the prefix ARG and derive index restriction specs. + (let* ((beg (min (point) (or (condition-case nil (mark) (error nil)) + (point-max)))) + (end (max (point) (or (condition-case nil (mark) (error nil)) + (point-min)))) + bor eor label here-I-am) + (cond + ((eq arg 2) + (setq here-I-am (car (reftex-where-am-I)) + bor (if (eq (car here-I-am) 'toc) + here-I-am + (reftex-last-assoc-before-elt + 'toc here-I-am docstruct)) + eor (car (memq (assq 'toc (cdr (memq bor docstruct))) docstruct)) + label (nth 6 bor))) + ((eq arg 3) + (save-excursion + (setq label "region") + (goto-char beg) + (setq bor (car (reftex-where-am-I))) + (setq bor (nth 1 (memq bor docstruct))) + (goto-char end) + (setq eor (nth 1 (memq (car (reftex-where-am-I)) docstruct))))) + (t nil)) + (if (and label (or bor eor)) + (list label bor eor) + nil))) + +(defun reftex-index-pre-command-hook () + ;; Used as pre command hook in *Index* buffer + (reftex-unhighlight 0) + (reftex-unhighlight 1)) + +(defun reftex-index-post-command-hook () + ;; Used in the post-command-hook for the *Index* buffer + (when (get-text-property (point) :data) + (and (> (point) 1) + (not (get-text-property (point) 'intangible)) + (memq reftex-highlight-selection '(cursor both)) + (reftex-highlight 1 + (or (previous-single-property-change (1+ (point)) :data) + (point-min)) + (or (next-single-property-change (point) :data) + (point-max))))) + (if (integerp reftex-index-follow-mode) + ;; Remove delayed action + (setq reftex-index-follow-mode t) + (and reftex-index-follow-mode + (not (equal reftex-last-follow-point (point))) + ;; Show context in other window + (setq reftex-last-follow-point (point)) + (condition-case nil + (reftex-index-visit-location nil (not reftex-revisit-to-follow)) + (error t))))) + +(defun reftex-index-show-help () + "Show a summary of special key bindings." + (interactive) + (with-output-to-temp-buffer "*RefTeX Help*" + (princ reftex-index-help)) + (reftex-enlarge-to-fit "*RefTeX Help*" t) + ;; If follow mode is active, arrange to delay it one command + (if reftex-index-follow-mode + (setq reftex-index-follow-mode 1))) + +(defun reftex-index-next (&optional arg) + "Move to next selectable item." + (interactive "p") + (setq reftex-callback-fwd t) + (or (eobp) (forward-char 1)) + (goto-char (or (next-single-property-change (point) :data) + (point))) + (unless (get-text-property (point) :data) + (goto-char (or (next-single-property-change (point) :data) + (point))))) +(defun reftex-index-previous (&optional arg) + "Move to previous selectable item." + (interactive "p") + (setq reftex-callback-fwd nil) + (goto-char (or (previous-single-property-change (point) :data) + (point))) + (unless (get-text-property (point) :data) + (goto-char (or (previous-single-property-change (point) :data) + (point))))) +(defun reftex-index-toggle-follow () + "Toggle follow (other window follows with context)." + (interactive) + (setq reftex-last-follow-point -1) + (setq reftex-index-follow-mode (not reftex-index-follow-mode))) +(defun reftex-index-toggle-context () + "Toggle inclusion of label context in *Index* buffer. +Label context is only displayed when the labels are there as well." + (interactive) + (setq reftex-index-include-context (not reftex-index-include-context)) + (reftex-index-revert)) +(defun reftex-index-view-entry () + "View document location in other window." + (interactive) + (reftex-index-visit-location)) +(defun reftex-index-goto-entry-and-hide () + "Go to document location in other window. Hide the *Index* window." + (interactive) + (reftex-index-visit-location 'hide)) +(defun reftex-index-goto-entry () + "Go to document location in other window. *Index* window stays." + (interactive) + (reftex-index-visit-location t)) +(defun reftex-index-mouse-goto-line-and-hide (ev) + "Go to document location in other window. Hide the *Index* window." + (interactive "e") + (mouse-set-point ev) + (reftex-index-visit-location 'hide)) +(defun reftex-index-quit () + "Hide the *Index* window and do not move point." + (interactive) + (or (one-window-p) (delete-window)) + (switch-to-buffer (marker-buffer reftex-index-return-marker)) + (goto-char (or (marker-position reftex-index-return-marker) (point)))) +(defun reftex-index-quit-and-kill () + "Kill the *Index* buffer." + (interactive) + (kill-buffer (current-buffer)) + (or (one-window-p) (delete-window)) + (switch-to-buffer (marker-buffer reftex-index-return-marker)) + (goto-char (or (marker-position reftex-index-return-marker) (point)))) +(defun reftex-index-goto-toc (&rest ignore) + "Switch to the table of contents of the current document. +The function will go to the section where the entry at point was defined." + (interactive) + (if (get-text-property (point) :data) + (reftex-index-goto-entry) + (switch-to-buffer (marker-buffer reftex-index-return-marker))) + (delete-other-windows) + (reftex-toc)) +(defun reftex-index-rescan (&rest ignore) + "Regenerate the *Index* buffer after reparsing file of section at point." + (interactive) + (let ((index-tag reftex-index-tag)) + (if (and reftex-enable-partial-scans + (null current-prefix-arg)) + (let* ((data (get-text-property (point) :data)) + (file (nth 3 data)) + (line (+ (count-lines (point-min) (point)) (if (bolp) 1 0)))) + (if (not file) + (error "Don't know which file to rescan. Try `C-u r'") + (switch-to-buffer (reftex-get-file-buffer-force file)) + (setq current-prefix-arg '(4)) + (reftex-display-index index-tag nil line))) + (reftex-index-Rescan)) + (reftex-kill-temporary-buffers))) +(defun reftex-index-Rescan (&rest ignore) + "Regenerate the *Index* buffer after reparsing the entire document." + (interactive) + (let ((index-tag reftex-index-tag) + (line (+ (count-lines (point-min) (point)) (if (bolp) 1 0)))) + (switch-to-buffer + (reftex-get-file-buffer-force reftex-last-index-file)) + (setq current-prefix-arg '(16)) + (reftex-display-index index-tag nil line))) +(defun reftex-index-revert (&rest ignore) + "Regenerate the *Index* from the internal lists. No reparsing os done." + (interactive) + (let ((buf (current-buffer)) + (index-tag reftex-index-tag) + (data (get-text-property (point) :data)) + (line (+ (count-lines (point-min) (point)) (if (bolp) 1 0)))) + (switch-to-buffer + (reftex-get-file-buffer-force reftex-last-index-file)) + (reftex-erase-buffer buf) + (setq current-prefix-arg nil + reftex-last-follow-point 1) + (reftex-display-index index-tag nil data line))) +(defun reftex-index-switch-index-tag (&rest ignore) + "Switch to a different index of the same document." + (interactive) + (switch-to-buffer + (reftex-get-file-buffer-force reftex-last-index-file)) + (setq current-prefix-arg nil) + (reftex-display-index)) + +(defun reftex-index-restrict-to-section (&optional force) + "Restrict index to entries defined in same document sect. as entry at point." + ;; Optional FORCE means, even if point is not on an index entry. + (interactive) + (let* ((data (get-text-property (point) :data)) + (docstruct (symbol-value reftex-docstruct-symbol)) + bor eor) + (if (and (not data) force) + (setq data (assq 'toc docstruct))) + (when data + (setq bor (reftex-last-assoc-before-elt 'toc data docstruct) + eor (car (memq (assq 'toc (cdr (memq bor docstruct))) + docstruct)) + reftex-index-restriction-data (list bor eor) + reftex-index-restriction-indicator (nth 6 bor) ))) + (reftex-index-revert)) + +(defun reftex-index-widen (&rest ignore) + "Show the unrestricted index (all entries)." + (interactive) + (setq reftex-index-restriction-indicator nil + reftex-index-restriction-data nil) + (reftex-index-revert) + (message "Index widened")) +(defun reftex-index-restriction-forward (&rest ignore) + "Restrict to previous section. +When index is currently unrestricted, restrict it to a section. +When index is restricted, select the next section as restriction criterion." + (interactive) + (let* ((docstruct (symbol-value reftex-docstruct-symbol)) + (bor (nth 1 reftex-index-restriction-data))) + (if (or (not bor) + (not (eq (car bor) 'toc))) + (reftex-index-restrict-to-section t) + (setq reftex-index-restriction-indicator (nth 6 bor) + reftex-index-restriction-data + (list bor + (car (memq (assq 'toc (cdr (memq bor docstruct))) + docstruct)))) + (reftex-index-revert)))) +(defun reftex-index-restriction-backward (&rest ignore) + "Restrict to next section. +When index is currently unrestricted, restrict it to a section. +When index is restricted, select the previous section as restriction criterion." + (interactive) + (let* ((docstruct (symbol-value reftex-docstruct-symbol)) + (eor (car reftex-index-restriction-data)) + (bor (reftex-last-assoc-before-elt 'toc eor docstruct t))) + (if (or (not bor) + (not (eq (car bor) 'toc))) + (reftex-index-restrict-to-section t) + (setq reftex-index-restriction-indicator (nth 6 bor) + reftex-index-restriction-data + (list bor eor)) + (reftex-index-revert)))) + +(defun reftex-index-visit-location (&optional final no-revisit) + ;; Visit the tex file corresponding to the index entry on the current line. + ;; If FINAL is t, stay there + ;; If FINAL is 'hide, hide the *Index* window. + ;; Otherwise, move cursor back into *Index* window. + ;; NO-REVISIT means don't visit files, just use live biffers. + + (let* ((data (get-text-property (point) :data)) + (index-window (selected-window)) + show-window show-buffer match) + + (unless data (error "Don't know which index entry to visit")) + + (if (eq (car data) 'index) + (setq match (reftex-index-show-entry data no-revisit))) + + (setq show-window (selected-window) + show-buffer (current-buffer)) + + (unless match + (select-window index-window) + (error "Cannot find location")) + + (select-window index-window) + + ;; Use the `final' parameter to decide what to do next + (cond + ((eq final t) + (reftex-unhighlight 0) + (select-window show-window)) + ((eq final 'hide) + (reftex-unhighlight 0) + (or (one-window-p) (delete-window)) + (switch-to-buffer show-buffer)) + (t nil)))) + +(defun reftex-index-analyze-entry (data) + ;; This splits the index context so that key, attribute and visual + ;; values are accessible individually. + (interactive) + (let* ((arg (nth 5 data)) + (context (nth 2 data)) + (sc reftex-index-special-chars) + (boa (if (string-match (regexp-quote (concat "{" arg "}")) context) + (1+ (match-beginning 0)) + (error "Something is wrong here"))) + (eoa (1- (match-end 0))) + (boactual (if (string-match (concat "[^" (nth 3 sc) "]" (nth 2 sc)) + context boa) + (1+ (match-beginning 0)) + eoa)) + (boattr (if (string-match (concat "[^" (nth 3 sc) "]" (nth 1 sc)) + context boa) + (1+ (match-beginning 0)) + boactual)) + (pre (substring context 0 boa)) + (key (substring context boa boattr)) + (attr (substring context boattr boactual)) + (actual (substring context boactual eoa)) + (post (substring context eoa))) + (list pre key attr actual post))) + +(defun reftex-index-globalize (&optional arg) + "Globalize the current index entry. +This starts a global search and replace to index the same word +at other places in the document. After this function completes, you +need to rescan the document with `r' or `C-u r' in order to get the +entries into the index buffer. +Defaults for the search and replace strings are derived from +the current entry. See the command `reftex-index-globally'." + (interactive) + (let* ((data (get-text-property (point) :data)) + (buf (current-buffer))) + (unless data + (error "No index entry at point")) + (reftex-index-globally data) + (switch-to-buffer buf))) + +(defun reftex-index-edit () + "Edit the index entry at point." + (interactive) + (let* ((data (get-text-property (point) :data)) + old new) + (unless data (error "Don't know which index entry to edit")) + (reftex-index-view-entry) + (setq old (nth 2 data) new (read-string "Edit: " old)) + (reftex-index-change-entry new))) + +(defun reftex-index-toggle-range-beginning () + "Toggle the page range start attribute `|('." + (interactive) + (let* ((data (get-text-property (point) :data)) + (bor (concat (nth 1 reftex-index-special-chars) "(")) + new analyze attr) + (unless data (error "Don't know which index entry to edit")) + (setq analyze (reftex-index-analyze-entry data) + attr (nth 2 analyze)) + (setf (nth 2 analyze) (if (string= attr bor) "" bor)) + (setq new (apply 'concat analyze)) + (reftex-index-change-entry + new (if (string= (nth 2 analyze) bor) + "Entry is now START-OF-PAGE-RANGE" + "START-OF-PAGE-RANGE canceled")))) + +(defun reftex-index-toggle-range-end () + "Toggle the page-range-end attribute `|)'." + (interactive) + (let* ((data (get-text-property (point) :data)) + (eor (concat (nth 1 reftex-index-special-chars) "(")) + new analyze attr) + (unless data (error "Don't know which index entry to edit")) + (setq analyze (reftex-index-analyze-entry data) + attr (nth 2 analyze)) + (setf (nth 2 analyze) (if (string= attr eor) "" eor)) + (setq new (apply 'concat analyze)) + (reftex-index-change-entry + new (if (string= (nth 2 analyze) eor) + "Entry is now END-OF-PAGE-RANGE" + "END-OF-PAGE-RANGE canceled")))) + +(defun reftex-index-edit-key () + "Edit the KEY part of the index entry." + (interactive) + (reftex-index-edit-part nil 1 "" "Key: " t)) + +(defun reftex-index-edit-attribute (&optional arg) + "EDIT the ATTRIBUTE part of the entry. With arg: remove entire ATTRIBUTE." + (interactive "P") + (reftex-index-edit-part arg 2 (nth 1 reftex-index-special-chars) + "Attribute: ")) + +(defun reftex-index-edit-visual (&optional arg) + "EDIT the VISUAL part of the entry. With arg: remove entire VISUAL string." + (interactive "P") + (reftex-index-edit-part arg 3 (nth 2 reftex-index-special-chars) "Visual: ")) + +(defun reftex-index-edit-part (arg n initial prompt &optional dont-allow-empty) + ;; This function does the work for all partial editing commands + (let* ((data (get-text-property (point) :data)) + new analyze opart npart) + (unless data (error "Don't know which index entry to edit")) + ;; Analyze the whole context string + (setq analyze (reftex-index-analyze-entry data) + opart (nth n analyze)) + (and (> (length opart) 0) (setq opart (substring opart 1))) + ;; Have the user editing the part + (setq npart (if arg "" (read-string (concat prompt initial) opart))) + ;; Tests: + (cond ((string= npart opart) + (error "Not changed")) + ((string= npart "") + (if dont-allow-empty + (error "Illegal value") + (setf (nth n analyze) npart))) + (t (setf (nth n analyze) (concat initial npart)))) + (setq new (apply 'concat analyze)) + ;; Change the entry and insert the changed version into the index. + (reftex-index-change-entry + new (if (string= npart "") + (format "Deleted: %s" opart) + (format "New value is: %s" npart))))) + +(defun reftex-index-level-down () + "Make index entry a subitem of another entry." + (interactive) + (let* ((data (get-text-property (point) :data)) + (docstruct (symbol-value reftex-docstruct-symbol)) + old new prefix key) + (unless data (error "Don't know which index entry to change")) + (setq old (nth 2 data) + key (nth 6 data) + prefix (completing-read + "Prefix: " + (reftex-sublist-nth + docstruct 6 + (lambda (x) + (and (eq (car x) 'index) + (string= (nth 1 x) reftex-index-tag))) t))) + (unless (string-match + (concat (regexp-quote (car reftex-index-special-chars)) "\\'") + prefix) + (setq prefix (concat prefix (car reftex-index-special-chars)))) + (if (string-match (regexp-quote key) old) + (setq new (replace-match (concat prefix key) t t old)) + (error "Cannot construct new index key")) + (reftex-index-change-entry new (format "Added prefix: %s" prefix)))) + +(defun reftex-index-level-up () + "Remove the highest level of a hierarchical index entry." + (interactive) + (let* ((data (get-text-property (point) :data)) + old new prefix) + (unless data (error "Don't know which entry to change")) + (setq old (nth 2 data)) + (if (string-match (concat "{\\([^" (nth 0 reftex-index-special-chars) "]*" + "[^" (nth 3 reftex-index-special-chars) "]" + (regexp-quote (nth 0 reftex-index-special-chars)) + "\\)") + old) + (setq prefix (substring old (match-beginning 1) (match-end 1)) + new (concat (substring old 0 (match-beginning 1)) + (substring old (match-end 1)))) + (error "Entry is not a subitem")) + (reftex-index-change-entry new (format "Removed prefix: %s" prefix)))) + +(defun reftex-index-kill () + "FIXME: Not yet implemented" + (interactive) + (error "This function is currently not implemented")) + +(defun reftex-index-undo () + "FIXME: Not yet implemented" + (interactive) + (error "This function is currently not implemented")) + +(defun reftex-index-change-entry (new &optional message) + ;; Change the full context string of the index entry at point to + ;; NEW. This actually edits the buffer where the entry is defined. + + (let* ((data (get-text-property (point) :data)) + old beg end info) + (unless data (error "Cannot change entry")) + (reftex-index-view-entry) + (setq beg (match-beginning 0) end (match-end 0)) + (setq old (nth 2 data)) + (and (equal old new) (error "Entry unchanged")) + (save-excursion + (set-buffer (get-file-buffer (nth 3 data))) + (goto-char beg) + (unless (looking-at (regexp-quote old)) + (error "This should not happen (reftex-index-change-entry)")) + (delete-region beg end) + (insert new) + (goto-char (1- beg)) + (when (and (re-search-forward (reftex-everything-regexp) nil t) + (match-end 10) + (< (abs (- (match-beginning 10) beg)) (length new)) + (setq info (reftex-index-info-safe buffer-file-name))) + (setcdr data (cdr info)))) + (let ((buffer-read-only nil)) + (save-excursion + (reftex-insert-index (list data) reftex-index-tag t + "EDITED"))) + (setq reftex-last-follow-point 1) + (and message (message message)))) + +;; Index map +(define-key reftex-index-map (if (featurep 'xemacs) [(button2)] [(mouse-2)]) + 'reftex-index-mouse-goto-line-and-hide) + +(substitute-key-definition + 'next-line 'reftex-index-next reftex-index-map global-map) +(substitute-key-definition + 'previous-line 'reftex-index-previous reftex-index-map global-map) + +(loop for x in + '(("n" . reftex-index-next) + ("p" . reftex-index-previous) + ("?" . reftex-index-show-help) + (" " . reftex-index-view-entry) + ("\C-m" . reftex-index-goto-entry-and-hide) + ("\C-i" . reftex-index-goto-entry) + ("\C-k" . reftex-index-kill) + ("r" . reftex-index-rescan) + ("R" . reftex-index-Rescan) + ("g" . revert-buffer) + ("q" . reftex-index-quit) + ("k" . reftex-index-quit-and-kill) + ("f" . reftex-index-toggle-follow) + ("s" . reftex-index-switch-index-tag) + ("e" . reftex-index-edit) + ("^" . reftex-index-level-up) + ("_" . reftex-index-level-down) + ("}" . reftex-index-restrict-to-section) + ("{" . reftex-index-widen) + (">" . reftex-index-restriction-forward) + ("<" . reftex-index-restriction-backward) + ("(" . reftex-index-toggle-range-beginning) + (")" . reftex-index-toggle-range-end) + ("|" . reftex-index-edit-attribute) + ("@" . reftex-index-edit-visual) + ("*" . reftex-index-edit-key) + ("&" . reftex-index-globalize) + ("\C-c=". reftex-index-goto-toc) + ("c" . reftex-index-toggle-context)) + do (define-key reftex-index-map (car x) (cdr x))) + +(loop for key across "0123456789" do + (define-key reftex-index-map (vector (list key)) 'digit-argument)) +(define-key reftex-index-map "-" 'negative-argument) + +;; The capital letters and the exclamation mark +(loop for key across (concat "!" reftex-index-section-letters) do + (define-key reftex-index-map (vector (list key)) + (list 'lambda '() '(interactive) + (list 'reftex-index-goto-letter key)))) + +(defun reftex-index-goto-letter (char) + "Go to the CHAR section in the index." + (let ((pos (point)) + (case-fold-search nil)) + (goto-line 3) + (if (re-search-forward (concat "^" (char-to-string char)) nil t) + (progn + (beginning-of-line) + (recenter 0) + (reftex-index-next)) + (goto-char pos) + (error "This <%s> index does not contain entries starting with `%c'" + reftex-index-tag char)))) + +(easy-menu-define + reftex-index-menu reftex-index-map + "Menu for Index buffer" + `("Index" + ["Goto section A-Z" + (message "To go to a section, just press any of: !%s" + reftex-index-section-letters) t] + ["Show Entry" reftex-index-view-entry t] + ["Go To Entry" reftex-index-goto-entry t] + ["Exit & Go To Entry" reftex-index-goto-entry-and-hide t] + ["Table of Contents" reftex-index-goto-toc t] + ["Quit" reftex-index-quit t] + "--" + ("Update" + ["Rebuilt *Index* Buffer" revert-buffer t] + "--" + ["Rescan One File" reftex-index-rescan reftex-enable-partial-scans] + ["Rescan Entire Document" reftex-index-Rescan t]) + ("Restrict" + ["Restrict to section" reftex-index-restrict-to-section t] + ["Widen" reftex-index-widen reftex-index-restriction-indicator] + ["Next Section" reftex-index-restriction-forward + reftex-index-restriction-indicator] + ["Previous Section" reftex-index-restriction-backward + reftex-index-restriction-indicator]) + ("Edit" + ["Edit Entry" reftex-index-edit t] + ["Edit Key" reftex-index-edit-key t] + ["Edit Attribute" reftex-index-edit-attribute t] + ["Edit Visual" reftex-index-edit-visual t] + "--" + ["Add Parentkey" reftex-index-level-down t] + ["Remove Parentkey " reftex-index-level-up t] + "--" + ["Make Start-of-Range" reftex-index-toggle-range-beginning t] + ["Make End-of-Range" reftex-index-toggle-range-end t] + "--" + ["Globalize" reftex-index-globalize t] + ["Kill Entry" reftex-index-kill nil] + "--" + ["Undo" reftex-index-undo nil]) + ("Options" + ["Context" reftex-index-toggle-context :style toggle + :selected reftex-index-include-context] + "--" + ["Follow Mode" reftex-index-toggle-follow :style toggle + :selected reftex-index-follow-mode]) + "--" + ["Help" reftex-index-show-help t])) + +;;; reftex-index.el ends here diff --git a/lisp/textmodes/reftex-parse.el b/lisp/textmodes/reftex-parse.el new file mode 100644 index 00000000000..8b69efe9de0 --- /dev/null +++ b/lisp/textmodes/reftex-parse.el @@ -0,0 +1,987 @@ +;;; reftex-parse.el - Parser Functions for RefTeX +;;; Version: 4.5 +;;; +;;; See main file reftex.el for licensing information + +(provide 'reftex-parse) +(require 'reftex) + +(defmacro reftex-with-special-syntax (&rest body) + `(let ((saved-syntax (syntax-table))) + (unwind-protect + (progn + (set-syntax-table reftex-syntax-table) + ,@body) + (set-syntax-table saved-syntax)))) + +(defun reftex-parse-one () + "Re-parse this file." + (interactive) + (let ((reftex-enable-partial-scans t)) + (reftex-access-scan-info '(4)))) + +(defun reftex-parse-all () + "Re-parse entire document." + (interactive) + (reftex-access-scan-info '(16))) + +(defun reftex-do-parse (rescan &optional file) + "Do a document rescan. When allowed, do only a partial scan from FILE." + + ;; Normalize the rescan argument + (setq rescan (cond ((eq rescan t) t) + ((eq rescan 1) 1) + ((equal rescan '(4)) t) + ((equal rescan '(16)) 1) + (t 1))) + + ;; Partial scans only when allowed + (unless reftex-enable-partial-scans + (setq rescan 1)) + + ;; Do the scanning. + + (let* ((old-list (symbol-value reftex-docstruct-symbol)) + (master (reftex-TeX-master-file)) + (true-master (file-truename master)) + (master-dir (file-name-as-directory (file-name-directory master))) + (file (or file (buffer-file-name))) + (true-file (file-truename file)) + (bibview-cache (assq 'bibview-cache old-list)) + (index-tags (cdr (assq 'index-tags old-list))) + from-file appendix docstruct tmp) + + ;; Make sure replacement is really an option here + (when (and (eq rescan t) + (not (and (member (list 'bof file) old-list) + (member (list 'eof file) old-list)))) + ;; Scan whole document because no such file section exists + (setq rescan 1)) + (when (string= true-file true-master) + ;; Scan whole document because this file is the master + (setq rescan 1)) + + ;; From which file do we start? + (setq from-file + (cond ((eq rescan t) (or file master)) + ((eq rescan 1) master) + (t (error "This should not happen (reftex-do-parse)")))) + + ;; Reset index-tags if we scan everything + (if (equal rescan 1) (setq index-tags nil)) + + ;; Find active toc entry and initialize section-numbers + (setq reftex-active-toc (reftex-last-assoc-before-elt + 'toc (list 'bof from-file) old-list) + appendix (reftex-last-assoc-before-elt + 'appendix (list 'bof from-file) old-list)) + + (reftex-init-section-numbers reftex-active-toc appendix) + + (if (eq rescan 1) + (message "Scanning entire document...") + (message "Scanning document from %s..." from-file)) + + (reftex-with-special-syntax + (save-window-excursion + (save-excursion + (unwind-protect + (setq docstruct + (reftex-parse-from-file + from-file docstruct master-dir)) + (reftex-kill-temporary-buffers))))) + + (message "Scanning document... done") + + ;; Turn the list around. + (setq docstruct (nreverse docstruct)) + + ;; Set or insert + (setq docstruct (reftex-replace-label-list-segment + old-list docstruct (eq rescan 1))) + + ;; Add all missing information + (unless (assq 'label-numbers docstruct) + (push (cons 'label-numbers nil) docstruct)) + (unless (assq 'master-dir docstruct) + (push (cons 'master-dir master-dir) docstruct)) + (unless (assq 'bibview-cache docstruct) + (push (cons 'bibview-cache (cdr bibview-cache)) docstruct)) + (let* ((bof1 (memq (assq 'bof docstruct) docstruct)) + (bof2 (assq 'bof (cdr bof1))) + (is-multi (not (not (and bof1 bof2)))) + (entry (or (assq 'is-multi docstruct) + (car (push (list 'is-multi is-multi) docstruct))))) + (setcdr entry (cons is-multi nil))) + (and index-tags (setq index-tags (sort index-tags 'string<))) + (let ((index-tag-cell (assq 'index-tags docstruct))) + (if index-tag-cell + (setcdr index-tag-cell index-tags) + (push (cons 'index-tags index-tags) docstruct))) + (unless (assq 'xr docstruct) + (let* ((allxr (reftex-all-assq 'xr-doc docstruct)) + (alist (mapcar + (lambda (x) + (if (setq tmp (reftex-locate-file (nth 2 x) "tex" + master-dir)) + (cons (nth 1 x) tmp) + (message "Can't find external document %s" + (nth 2 x)) + nil)) + allxr)) + (alist (delq nil alist)) + (allprefix (delq nil (mapcar 'car alist))) + (regexp (if allprefix + (concat "\\`\\(" + (mapconcat 'identity allprefix "\\|") + "\\)") + "\\\\\\\\\\\\"))) ; this will never match + (push (list 'xr alist regexp) docstruct))) + + (set reftex-docstruct-symbol docstruct) + (put reftex-docstruct-symbol 'modified t))) + +(defun reftex-everything-regexp () + (if reftex-support-index + reftex-everything-regexp + reftex-everything-regexp-no-index)) + +(defun reftex-all-document-files (&optional relative) + "Return a list of all files belonging to the current document. +When RELATIVE is non-nil, give file names relative to directory +of master file." + (let* ((all (symbol-value reftex-docstruct-symbol)) + (master-dir (file-name-directory (reftex-TeX-master-file))) + (re (concat "\\`" (regexp-quote master-dir))) + file-list tmp file) + (while (setq tmp (assoc 'bof all)) + (setq file (nth 1 tmp) + all (cdr (memq tmp all))) + (and relative + (string-match re file) + (setq file (substring file (match-end 0)))) + (push file file-list)) + (nreverse file-list))) + +(defun reftex-parse-from-file (file docstruct master-dir) + ;; Scan the buffer for labels and save them in a list. + (let ((regexp (reftex-everything-regexp)) + (bound 0) + file-found tmp include-file + (level 1) + (highest-level 100) + toc-entry index-entry next-buf buf) + + (catch 'exit + (setq file-found (reftex-locate-file file "tex" master-dir)) + (if (and (not file-found) + (setq buf (reftex-get-buffer-visiting file))) + (setq file-found (buffer-file-name buf))) + + (unless file-found + (push (list 'file-error file) docstruct) + (throw 'exit nil)) + + (save-excursion + + (message "Scanning file %s" file) + (set-buffer + (setq next-buf + (reftex-get-file-buffer-force + file-found + (not (eq t reftex-keep-temporary-buffers))))) + + ;; Begin of file mark + (setq file (buffer-file-name)) + (push (list 'bof file) docstruct) + + (reftex-with-special-syntax + (save-excursion + (save-restriction + (widen) + (goto-char 1) + + (while (re-search-forward regexp nil t) + + (cond + + ((match-end 1) + ;; It is a label + (push (reftex-label-info (reftex-match-string 1) file bound) + docstruct)) + + ((match-end 3) + ;; It is a section + (setq bound (point)) + + ;; Insert in List + (setq toc-entry (reftex-section-info file)) + (setq level (nth 5 toc-entry)) + (setq highest-level (min highest-level level)) + (if (= level highest-level) + (message + "Scanning %s %s ..." + (car (rassoc level reftex-section-levels-all)) + (nth 6 toc-entry))) + + (push toc-entry docstruct) + (setq reftex-active-toc toc-entry)) + + ((match-end 7) + ;; It's an include or input + (setq include-file (reftex-match-string 7)) + ;; Test if this file should be ignored + (unless (delq nil (mapcar + (lambda (x) (string-match x include-file)) + reftex-no-include-regexps)) + ;; Parse it + (setq docstruct + (reftex-parse-from-file + include-file + docstruct master-dir)))) + + ((match-end 9) + ;; Appendix starts here + (reftex-init-section-numbers nil t) + (push (cons 'appendix t) docstruct)) + + ((match-end 10) + ;; Index entry + (when reftex-support-index + (setq index-entry (reftex-index-info file)) + (when index-entry + (add-to-list 'index-tags (nth 1 index-entry)) + (push index-entry docstruct)))) + + ((match-end 11) + ;; A macro with label + (save-excursion + (let* ((mac (reftex-match-string 11)) + (label (progn (goto-char (match-end 11)) + (save-match-data + (reftex-no-props + (reftex-nth-arg-wrapper + mac))))) + (typekey (nth 1 (assoc mac reftex-env-or-mac-alist))) + (entry (progn (if typekey + ;; A typing macro + (goto-char (match-end 0)) + ;; A neutral macro + (goto-char (match-end 11)) + (reftex-move-over-touching-args)) + (reftex-label-info + label file bound nil nil)))) + (push entry docstruct)))) + (t (error "This should not happen (reftex-parse-from-file)"))) + ) + + ;; Find bibliography statement + (when (setq tmp (reftex-locate-bibliography-files master-dir)) + (push (cons 'bib tmp) docstruct)) + + (goto-char 1) + (when (re-search-forward + "\\(\\`\\|[\n\r]\\)[ \t]*\\\\begin{thebibliography}" nil t) + (push (cons 'thebib file) docstruct)) + + ;; Find external document specifications + (goto-char 1) + (while (re-search-forward "[\n\r][ \t]*\\\\externaldocument\\(\\[\\([^]]*\\)\\]\\)?{\\([^}]+\\)}" nil t) + (push (list 'xr-doc (reftex-match-string 2) + (reftex-match-string 3)) + docstruct)) + + ;; End of file mark + (push (list 'eof file) docstruct))))) + + ;; Kill the scanned buffer + (reftex-kill-temporary-buffers next-buf)) + + ;; Return the list + docstruct)) + +(defun reftex-locate-bibliography-files (master-dir &optional files) + ;; Scan buffer for bibliography macro and return file list. + + (unless files + (save-excursion + (goto-char (point-min)) + (if (re-search-forward + "\\(\\`\\|[\n\r]\\)[ \t]*\\\\bibliography{[ \t]*\\([^}]+\\)" nil t) + (setq files + (split-string (reftex-match-string 2) + "[ \t\n\r]*,[ \t\n\r]*"))))) + (when files + (setq files + (mapcar + (lambda (x) + (if (or (member x reftex-bibfile-ignore-list) + (delq nil (mapcar (lambda (re) (string-match re x)) + reftex-bibfile-ignore-regexps))) + ;; excluded file + nil + ;; find the file + (reftex-locate-file x "bib" master-dir))) + files)) + (delq nil files))) + +(defun reftex-replace-label-list-segment (old insert &optional entirely) + ;; Replace the segment in OLD which corresponds to INSERT. + ;; Works with side effects, directly changes old. + ;; If entirely is t, just return INSERT. + ;; This function also makes sure the old toc markers do not point anywhere. + + (cond + (entirely + (reftex-silence-toc-markers old (length old)) + insert) + (t (let* ((new old) + (file (nth 1 (car insert))) + (eof-list (member (list 'eof file) old)) + (bof-list (member (list 'bof file) old)) + n) + (if (not (and bof-list eof-list)) + (error "Cannot splice") + ;; Splice + (reftex-silence-toc-markers bof-list (- (length bof-list) + (length eof-list))) + (setq n (- (length old) (length bof-list))) + (setcdr (nthcdr n new) (cdr insert)) + (setcdr (nthcdr (1- (length new)) new) (cdr eof-list))) + new)))) + +(defun reftex-section-info (file) + ;; Return a section entry for the current match. + ;; Carefull: This function expects the match-data to be still in place! + (let* ((marker (set-marker (make-marker) (1- (match-beginning 3)))) + (macro (reftex-match-string 3)) + (level (cdr (assoc macro reftex-section-levels-all))) + (star (= ?* (char-after (match-end 3)))) + (unnumbered (or star (< level 0))) + (level (abs level)) + (section-number (reftex-section-number level unnumbered)) + (text1 (save-match-data (save-excursion (reftex-context-substring)))) + (literal (buffer-substring-no-properties + (1- (match-beginning 3)) + (min (point-max) (+ (match-end 0) (length text1) 1)))) + ;; Literal can be too short since text1 too short. No big problem. + (text (reftex-nicify-text text1))) + + ;; Add section number and indentation + (setq text + (concat + (make-string (* reftex-level-indent level) ?\ ) + (if (nth 1 reftex-label-menu-flags) ; section number flag + (concat section-number " ")) + text)) + (list 'toc "toc" text file marker level section-number + literal (marker-position marker)))) + +(defun reftex-ensure-index-support (&optional abort) + ;; When index support is turned off, ask to turn it on and + ;; set the current prefix argument so that `reftex-access-scan-info' + ;; will rescan the entire document. + (cond + (reftex-support-index t) + ((y-or-n-p "Turn on index support and rescan entire document? ") + (setq reftex-support-index 'demanded + current-prefix-arg '(16))) + (t (if abort + (error "No index support") + (message "No index support") + (ding) + (sit-for 1))))) + +(defun reftex-index-info-safe (file) + (reftex-with-special-syntax + (reftex-index-info file))) + +(defvar test-dummy) +(defun reftex-index-info (file) + ;; Return an index entry for the current match. + ;; Carefull: This function expects the match-data to be still in place! + (catch 'exit + (let* ((macro (reftex-match-string 10)) + (bom (match-beginning 10)) + (boa (match-end 10)) + (entry (or (assoc macro reftex-index-macro-alist) + (throw 'exit nil))) + (exclude (nth 3 entry)) + ;; The following is a test if this match should be excluded + (test-dummy (and (fboundp exclude) + (funcall exclude) + (throw 'exit nil))) + (itag (nth 1 entry)) + (prefix (nth 2 entry)) + (index-tag + (cond ((stringp itag) itag) + ((integerp itag) + (progn (goto-char boa) + (or (reftex-nth-arg itag (nth 6 entry)) "idx"))) + (t "idx"))) + (arg (or (progn (goto-char boa) + (reftex-nth-arg (nth 5 entry) (nth 6 entry))) + "")) + (end-of-args (progn (goto-char boa) + (reftex-move-over-touching-args) + (point))) + (end-of-context (progn (skip-chars-forward "^ \t\n\r") (point))) + (begin-of-context + (progn (goto-char bom) + (skip-chars-backward "^ \t\r\n") + (point))) + (context (buffer-substring-no-properties + begin-of-context end-of-context)) + (key-end (if (string-match reftex-index-key-end-re arg) + (1+ (match-beginning 0)))) + (rawkey (substring arg 0 key-end)) + + (key (if prefix (concat prefix rawkey) rawkey)) + (sortkey (downcase key)) + (showkey (mapconcat 'identity + (split-string key reftex-index-level-re) + " ! "))) + (goto-char end-of-args) + ;; 0 1 2 3 4 5 6 7 8 9 + (list 'index index-tag context file bom arg key showkey sortkey key-end)))) + +(defun reftex-short-context (env parse &optional bound derive) + ;; Get about one line of useful context for the label definition at point. + + (if (consp parse) + (setq parse (if derive (cdr parse) (car parse)))) + + (reftex-nicify-text + + (cond + + ((null parse) + (save-excursion + (reftex-context-substring))) + + ((eq parse t) + (if (string= env "section") + ;; special treatment for section labels + (save-excursion + (if (and (re-search-backward reftex-section-or-include-regexp + (point-min) t) + (match-end 2)) + (progn + (goto-char (match-end 0)) + (reftex-context-substring)) + (if reftex-active-toc + (progn + (string-match "{\\([^}]*\\)" (nth 7 reftex-active-toc)) + (match-string 1 (nth 7 reftex-active-toc))) + "SECTION HEADING NOT FOUND"))) + (save-excursion + (goto-char reftex-default-context-position) + (unless (eq (string-to-char env) ?\\) + (reftex-move-over-touching-args)) + (reftex-context-substring)))) + + ((stringp parse) + (save-excursion + (if (re-search-backward parse bound t) + (progn + (goto-char (match-end 0)) + (reftex-context-substring)) + "NO MATCH FOR CONTEXT REGEXP"))) + + ((integerp parse) + (or (save-excursion + (goto-char reftex-default-context-position) + (reftex-nth-arg + parse + (nth 6 (assoc env reftex-env-or-mac-alist)))) + "")) + + ((fboundp parse) + ;; A hook function. Call it. + (save-excursion + (condition-case error-var + (funcall parse env) + (error (format "HOOK ERROR: %s" (cdr error-var)))))) + (t + "ILLEGAL VALUE OF PARSE")))) + +(defun reftex-where-am-I () + ;; Return the docstruct entry above point. Actually returns a cons + ;; cell in which the cdr is a flag indicating if the information is + ;; exact (t) or approximate (nil). + + (let ((docstruct (symbol-value reftex-docstruct-symbol)) + (cnt 0) rtn + found) + (save-excursion + (while (not rtn) + (incf cnt) + (setq found (re-search-backward (reftex-everything-regexp) nil t)) + (setq rtn + (cond + ((not found) + ;; no match + (or + (car (member (list 'bof (buffer-file-name)) docstruct)) + (not (setq cnt 2)) + (assq 'bof docstruct) ;; for safety reasons + 'corrupted)) + ((match-end 1) + ;; Label + (assoc (reftex-match-string 1) + (symbol-value reftex-docstruct-symbol))) + ((match-end 3) + ;; Section + (goto-char (1- (match-beginning 3))) + (let* ((list (member (list 'bof (buffer-file-name)) + docstruct)) + (endelt (car (member (list 'eof (buffer-file-name)) + list))) + rtn1) + (while (and list (not (eq endelt (car list)))) + (if (and (eq (car (car list)) 'toc) + (string= (buffer-file-name) + (nth 3 (car list)))) + (cond + ((equal (point) + (or (and (markerp (nth 4 (car list))) + (marker-position (nth 4 (car list)))) + (nth 8 (car list)))) + ;; Fits with marker position or recorded position + (setq rtn1 (car list) list nil)) + ((looking-at (reftex-make-regexp-allow-for-ctrl-m + (nth 7 (car list)))) + ;; Same title + (setq rtn1 (car list) list nil cnt 2)))) + (pop list)) + rtn1)) + ((match-end 7) + ;; Input or include... + (car + (member (list 'eof (reftex-locate-file + (reftex-match-string 7) "tex" + (cdr (assq 'master-dir docstruct)))) + docstruct))) + ((match-end 9) + (assq 'appendix (symbol-value reftex-docstruct-symbol))) + ((match-end 10) + ;; Index entry + (when reftex-support-index + (let* ((index-info (save-excursion + (reftex-index-info-safe nil))) + (list (member (list 'bof (buffer-file-name)) + docstruct)) + (endelt (car (member (list 'eof (buffer-file-name)) + list))) + dist last-dist last (n 0)) + ;; Check all index entries with equal text + (while (and list (not (eq endelt (car list)))) + (when (and (eq (car (car list)) 'index) + (string= (nth 2 index-info) + (nth 2 (car list)))) + (incf n) + (setq dist (abs (- (point) (nth 4 (car list))))) + (if (or (not last-dist) (< dist last-dist)) + (setq last-dist dist last (car list)))) + (setq list (cdr list))) + ;; We are sure if we have only one, or a zero distance + (cond ((or (= n 1) (= dist 0)) last) + ((> n 1) (setq cnt 2) last) + (t nil))))) + ((match-end 11) + (save-excursion + (goto-char (match-end 11)) + (assoc (reftex-no-props + (reftex-nth-arg-wrapper + (reftex-match-string 11))) + (symbol-value reftex-docstruct-symbol)))) + (t + (error "This should not happen (reftex-where-am-I)")))))) + (cons rtn (eq cnt 1)))) + +(defun reftex-notice-new (&optional n force) + "Hook to handshake with RefTeX after something new has been inserted." + ;; Add a new entry to the docstruct list. If it is a section, renumber + ;; the following sections. + ;; FIXME: Put in a WHAT parameter + ;; When N is given, go back that many matches of reftex-everything-regexp + ;; When FORCE is non-nil, also insert if `reftex-where-am-I' was uncertain. + (condition-case nil + (catch 'exit + (unless reftex-mode (throw 'exit nil)) + (reftex-access-scan-info) + (let* ((docstruct (symbol-value reftex-docstruct-symbol)) + here-I-am appendix tail entry star level + section-number context) + + (save-excursion + (when (re-search-backward (reftex-everything-regexp) nil t (or n 1)) + + ;; Find where we are + (setq here-I-am (reftex-where-am-I)) + (or here-I-am (throw 'exit nil)) + (unless (or force (cdr here-I-am)) (throw 'exit nil)) + (setq tail (memq (car here-I-am) docstruct)) + (or tail (throw 'exit nil)) + (setq reftex-active-toc (reftex-last-assoc-before-elt + 'toc (car here-I-am) docstruct) + appendix (reftex-last-assoc-before-elt + 'appendix (car here-I-am) docstruct)) + + ;; Initialize section numbers + (if (eq (car (car here-I-am)) 'appendix) + (reftex-init-section-numbers nil t) + (reftex-init-section-numbers reftex-active-toc appendix)) + + ;; Match the section command + (when (re-search-forward (reftex-everything-regexp) nil t) + (cond + ((match-end 1) + (push (reftex-label-info (reftex-match-string 1) buffer-file-name) + (cdr tail))) + + ((match-end 3) + (setq star (= ?* (char-after (match-end 3))) + entry (reftex-section-info (buffer-file-name)) + level (nth 5 entry)) + ;; Insert the section info + (push entry (cdr tail)) + + ;; We are done unless we use section numbers + (unless (nth 1 reftex-label-menu-flags) (throw 'exit nil)) + + ;; Update the remaining toc items + (setq tail (cdr tail)) + (while (and (setq tail (memq (assq 'toc (cdr tail)) tail)) + (setq entry (car tail)) + (>= (nth 5 entry) level)) + (setq star (string-match "\\*" (nth 6 entry)) + context (nth 2 entry) + section-number + (reftex-section-number (nth 5 entry) star)) + (when (string-match "\\`\\([ \t]*\\)\\([.0-9A-Z]+\\)\\(.*\\)" + context) + (when (and (not appendix) + (>= (string-to-char (match-string 2)) ?A)) + ;; Just entered the appendex. Get out. + (throw 'exit nil)) + + ;; Change the section number. + (setf (nth 2 entry) + (concat (match-string 1 context) + section-number + (match-string 3 context)))))) + ((match-end 10) + ;; Index entry + (and reftex-support-index + (setq entry (reftex-index-info-safe buffer-file-name)) + ;; FIXME: (add-to-list 'index-tags (nth 1 index-entry)) + (push entry (cdr tail)))))))))) + + (error nil)) + ) + +(defsubst reftex-move-to-previous-arg (&optional bound) + ;; Assuming that we are in front of a macro argument, + ;; move backward to the closing parenthesis of the previous argument. + ;; This function understands the splitting of macros over several lines + ;; in TeX. + (cond + ;; Just to be quick: + ((memq (preceding-char) '(?\] ?\}))) + ;; Do a search + ((and reftex-allow-detached-macro-args + (re-search-backward + "[]}][ \t]*[\n\r]?\\([ \t]*%[^\n\r]*[\n\r]\\)*[ \t]*\\=" bound t)) + (goto-char (1+ (match-beginning 0))) + t) + (t nil))) + +(defun reftex-what-macro-safe (which &optional bound) + ;; reftex-what-macro with special syntax table. + (reftex-with-special-syntax + (reftex-what-macro which bound))) + +(defun reftex-what-macro (which &optional bound) + ;; Find out if point is within the arguments of any TeX-macro. + ;; The return value is either ("\\macro" . (point)) or a list of them. + + ;; If WHICH is nil, immediately return nil. + ;; If WHICH is 1, return innermost enclosing macro. + ;; If WHICH is t, return list of all macros enclosing point. + ;; If WHICH is a list of macros, look only for those macros and return the + ;; name of the first macro in this list found to enclose point. + ;; If the optional BOUND is an integer, bound backwards directed + ;; searches to this point. If it is nil, limit to nearest \section - + ;; like statement. + + ;; This function is pretty stable, but can be fooled if the text contains + ;; things like \macro{aa}{bb} where \macro is defined to take only one + ;; argument. As RefTeX cannot know this, the string "bb" would still be + ;; considered an argument of macro \macro. + + (unless reftex-section-regexp (reftex-compile-variables)) + (catch 'exit + (if (null which) (throw 'exit nil)) + (let ((bound (or bound (save-excursion (re-search-backward + reftex-section-regexp nil 1) + (point)))) + pos cmd-list cmd cnt cnt-opt entry) + (save-restriction + (save-excursion + (narrow-to-region (max 1 bound) (point-max)) + ;; move back out of the current parenthesis + (while (condition-case nil + (progn (up-list -1) t) + (error nil)) + (setq cnt 1 cnt-opt 0) + ;; move back over any touching sexps + (while (and (reftex-move-to-previous-arg bound) + (condition-case nil + (progn (backward-sexp) t) + (error nil))) + (if (eq (following-char) ?\[) (incf cnt-opt)) + (incf cnt)) + (setq pos (point)) + (when (and (or (= (following-char) ?\[) + (= (following-char) ?\{)) + (re-search-backward "\\\\[*a-zA-Z]+\\=" nil t)) + (setq cmd (reftex-match-string 0)) + (when (looking-at "\\\\begin{[^}]*}") + (setq cmd (reftex-match-string 0) + cnt (1- cnt))) + ;; This does ignore optional arguments. Very hard to fix. + (when (setq entry (assoc cmd reftex-env-or-mac-alist)) + (if (> cnt (or (nth 4 entry) 100)) + (setq cmd nil))) + (cond + ((null cmd)) + ((eq t which) + (push (cons cmd (point)) cmd-list)) + ((or (eq 1 which) (member cmd which)) + (throw 'exit (cons cmd (point)))))) + (goto-char pos))) + (nreverse cmd-list))))) + +(defun reftex-what-environment (which &optional bound) + ;; Find out if point is inside a LaTeX environment. + ;; The return value is (e.g.) either ("equation" . (point)) or a list of + ;; them. + + ;; If WHICH is nil, immediately return nil. + ;; If WHICH is 1, return innermost enclosing environment. + ;; If WHICH is t, return list of all environments enclosing point. + ;; If WHICH is a list of environments, look only for those environments and + ;; return the name of the first environment in this list found to enclose + ;; point. + + ;; If the optional BOUND is an integer, bound backwards directed searches to + ;; this point. If it is nil, limit to nearest \section - like statement. + + (unless reftex-section-regexp (reftex-compile-variables)) + (catch 'exit + (save-excursion + (if (null which) (throw 'exit nil)) + (let ((bound (or bound (save-excursion (re-search-backward + reftex-section-regexp nil 1) + (point)))) + env-list end-list env) + (while (re-search-backward "\\\\\\(begin\\|end\\){\\([^}]+\\)}" + bound t) + (setq env (buffer-substring-no-properties + (match-beginning 2) (match-end 2))) + (cond + ((string= (match-string 1) "end") + (push env end-list)) + ((equal env (car end-list)) + (setq end-list (cdr end-list))) + ((eq t which) + (push (cons env (point)) env-list)) + ((or (eq 1 which) (member env which)) + (throw 'exit (cons env (point)))))) + (nreverse env-list))))) + +(defun reftex-what-special-env (which &optional bound) + ;; Run the special environment parsers and return the matches. + ;; + ;; The return value is (e.g.) either ("my-parser-function" . (point)) + ;; or a list of them. + + ;; If WHICH is nil, immediately return nil. + ;; If WHICH is 1, return innermost enclosing environment. + ;; If WHICH is t, return list of all environments enclosing point. + ;; If WHICH is a list of environments, look only for those environments and + ;; return the name of the first environment in this list found to enclose + ;; point. + + (unless reftex-section-regexp (reftex-compile-variables)) + (catch 'exit + (save-excursion + (if (null reftex-special-env-parsers) (throw 'exit nil)) + (if (null which) (throw 'exit nil)) + (let ((bound (or bound (save-excursion (re-search-backward + reftex-section-regexp nil 1) + (point)))) + (fun-list (if (listp which) + (mapcar (lambda (x) (if (memq x which) x nil)) + reftex-special-env-parsers) + reftex-special-env-parsers)) + specials rtn) + ;; Call all functions + (setq specials (mapcar + (lambda (fun) + (save-excursion + (setq rtn (and fun (funcall fun bound))) + (if rtn (cons (symbol-name fun) rtn) nil))) + fun-list)) + ;; Delete the non-matches + (setq specials (delq nil specials)) + ;; Sort + (setq specials (sort specials (lambda (a b) (> (cdr a) (cdr b))))) + (if (eq which t) + specials + (car specials)))))) + +(defsubst reftex-move-to-next-arg (&optional ignore) + ;; Assuming that we are at the end of a macro name or a macro argument, + ;; move forward to the opening parenthesis of the next argument. + ;; This function understands the splitting of macros over several lines + ;; in TeX. + (cond + ;; Just to be quick: + ((memq (following-char) '(?\[ ?\{))) + ;; Do a search + ((and reftex-allow-detached-macro-args + (looking-at "[ \t]*[\n\r]?\\([ \t]*%[^\n\r]*[\n\r]\\)*[ \t]*[[{]")) + (goto-char (1- (match-end 0))) + t) + (t nil))) + +(defun reftex-nth-arg-wrapper (key) + (let ((entry (assoc key reftex-env-or-mac-alist))) + (reftex-nth-arg (nth 5 entry) (nth 6 entry)))) + +(defun reftex-nth-arg (n &optional opt-args) + ;; Return the nth following {} or [] parentheses content. + ;; OPT-ARGS is a list of argument numbers which are optional. + + ;; If we are sitting at a macro start, skip to end of macro name. + (and (eq (following-char) ?\\) (skip-chars-forward "a-zA-Z*\\\\")) + + (if (= n 1000) + ;; Special case: Skip all touching arguments + (progn + (reftex-move-over-touching-args) + (reftex-context-substring)) + + ;; Do the real thing. + (let ((cnt 1)) + + (when (reftex-move-to-next-arg) + + (while (< cnt n) + (while (and (member cnt opt-args) + (eq (following-char) ?\{)) + (incf cnt)) + (when (< cnt n) + (unless (and (condition-case nil + (or (forward-list 1) t) + (error nil)) + (reftex-move-to-next-arg) + (incf cnt)) + (setq cnt 1000)))) + + (while (and (memq cnt opt-args) + (eq (following-char) ?\{)) + (incf cnt))) + (if (and (= n cnt) + (> (skip-chars-forward "{\\[") 0)) + (reftex-context-substring) + nil)))) + +(defun reftex-move-over-touching-args () + (condition-case nil + (while (memq (following-char) '(?\[ ?\{)) + (forward-list 1)) + (error nil))) + +(defun reftex-context-substring () + ;; Return up to 150 chars from point + ;; When point is just after a { or [, limit string to matching parenthesis + (cond + ((or (= (preceding-char) ?\{) + (= (preceding-char) ?\[)) + ;; Inside a list - get only the list. + (buffer-substring-no-properties + (point) + (min (+ (point) 150) + (point-max) + (condition-case nil + (progn + (up-list 1) + (1- (point))) + (error (point-max)))))) + (t + ;; no list - just grab 150 characters + (buffer-substring-no-properties (point) + (min (+ (point) 150) (point-max)))))) + +;; Variable holding the vector with section numbers +(defvar reftex-section-numbers [0 0 0 0 0 0 0 0]) + +(defun reftex-init-section-numbers (&optional toc-entry appendix) + ;; Initialize the section numbers with zeros or with what is found + ;; in the toc entry. + (let* ((level (or (nth 5 toc-entry) -1)) + (numbers (nreverse (split-string (or (nth 6 toc-entry) "") "\\."))) + (depth (1- (length reftex-section-numbers))) + (i depth) number-string) + (while (>= i 0) + (if (> i level) + (aset reftex-section-numbers i 0) + (setq number-string (or (car numbers) "0")) + (if (string-match "\\`[A-Z]\\'" number-string) + (aset reftex-section-numbers i + (- (string-to-char number-string) ?A -1)) + (aset reftex-section-numbers i (string-to-int number-string))) + (pop numbers)) + (decf i))) + (put 'reftex-section-numbers 'appendix appendix)) + +(defun reftex-section-number (&optional level star) + ;; Return a string with the current section number. + ;; When LEVEL is non-nil, increase section numbers on that level. + (let* ((depth (1- (length reftex-section-numbers))) idx n (string "") + (appendix (get 'reftex-section-numbers 'appendix))) + (when level + (when (and (> level -1) (not star)) + (aset reftex-section-numbers + level (1+ (aref reftex-section-numbers level)))) + (setq idx (1+ level)) + (when (not star) + (while (<= idx depth) + (aset reftex-section-numbers idx 0) + (incf idx)))) + (setq idx 0) + (while (<= idx depth) + (setq n (aref reftex-section-numbers idx)) + (setq string (concat string (if (not (string= string "")) "." "") + (int-to-string n))) + (incf idx)) + (save-match-data + (if (string-match "\\`\\([@0]\\.\\)+" string) + (setq string (replace-match "" nil nil string))) + (if (string-match "\\(\\.0\\)+\\'" string) + (setq string (replace-match "" nil nil string))) + (if (and appendix + (string-match "\\`[0-9]+" string)) + (setq string + (concat + (char-to-string + (1- (+ ?A (string-to-int (match-string 0 string))))) + (substring string (match-end 0)))))) + (if star + (concat (make-string (1- (length string)) ?\ ) "*") + string))) + +;;; reftex-parse.el ends here diff --git a/lisp/textmodes/reftex-ref.el b/lisp/textmodes/reftex-ref.el new file mode 100644 index 00000000000..8dc77013566 --- /dev/null +++ b/lisp/textmodes/reftex-ref.el @@ -0,0 +1,776 @@ +;;; reftex-ref.el - Code to create labels and references with RefTeX +;;; Version: 4.5 +;;; +;;; See main file reftex.el for licensing information + +(provide 'reftex-ref) +(require 'reftex) +;;; + +(defun reftex-label-location (&optional bound) + "Return the environment or macro which determines the label type at point. +If optional BOUND is an integer, limit backward searches to that point." + + (let* ((loc1 (reftex-what-macro reftex-label-mac-list bound)) + (loc2 (reftex-what-environment reftex-label-env-list bound)) + (loc3 (reftex-what-special-env 1 bound)) + (p1 (or (cdr loc1) 0)) + (p2 (or (cdr loc2) 0)) + (p3 (or (cdr loc3) 0)) + (pmax (max p1 p2 p3))) + + (setq reftex-location-start pmax) + (cond + ((= p1 pmax) + ;; A macro. Default context after macro name. + (setq reftex-default-context-position (+ p1 (length (car loc1)))) + (or (car loc1) "section")) + ((= p2 pmax) + ;; An environment. Default context after \begin{name}. + (setq reftex-default-context-position (+ p2 8 (length (car loc2)))) + (or (car loc2) "section")) + ((= p3 pmax) + ;; A special. Default context right there. + (setq reftex-default-context-position p3) + (setq loc3 (car loc3)) + (cond ((null loc3) "section") + ((symbolp loc3) (symbol-name loc3)) + ((stringp loc3) loc3) + (t "section"))) + (t ;; This should not happen, I think? + "section")))) + +(defun reftex-label-info-update (cell) + ;; Update information about just one label in a different file. + ;; CELL contains the old info list + (let* ((label (nth 0 cell)) + (typekey (nth 1 cell)) + ;; (text (nth 2 cell)) + (file (nth 3 cell)) + (comment (nth 4 cell)) + (note (nth 5 cell)) + (buf (reftex-get-file-buffer-force + file (not (eq t reftex-keep-temporary-buffers))))) + (if (not buf) + (list label typekey "" file comment "LOST LABEL. RESCAN TO FIX.") + (save-excursion + (set-buffer buf) + (save-restriction + (widen) + (goto-char 1) + + (if (or (re-search-forward + (format reftex-find-label-regexp-format + (regexp-quote label)) nil t) + (re-search-forward + (format reftex-find-label-regexp-format2 + (regexp-quote label)) nil t)) + + (progn + (backward-char 1) + (append (reftex-label-info label file) (list note))) + (list label typekey "" file "LOST LABEL. RESCAN TO FIX."))))))) + +(defun reftex-label-info (label &optional file bound derive env-or-mac) + ;; Return info list on LABEL at point. + (let* ((env-or-mac (or env-or-mac (reftex-label-location bound))) + (typekey (nth 1 (assoc env-or-mac reftex-env-or-mac-alist))) + (file (or file (buffer-file-name))) + (parse (nth 2 (assoc env-or-mac reftex-env-or-mac-alist))) + (text (reftex-short-context env-or-mac parse reftex-location-start + derive)) + (in-comment (reftex-in-comment))) + (list label typekey text file in-comment))) + +;;; Creating labels --------------------------------------------------------- + +(defun reftex-label (&optional environment no-insert) + "Insert a unique label. Return the label. +If ENVIRONMENT is given, don't bother to find out yourself. +If NO-INSERT is non-nil, do not insert label into buffer. +With prefix arg, force to rescan document first. +When you are prompted to enter or confirm a label, and you reply with +just the prefix or an empty string, no label at all will be inserted. +A new label is also recorded into the label list. +This function is controlled by the settings of reftex-insert-label-flags." + + (interactive) + + ;; Ensure access to scanning info and rescan buffer if prefix are is '(4). + (reftex-access-scan-info current-prefix-arg) + + ;; Find out what kind of environment this is and abort if necessary. + (if (or (not environment) + (not (assoc environment reftex-env-or-mac-alist))) + (setq environment (reftex-label-location))) + (unless environment + (error "Can't figure out what kind of label should be inserted")) + + ;; Ok, go ahead. + (catch 'exit + (let* ((entry (assoc environment reftex-env-or-mac-alist)) + (typekey (nth 1 entry)) + (format (nth 3 entry)) + (macro-cell (reftex-what-macro 1)) + (entry1 (assoc (car macro-cell) reftex-env-or-mac-alist)) + label naked prefix valid default force-prompt rescan-is-useful) + (when (and (or (nth 5 entry) (nth 5 entry1)) + (memq (preceding-char) '(?\[ ?\{))) + ;; This is an argument of a label macro. Insert naked label. + (setq naked t format "%s")) + + (setq prefix (or (cdr (assoc typekey reftex-typekey-to-prefix-alist)) + (concat typekey "-"))) + ;; Replace any escapes in the prefix + (setq prefix (reftex-replace-prefix-escapes prefix)) + + ;; Make a default label. + (cond + + ((reftex-typekey-check typekey (nth 0 reftex-insert-label-flags)) + ;; Derive a label from context. + (setq reftex-active-toc (reftex-last-assoc-before-elt + 'toc (car (reftex-where-am-I)) + (symbol-value reftex-docstruct-symbol))) + (setq default (reftex-no-props + (nth 2 (reftex-label-info " " nil nil t)))) + ;; Catch the cases where the is actually no context available. + (if (or (string-match "NO MATCH FOR CONTEXT REGEXP" default) + (string-match "ILLEGAL VALUE OF PARSE" default) + (string-match "SECTION HEADING NOT FOUND" default) + (string-match "HOOK ERROR" default) + (string-match "^[ \t]*$" default)) + (setq default prefix + force-prompt t) ; need to prompt + (setq default + (concat prefix + (funcall reftex-string-to-label-function default))) + + ;; Make it unique. + (setq default (reftex-uniquify-label default nil "-")))) + + ((reftex-typekey-check typekey (nth 1 reftex-insert-label-flags)) + ;; Minimal default: the user will be prompted. + (setq default prefix)) + + (t + ;; Make an automatic label. + (setq default (reftex-uniquify-label prefix t)))) + + ;; Should we ask the user? + (if (or (reftex-typekey-check typekey + (nth 1 reftex-insert-label-flags)) ; prompt + force-prompt) + + (while (not valid) + ;; iterate until we get a legal label + + (setq label (read-string + (if naked "Naked Label: " "Label: ") + default)) + + ;; Lets make sure that this is a legal label + (cond + + ((string-match (concat "\\`\\(" (regexp-quote prefix) + "\\)?[ \t]*\\'") + label) + ;; No label at all, please + (message "No label inserted.") + (throw 'exit nil)) + + ;; Test if label contains strange characters + ((string-match reftex-label-illegal-re label) + (message "Label \"%s\" contains illegal characters" label) + (ding) + (sit-for 2)) + + ;; Look it up in the label list + ((setq entry (assoc label + (symbol-value reftex-docstruct-symbol))) + (ding) + (if (y-or-n-p + (format "Label '%s' exists. Use anyway? " label)) + (setq valid t))) + + ;; Label is ok + (t + (setq valid t)))) + (setq label default)) + + ;; Insert the label into the label list + (let* ((here-I-am-info + (save-excursion + (if (and (or naked no-insert) + (integerp (cdr macro-cell))) + (goto-char (cdr macro-cell))) + (reftex-where-am-I))) + (here-I-am (car here-I-am-info)) + (note (if (cdr here-I-am-info) + "" + "POSITION UNCERTAIN. RESCAN TO FIX.")) + (file (buffer-file-name)) + (text nil) + (tail (memq here-I-am (symbol-value reftex-docstruct-symbol)))) + + (or (cdr here-I-am-info) (setq rescan-is-useful t)) + + (when tail + (push (list label typekey text file nil note) (cdr tail)) + (put reftex-docstruct-symbol 'modified t))) + + ;; Insert the label into the buffer + (unless no-insert + (insert + (if reftex-format-label-function + (funcall reftex-format-label-function label format) + (format format label))) + (if (and reftex-plug-into-AUCTeX + (fboundp 'LaTeX-add-labels)) + ;; Tell AUCTeX about this + (LaTeX-add-labels label))) + + ;; Delete the corresponding selection buffers to force update on next use. + (when reftex-auto-update-selection-buffers + (reftex-erase-buffer (reftex-make-selection-buffer-name typekey)) + (reftex-erase-buffer (reftex-make-selection-buffer-name " "))) + + (when (and rescan-is-useful reftex-allow-automatic-rescan) + (reftex-parse-one)) + + ;; return value of the function is the label + label))) + +(defun reftex-string-to-label (string) + "Convert a string (a sentence) to a label. +Uses `reftex-derive-label-parameters' and `reftex-label-illegal-re'. It +also applies `reftex-translate-to-ascii-function' to the string." + (when (and reftex-translate-to-ascii-function + (fboundp reftex-translate-to-ascii-function)) + (setq string (funcall reftex-translate-to-ascii-function string))) + (apply 'reftex-convert-string string + "[-~ \t\n\r,;]+" reftex-label-illegal-re nil nil + reftex-derive-label-parameters)) + +(defun reftex-latin1-to-ascii (string) + ;; Translate the upper 128 chars in the Latin-1 charset to ASCII equivalents + (let ((tab "@@@@@@@@@@@@@@@@@@'@@@@@@@@@@@@@ icLxY|S\"ca<--R-o|23'uq..1o>423?AAAAAAACEEEEIIIIDNOOOOOXOUUUUYP3aaaaaaaceeeeiiiidnooooo:ouuuuypy") + (emacsp (not (featurep 'xemacs)))) + (mapconcat + (lambda (c) + (cond ((and (> c 127) (< c 256)) ; 8 bit Latin-1 + (char-to-string (aref tab (- c 128)))) + ((and emacsp ; Not for XEmacs + (> c 2175) (< c 2304)) ; Mule Latin-1 + (char-to-string (aref tab (- c 2176)))) + (t (char-to-string c)))) + string ""))) + +(defun reftex-replace-prefix-escapes (prefix) + ;; Replace %escapes in a label prefix + (save-match-data + (let (letter (num 0) replace) + (while (string-match "\\%\\([a-zA-Z]\\)" prefix num) + (setq letter (match-string 1 prefix)) + (setq replace + (cond + ((equal letter "f") + (file-name-sans-extension + (file-name-nondirectory (buffer-file-name)))) + ((equal letter "F") + (let ((masterdir (file-name-directory (reftex-TeX-master-file))) + (file (file-name-sans-extension (buffer-file-name)))) + (if (string-match (concat "\\`" (regexp-quote masterdir)) + file) + (substring file (length masterdir)) + file))) + ((equal letter "u") + (or (user-login-name) "")) + ((equal letter "S") + (let* (macro level) + (save-excursion + (save-match-data + (when (re-search-backward reftex-section-regexp nil t) + (setq macro (reftex-match-string 2) + level + (abs + (cdr (assoc macro reftex-section-levels-all))))) + (cdr (or (assoc macro reftex-section-prefixes) + (assoc level reftex-section-prefixes) + (assq t reftex-section-prefixes) + (list t "sec:"))))))) + (t ""))) + (setq num (1- (+ (match-beginning 1) (length replace))) + prefix (replace-match replace nil nil prefix))) + prefix))) + +(defun reftex-uniquify-label (label &optional force separator) + ;; Make label unique by appending a number. + ;; Optional FORCE means, force appending a number, even if label is unique. + ;; Optional SEPARATOR is a string to stick between label and number. + + ;; Ensure access to scanning info + (reftex-access-scan-info) + + (cond + ((and (not force) + (not (assoc label (symbol-value reftex-docstruct-symbol)))) + label) + (t + (let* ((label-numbers (assq 'label-numbers + (symbol-value reftex-docstruct-symbol))) + (label-numbers-alist (cdr label-numbers)) + (cell (or (assoc label label-numbers-alist) + (car (setcdr label-numbers + (cons (cons label 0) + label-numbers-alist))))) + (num (1+ (cdr cell))) + (sep (or separator ""))) + (while (assoc (concat label sep (int-to-string num)) + (symbol-value reftex-docstruct-symbol)) + (incf num)) + (setcdr cell num) + (concat label sep (int-to-string num)))))) + +;;; Referencing labels ------------------------------------------------------ + +;; Help string for the reference label menu +(defconst reftex-select-label-prompt + "Select: [n]ext [p]revious [r]escan [ ]context e[x]tern [q]uit RET [?]HELP+more") + +(defconst reftex-select-label-help + " n / p Go to next/previous label (Cursor motion works as well) + C-c C-n/p Go to next/previous section heading. + b / l Jump back to previous selection / Reuse last referenced label. + g / s Update menu / Switch label type. + r / C-u r Reparse document / Reparse entire document. + x Switch to label menu of external document (with LaTeX package `xr'). + F t c Toggle: [F]ile borders, [t]able of contents, [c]ontext + # % Toggle: [#] label counters, [%] labels in comments + SPC / f Show full context in other window / Toggle follow mode. + . Show insertion point in other window. + v / V Toggle \\ref <-> \\vref / Rotate \\ref <=> \\fref <=> \\Fref + TAB Enter a label with completion. + m , - + Mark entry. `,-+' also assign a separator. + a / A Put all marked entries into one/many \\ref commands. + q / RET Quit without referencing / Accept current label (also on mouse-2).") + +(defun reftex-reference (&optional type no-insert cut) + "Make a LaTeX reference. Look only for labels of a certain TYPE. +With prefix arg, force to rescan buffer for labels. This should only be +necessary if you have recently entered labels yourself without using +reftex-label. Rescanning of the buffer can also be requested from the +label selection menu. +The function returns the selected label or nil. +If NO-INSERT is non-nil, do not insert \\ref command, just return label. +When called with 2 C-u prefix args, disable magic word recognition." + + (interactive) + + ;; check for active recursive edits + (reftex-check-recursive-edit) + + ;; Ensure access to scanning info and rescan buffer if prefix are is '(4) + (reftex-access-scan-info current-prefix-arg) + + (unless type + ;; guess type from context + (if (and reftex-guess-label-type + (setq type (reftex-guess-label-type))) + (setq cut (cdr type) + type (car type)) + (setq type (reftex-query-label-type)))) + + (let* ((refstyle + (cond ((reftex-typekey-check type reftex-vref-is-default) "\\vref") + ((reftex-typekey-check type reftex-fref-is-default) "\\fref") + (t "\\ref"))) + (reftex-format-ref-function reftex-format-ref-function) + (form "\\ref{%s}") + label labels sep sep1) + + ;; Have the user select a label + (set-marker reftex-select-return-marker (point)) + (setq labels (save-excursion + (reftex-offer-label-menu type))) + (reftex-ensure-compiled-variables) + (set-marker reftex-select-return-marker nil) + ;; If the first entry is the symbol 'concat, concat all all labels. + ;; We keep the cdr of the first label for typekey etc information. + (if (eq (car labels) 'concat) + (setq labels (list (list (mapconcat 'car (cdr labels) ",") + (cdr (nth 1 labels)))))) + (setq type (nth 1 (car labels)) + form (or (cdr (assoc type reftex-typekey-to-format-alist)) + form)) + + (cond + (no-insert + ;; Just return the first label + (car (car labels))) + ((null labels) + (message "Quit") + nil) + (t + (while labels + (setq label (car (car labels)) + sep (nth 2 (car labels)) + sep1 (cdr (assoc sep reftex-multiref-punctuation)) + labels (cdr labels)) + (when cut + (backward-delete-char cut) + (setq cut nil)) + + ;; remove ~ if we do already have a space + (when (and (= ?~ (string-to-char form)) + (member (preceding-char) '(?\ ?\t ?\n))) + (setq form (substring form 1))) + ;; do we have a special format? + (setq reftex-format-ref-function + (cond + ((string= refstyle "\\vref") 'reftex-format-vref) + ((string= refstyle "\\fref") 'reftex-format-fref) + ((string= refstyle "\\Fref") 'reftex-format-Fref) + (t reftex-format-ref-function))) + ;; ok, insert the reference + (if sep1 (insert sep1)) + (insert + (if reftex-format-ref-function + (funcall reftex-format-ref-function label form) + (format form label label))) + ;; take out the initial ~ for good + (and (= ?~ (string-to-char form)) + (setq form (substring form 1)))) + (message "") + label)))) + +(defun reftex-guess-label-type () + ;; Examine context to guess what a \ref might want to reference. + (let ((words reftex-words-to-typekey-alist) + (case-fold-search t) + (bound (max (point-min) (- (point) 35))) + matched cell) + (save-excursion + (while (and (setq cell (pop words)) + (not (setq matched + (re-search-backward (car cell) bound t)))))) + (if matched + (cons (cdr cell) (- (match-end 0) (match-end 1))) + nil))) + +(defvar reftex-select-label-map) +(defun reftex-offer-label-menu (typekey) + ;; Offer a menu with the appropriate labels. + (let* ((buf (current-buffer)) + (xr-data (assq 'xr (symbol-value reftex-docstruct-symbol))) + (xr-alist (cons (cons "" (buffer-file-name)) (nth 1 xr-data))) + (xr-index 0) + (here-I-am (car (reftex-where-am-I))) + (here-I-am1 here-I-am) + (toc (reftex-typekey-check typekey reftex-label-menu-flags 0)) + (files (reftex-typekey-check typekey reftex-label-menu-flags 7)) + (context (not (reftex-typekey-check + typekey reftex-label-menu-flags 3))) + (counter (reftex-typekey-check + typekey reftex-label-menu-flags 2)) + (follow (reftex-typekey-check + typekey reftex-label-menu-flags 4)) + (commented (nth 5 reftex-label-menu-flags)) + (prefix "") + selection-buffers + offset rtn key data last-data entries) + + (unwind-protect + (catch 'exit + (while t + (save-window-excursion + (delete-other-windows) + (setq reftex-call-back-to-this-buffer buf + reftex-latex-syntax-table (syntax-table)) + (let ((default-major-mode 'reftex-select-label-mode)) + (if reftex-use-multiple-selection-buffers + (switch-to-buffer-other-window + (save-excursion + (set-buffer buf) + (reftex-make-selection-buffer-name typekey))) + (switch-to-buffer-other-window "*RefTeX Select*") + (reftex-erase-buffer))) + (unless (eq major-mode 'reftex-select-label-mode) + (reftex-select-label-mode)) + (add-to-list 'selection-buffers (current-buffer)) + (setq truncate-lines t) + (setq mode-line-format + (list "---- " 'mode-line-buffer-identification + " " 'global-mode-string " (" mode-name ")" + " S<" 'refstyle ">" + " -%-")) + (cond + ((= 0 (buffer-size)) + (let ((buffer-read-only nil)) + (message "Creating Selection Buffer...") + (setq offset (reftex-insert-docstruct + buf + toc + typekey + nil ; index + files + context + counter + commented + (or here-I-am offset) + prefix + nil ; no a toc buffer + )))) + (here-I-am + (setq offset (reftex-get-offset buf here-I-am typekey))) + (t (setq offset t))) + (setq buffer-read-only t) + (setq offset (or offset t)) + + (setq here-I-am nil) ; turn off determination of offset + (setq rtn + (reftex-select-item + reftex-select-label-prompt + reftex-select-label-help + reftex-select-label-map + offset + 'reftex-show-label-location follow)) + (setq key (car rtn) + data (nth 1 rtn) + last-data (nth 2 rtn) + offset t) + (unless key (throw 'exit nil)) + (cond + ((eq key ?g) + ;; update buffer + (reftex-erase-buffer)) + ((or (eq key ?r) + (eq key ?R)) + ;; rescan buffer + (and current-prefix-arg (setq key ?R)) + (reftex-erase-buffer) + (reftex-reparse-document buf last-data key)) + ((eq key ?c) + ;; toggle context mode + (reftex-erase-buffer) + (setq context (not context))) + ((eq key ?s) + ;; switch type + (setq here-I-am here-I-am1) + (setq typekey (reftex-query-label-type))) + ((eq key ?t) + ;; toggle table of contents display + (reftex-erase-buffer) + (setq toc (not toc))) + ((eq key ?F) + ;; toggle display of included file borders + (reftex-erase-buffer) + (setq files (not files))) + ((eq key ?#) + ;; toggle counter display + (reftex-erase-buffer) + (setq counter (not counter))) + ((eq key ?%) + ;; toggle display of commented labels + (reftex-erase-buffer) + (setq commented (not commented))) + ((eq key ?l) + ;; reuse the last referenced label again + (setq entries reftex-last-used-reference) + (throw 'exit t)) + ((eq key ?x) + ;; select an external document + (setq xr-index (reftex-select-external-document + xr-alist xr-index)) + (setq buf (or (reftex-get-file-buffer-force + (cdr (nth xr-index xr-alist))) + (error "Cannot switch document")) + prefix (or (car (nth xr-index xr-alist)) "")) + (set-buffer buf) + (reftex-access-scan-info)) + ((stringp key) + (setq entries + (list + (list + (or (assoc key (symbol-value reftex-docstruct-symbol)) + (list key typekey))))) + (throw 'exit t)) + ((memq key '(?a ?A return)) + (cond + (reftex-select-marked + (setq entries (nreverse reftex-select-marked))) + (data + (setq entries (list (list data)))) + (t (setq entries nil))) + (when entries + (if (equal key ?a) (push 'concat entries)) + (setq reftex-last-used-reference entries)) + (set-buffer buf) + (throw 'exit t)) + (t (error "This should not happen (reftex-offer-label-menu)")))))) + (save-excursion + (while reftex-buffers-with-changed-invisibility + (set-buffer (car (car reftex-buffers-with-changed-invisibility))) + (setq buffer-invisibility-spec + (cdr (pop reftex-buffers-with-changed-invisibility))))) + (mapcar (lambda (buf) (and (buffer-live-p buf) (bury-buffer buf))) + selection-buffers) + (reftex-kill-temporary-buffers)) + ;; Add the prefixes, put together the relevant information in the form + ;; (LABEL TYPEKEY SEPERATOR) and return a list of those. + (mapcar (lambda (x) + (if (listp x) + (list (concat prefix (car (car x))) + (nth 1 (car x)) + (nth 2 x)) + x)) + entries))) + +(defun reftex-reparse-document (&optional buffer data key) + ;; Rescan the document. + (save-window-excursion + (save-excursion + (if buffer + (if (not (bufferp buffer)) + (error "No such buffer %s" (buffer-name buffer)) + (set-buffer buffer))) + (let ((arg (if (eq key ?R) '(16) '(4))) + (file (nth 3 data))) + (reftex-access-scan-info arg file))))) + +(defun reftex-query-label-type () + ;; Ask for label type + (let ((key (reftex-select-with-char + reftex-type-query-prompt reftex-type-query-help 3))) + (unless (member (char-to-string key) reftex-typekey-list) + (error "No such label type: %s" (char-to-string key))) + (char-to-string key))) + +(defun reftex-show-label-location (data forward no-revisit + &optional stay error) + ;; View the definition site of a label in another window. + ;; DATA is an entry from the docstruct list. + ;; FORWARD indicates if the label is likely forward from current point. + ;; NO-REVISIT means do not load a file to show this label. + ;; STAY means leave the new window selected. + ;; ERROR means throw an error exception when the label cannot be found. + ;; If ERROR is nil, the return value of this function indicates success. + (let* ((this-window (selected-window)) + (errorf (if error 'error 'message)) + label file buffer re found) + + (catch 'exit + (setq label (nth 0 data) + file (nth 3 data)) + + (unless file + (funcall errorf "Unknown label - reparse might help") + (throw 'exit nil)) + + ;; Goto the file in another window + (setq buffer + (if no-revisit + (reftex-get-buffer-visiting file) + (reftex-get-file-buffer-force + file (not reftex-keep-temporary-buffers)))) + (if buffer + ;; good - the file is available + (switch-to-buffer-other-window buffer) + ;; we have got a problem here. The file does not exist. + ;; Let' get out of here.. + (funcall errorf "Label %s not found" label) + (throw 'exit nil)) + + ;; search for that label + (setq re (format reftex-find-label-regexp-format (regexp-quote label))) + (setq found + (if forward + (re-search-forward re nil t) + (re-search-backward re nil t))) + (unless found + (goto-char (point-min)) + (unless (setq found (re-search-forward re nil t)) + ;; Ooops. Must be in a macro with distributed args. + (setq found + (re-search-forward + (format reftex-find-label-regexp-format2 + (regexp-quote label)) nil t)))) + (if (match-end 3) + (progn + (reftex-highlight 0 (match-beginning 3) (match-end 3)) + (reftex-show-entry (match-beginning 3) (match-end 3)) + (recenter '(4)) + (unless stay (select-window this-window))) + (select-window this-window) + (funcall errorf "Label %s not found" label)) + found))) + +(defvar font-lock-mode) +(defun reftex-show-entry (beg-hlt end-hlt) + ;; Show entry if point is hidden + (let* ((n (/ (reftex-window-height) 2)) + (beg (save-excursion + (re-search-backward "[\n\r]" nil 1 n) (point))) + (end (save-excursion + (re-search-forward "[\n\r]" nil 1 n) (point)))) + (cond + ((and (boundp 'buffer-invisibility-spec) buffer-invisibility-spec + (get-char-property (1+ beg-hlt) 'invisible)) + ;; Invisible with text properties. That is easy to change. + (push (cons (current-buffer) buffer-invisibility-spec) + reftex-buffers-with-changed-invisibility) + (setq buffer-invisibility-spec nil)) + ((string-match "\r" (buffer-substring beg end)) + ;; Invisible with selective display. We need to copy it. + (let ((string (buffer-substring-no-properties beg end))) + (switch-to-buffer "*RefTeX Context Copy*") + (setq buffer-read-only nil) + (erase-buffer) + (insert string) + (subst-char-in-region (point-min) (point-max) ?\r ?\n t) + (goto-char (- beg-hlt beg)) + (reftex-highlight 0 (1+ (- beg-hlt beg)) (1+ (- end-hlt beg))) + (if (reftex-refontify) + (when (or (not (eq major-mode 'latex-mode)) + (not font-lock-mode)) + (latex-mode) + (run-hook-with-args + 'reftex-pre-refontification-functions + reftex-call-back-to-this-buffer 'reftex-hidden) + (turn-on-font-lock)) + (when (or (not (eq major-mode 'fundamental-mode)) + font-lock-mode) + (fundamental-mode))) + (run-hooks 'reftex-display-copied-context-hook) + (setq buffer-read-only t)))))) + +(defun reftex-varioref-vref () + "Insert a reference using the `\vref' macro from the varioref package." + (interactive) + (let ((reftex-format-ref-function 'reftex-format-vref)) + (reftex-reference))) +(defun reftex-fancyref-fref () + "Insert a reference using the `\fref' macro from the fancyref package." + (interactive) + (let ((reftex-format-ref-function 'reftex-format-fref) + ;;(reftex-guess-label-type nil) ;FIXME do we want this???? + ) + (reftex-reference))) +(defun reftex-fancyref-Fref () + "Insert a reference using the `\Fref' macro from the fancyref package." + (interactive) + (let ((reftex-format-ref-function 'reftex-format-Fref) + ;;(reftex-guess-label-type nil) ;FIXME do we want this???? + ) + (reftex-reference))) + +(defun reftex-format-vref (label fmt) + (while (string-match "\\\\ref{" fmt) + (setq fmt (replace-match "\\vref{" t t fmt))) + (format fmt label label)) +(defun reftex-format-Fref (label def-fmt) + (format "\\Fref{%s}" label)) +(defun reftex-format-fref (label def-fmt) + (format "\\fref{%s}" label)) + +;;; reftex-ref.el ends here diff --git a/lisp/textmodes/reftex-sel.el b/lisp/textmodes/reftex-sel.el new file mode 100644 index 00000000000..a761705ad2a --- /dev/null +++ b/lisp/textmodes/reftex-sel.el @@ -0,0 +1,671 @@ +;;; reftex-sel.el - The selection modes for RefTeX +;;; Version: 4.5 +;;; +;;; See main file reftex.el for licensing information + +(provide 'reftex-sel) +(require 'reftex) +;;; + +(defvar reftex-select-label-map nil + "Keymap used for *RefTeX Select* buffer, when selecting a label. +This keymap can be used to configure the label selection process which is +started with the command \\[reftex-reference].") + +(defun reftex-select-label-mode () + "Major mode for selecting a label in a LaTeX document. +This buffer was created with RefTeX. +It only has a meaningful keymap when you are in the middle of a +selection process. +To select a label, move the cursor to it and press RET. +Press `?' for a summary of important key bindings. + +During a selection process, these are the local bindings. + +\\{reftex-select-label-map}" + + (interactive) + (kill-all-local-variables) + (make-local-hook 'pre-command-hook) + (make-local-hook 'post-command-hook) + (setq major-mode 'reftex-select-label-mode + mode-name "LSelect") + (set (make-local-variable 'reftex-select-marked) nil) + (when (syntax-table-p reftex-latex-syntax-table) + (set-syntax-table reftex-latex-syntax-table)) + ;; We do not set a local map - reftex-select-item does this. + (run-hooks 'reftex-select-label-mode-hook)) + +(defvar reftex-select-bib-map nil + "Keymap used for *RefTeX Select* buffer, when selecting a BibTeX entry. +This keymap can be used to configure the BibTeX selection process which is +started with the command \\[reftex-citation].") + +(defun reftex-select-bib-mode () + "Major mode for selecting a citation key in a LaTeX document. +This buffer was created with RefTeX. +It only has a meaningful keymap when you are in the middle of a +selection process. +In order to select a citation, move the cursor to it and press RET. +Press `?' for a summary of important key bindings. + +During a selection process, these are the local bindings. + +\\{reftex-select-label-map}" + (interactive) + (kill-all-local-variables) + (make-local-hook 'pre-command-hook) + (make-local-hook 'post-command-hook) + (setq major-mode 'reftex-select-bib-mode + mode-name "BSelect") + (set (make-local-variable 'reftex-select-marked) nil) + ;; We do not set a local map - reftex-select-item does this. + (run-hooks 'reftex-select-bib-mode-hook)) + +(defun reftex-get-offset (buf here-am-I &optional typekey toc index file) + ;; Find the correct offset data, like insert-docstruct would, but faster. + ;; Buffer BUF knows the correct docstruct to use. + ;; Basically this finds the first docstruct entry after HERE-I-AM which + ;; is of allowed type. The optional arguments specify what is allowed. + (catch 'exit + (save-excursion + (set-buffer buf) + (reftex-access-scan-info) + (let* ((rest (memq here-am-I (symbol-value reftex-docstruct-symbol))) + entry) + (while (setq entry (pop rest)) + (if (or (and typekey + (stringp (car entry)) + (or (equal typekey " ") + (equal typekey (nth 1 entry)))) + (and toc (eq (car entry) 'toc)) + (and index (eq (car entry) 'index)) + (and file + (memq (car entry) '(bof eof file-error)))) + (throw 'exit entry))) + nil)))) + +(defun reftex-insert-docstruct + (buf toc labels index-entries files context counter show-commented + here-I-am xr-prefix toc-buffer) + ;; Insert an excerpt of the docstruct list. + ;; Return the data property of the entry corresponding to HERE-I-AM. + ;; BUF is the buffer which has the correct docstruct-symbol. + ;; LABELS non-nil means to include labels into the list. + ;; When a string, indicates the label type to include + ;; FILES non-nil menas to display file boundaries. + ;; CONTEXT non-nil means to include label context. + ;; COUNTER means to count the labels. + ;; SHOW-COMMENTED means to include also labels which are commented out. + ;; HERE-I-AM is a member of the docstruct list. The function will return + ;; a used member near to this one, as a possible starting point. + ;; XR-PREFIX is the prefix to put in front of labels. + ;; TOC-BUFFER means this is to fill the toc buffer. + (let* ((font (reftex-use-fonts)) + (cnt 0) + (index -1) + (toc-indent " ") + (label-indent + (concat "> " + (if toc (make-string (* 7 reftex-level-indent) ?\ ) ""))) + (context-indent + (concat ". " + (if toc (make-string (* 7 reftex-level-indent) ?\ ) ""))) + (mouse-face + (if (memq reftex-highlight-selection '(mouse both)) + reftex-mouse-selected-face + nil)) + (label-face (reftex-verified-face reftex-label-face + 'font-lock-constant-face + 'font-lock-reference-face)) + (index-face (reftex-verified-face reftex-index-face + 'font-lock-constant-face + 'font-lock-reference-face)) + all cell text label typekey note comment master-dir-re + offset from to index-tag docstruct-symbol) + + ;; Pop to buffer buf to get the correct buffer-local variables + (save-excursion + (set-buffer buf) + + ;; Ensure access to scanning info + (reftex-access-scan-info) + + (setq docstruct-symbol reftex-docstruct-symbol + all (symbol-value reftex-docstruct-symbol) + reftex-active-toc nil + master-dir-re + (concat "\\`" (regexp-quote + (file-name-directory (reftex-TeX-master-file)))))) + + (set (make-local-variable 'reftex-docstruct-symbol) docstruct-symbol) + (set (make-local-variable 'reftex-prefix) + (cdr (assoc labels reftex-typekey-to-prefix-alist))) + (if (equal reftex-prefix " ") (setq reftex-prefix nil)) + + ;; Walk the docstruct and insert the appropriate stuff + (while (setq cell (pop all)) + + (incf index) + (setq from (point)) + + (if (eq cell here-I-am) (setq offset 'attention)) + + (cond + + ((memq (car cell) '(bib thebib label-numbers appendix + master-dir bibview-cache is-multi xr xr-doc))) + ;; These are currently ignored + + ((memq (car cell) '(bof eof file-error)) + ;; Beginning or end of a file + (when files + (if (eq offset 'attention) (setq offset cell)) + (insert + " File " (if (string-match master-dir-re (nth 1 cell)) + (substring (nth 1 cell) (match-end 0)) + (nth 1 cell)) + (cond ((eq (car cell) 'bof) " starts here\n") + ((eq (car cell) 'eof) " ends here\n") + ((eq (car cell) 'file-error) " was not found\n"))) + (setq to (point)) + (when font + (put-text-property from to + 'face reftex-file-boundary-face)) + (when toc-buffer + (if mouse-face + (put-text-property from (1- to) + 'mouse-face mouse-face)) + (put-text-property from to :data cell)))) + + ((eq (car cell) 'toc) + ;; a table of contents entry + (when toc + (if (eq offset 'attention) (setq offset cell)) + (setq reftex-active-toc cell) + (insert (concat toc-indent (nth 2 cell) "\n")) + (setq to (point)) + (when font + (put-text-property from to + 'face reftex-section-heading-face)) + (when toc-buffer + (if mouse-face + (put-text-property from (1- to) + 'mouse-face mouse-face)) + (put-text-property from to :data cell)) + (goto-char to))) + + ((stringp (car cell)) + ;; a label + (when (null (nth 2 cell)) + ;; No context yet. Quick update. + (setcdr cell (cdr (reftex-label-info-update cell))) + (put docstruct-symbol 'modified t)) + + (setq label (car cell) + typekey (nth 1 cell) + text (nth 2 cell) + comment (nth 4 cell) + note (nth 5 cell)) + + (when (and labels + (or (eq labels t) + (string= typekey labels) + (string= labels " ")) + (or show-commented (null comment))) + + ;; Yes we want this one + (incf cnt) + (if (eq offset 'attention) (setq offset cell)) + + (setq label (concat xr-prefix label)) + (when comment (setq label (concat "% " label))) + (insert label-indent label) + (when font + (setq to (point)) + (put-text-property + (- (point) (length label)) to + 'face (if comment + 'font-lock-comment-face + label-face)) + (goto-char to)) + + (insert (if counter (format " (%d) " cnt) "") + (if comment " LABEL IS COMMENTED OUT " "") + (if (stringp note) (concat " " note) "") + "\n") + (setq to (point)) + + (when context + (insert context-indent text "\n") + (setq to (point))) + (put-text-property from to :data cell) + (when mouse-face + (put-text-property from (1- to) + 'mouse-face mouse-face)) + (goto-char to))) + + ((eq (car cell) 'index) + ;; index entry + (when (and index-entries + (or (eq t index-entries) + (string= index-entries (nth 1 cell)))) + (if (eq offset 'attention) (setq offset cell)) + (setq index-tag (format "<%s>" (nth 1 cell))) + (and font + (put-text-property 0 (length index-tag) + 'face reftex-index-tag-face index-tag)) + (insert label-indent index-tag " " (nth 7 cell)) + + (when font + (setq to (point)) + (put-text-property + (- (point) (length (nth 7 cell))) to + 'face index-face) + (goto-char to)) + (insert "\n") + (setq to (point)) + + (when context + (insert context-indent (nth 2 cell) "\n") + (setq to (point))) + (put-text-property from to :data cell) + (when mouse-face + (put-text-property from (1- to) + 'mouse-face mouse-face)) + (goto-char to))))) + + (when (reftex-refontify) + ;; we need to fontify the buffer + (reftex-fontify-select-label-buffer buf)) + (run-hooks 'reftex-display-copied-context-hook) + offset)) + +(defun reftex-find-start-point (fallback &rest locations) + ;; Set point to the first available LOCATION. When a LOCATION is a list, + ;; search for such a :data text property. When it is an integer, + ;; use is as line number. FALLBACK is a buffer position used if everything + ;; else fails. + (catch 'exit + (goto-char (point-min)) + (let (loc pos) + (while locations + (setq loc (pop locations)) + (cond + ((null loc)) + ((listp loc) + (setq pos (text-property-any (point-min) (point-max) :data loc)) + (when pos + (goto-char pos) + (throw 'exit t))) + ((integerp loc) + (when (<= loc (count-lines (point-min) (point-max))) + (goto-line loc) + (throw 'exit t))))) + (goto-char fallback)))) + +(defvar reftex-last-data nil) +(defvar reftex-last-line nil) +(defvar reftex-select-marked nil) + +(defun reftex-select-item (prompt help-string keymap + &optional offset + call-back cb-flag) +;; Select an item, using PROMPT. The function returns a key indicating +;; an exit status, along with a data structure indicating which item was +;; selected. +;; HELP-STRING contains help. KEYMAP is a keymap with the available +;; selection commands. +;; OFFSET can be a label list item which will be selected at start. +;; When it is t, point will start out at the beginning of the buffer. +;; Any other value will cause restart where last selection left off. +;; When CALL-BACK is given, it is a function which is called with the index +;; of the element. +;; CB-FLAG is the initial value of that flag. + + (let* (ev data last-data (selection-buffer (current-buffer))) + + (setq reftex-select-marked nil) + + (setq ev + (catch 'myexit + (save-window-excursion + (setq truncate-lines t) + + ;; Find a good starting point + (reftex-find-start-point + (point-min) offset reftex-last-data reftex-last-line) + (beginning-of-line 1) + (set (make-local-variable 'reftex-last-follow-point) (point)) + + (unwind-protect + (progn + (use-local-map keymap) + (add-hook 'pre-command-hook 'reftex-select-pre-command-hook nil t) + (add-hook 'post-command-hook 'reftex-select-post-command-hook nil t) + (princ prompt) + (set-marker reftex-recursive-edit-marker (point)) + ;; XEmacs does not run post-command-hook here + (and (featurep 'xemacs) (run-hooks 'post-command-hook)) + (recursive-edit)) + + (set-marker reftex-recursive-edit-marker nil) + (save-excursion + (set-buffer selection-buffer) + (use-local-map nil) + (remove-hook 'pre-command-hook 'reftex-select-pre-command-hook t) + (remove-hook 'post-command-hook + 'reftex-select-post-command-hook t)) + ;; Kill the mark overlays + (mapcar (lambda (c) (delete-overlay (nth 1 c))) + reftex-select-marked))))) + + (set (make-local-variable 'reftex-last-line) + (+ (count-lines (point-min) (point)) (if (bolp) 1 0))) + (set (make-local-variable 'reftex-last-data) last-data) + (reftex-kill-buffer "*RefTeX Help*") + (setq reftex-callback-fwd (not reftex-callback-fwd)) ;; ;-))) + (message "") + (list ev data last-data))) + +;; The following variables are all bound dynamically in `reftex-select-item'. +;; The defvars are here only to silence the byte compiler. + +(defvar found-list) +(defvar cb-flag) +(defvar data) +(defvar prompt) +(defvar last-data) +(defvar call-back) +(defvar help-string) +(defvar refstyle) + +;; The selection commands + +(defun reftex-select-pre-command-hook () + (reftex-unhighlight 1) + (reftex-unhighlight 0)) + +(defun reftex-select-post-command-hook () + (let (b e) + (setq data (get-text-property (point) :data)) + (setq last-data (or data last-data)) + + (when (and data cb-flag + (not (equal reftex-last-follow-point (point)))) + (setq reftex-last-follow-point (point)) + (funcall call-back data reftex-callback-fwd + (not reftex-revisit-to-follow))) + (if data + (setq b (or (previous-single-property-change + (1+ (point)) :data) + (point-min)) + e (or (next-single-property-change + (point) :data) + (point-max))) + (setq b (point) e (point))) + (and (memq reftex-highlight-selection '(cursor both)) + (reftex-highlight 1 b e)) + (if (or (not (pos-visible-in-window-p b)) + (not (pos-visible-in-window-p e))) + (recenter '(4))) + (unless (current-message) + (princ prompt)))) + +(defun reftex-select-next (&optional arg) + "Move to next selectable item." + (interactive "p") + (setq reftex-callback-fwd t) + (or (eobp) (forward-char 1)) + (re-search-forward "^[^. \t\n\r]" nil t arg) + (beginning-of-line 1)) +(defun reftex-select-previous (&optional arg) + "Move to previous selectable item." + (interactive "p") + (setq reftex-callback-fwd nil) + (re-search-backward "^[^. \t\n\r]" nil t arg)) +(defun reftex-select-next-heading (&optional arg) + "Move to next table of contentes line." + (interactive "p") + (end-of-line) + (re-search-forward "^ " nil t arg) + (beginning-of-line)) +(defun reftex-select-previous-heading (&optional arg) + "Move to previous table of contentes line." + (interactive "p") + (re-search-backward "^ " nil t arg)) +(defun reftex-select-quit () + "Abort selection process." + (interactive) + (throw 'myexit nil)) +(defun reftex-select-keyboard-quit () + "Abort selection process." + (interactive) + (throw 'exit t)) +(defun reftex-select-jump-to-previous () + "Jump back to where previous selection process left off." + (interactive) + (let (pos) + (cond + ((and (local-variable-p 'reftex-last-data (current-buffer)) + reftex-last-data + (setq pos (text-property-any (point-min) (point-max) + :data reftex-last-data))) + (goto-char pos)) + ((and (local-variable-p 'reftex-last-line (current-buffer)) + (integerp reftex-last-line)) + (goto-line reftex-last-line)) + (t (ding))))) +(defun reftex-select-toggle-follow () + "Toggle follow mode: Other window follows with full context." + (interactive) + (setq reftex-last-follow-point -1) + (setq cb-flag (not cb-flag))) +(defun reftex-select-toggle-varioref () + "Toggle the macro used for referencing the label between \\ref and \\vref." + (interactive) + (if (string= refstyle "\\ref") + (setq refstyle "\\vref") + (setq refstyle "\\ref")) + (force-mode-line-update)) +(defun reftex-select-toggle-fancyref () + "Toggle the macro used for referencing the label between \\ref and \\vref." + (interactive) + (setq refstyle + (cond ((string= refstyle "\\ref") "\\fref") + ((string= refstyle "\\fref") "\\Fref") + (t "\\ref"))) + (force-mode-line-update)) +(defun reftex-select-show-insertion-point () + "Show the point from where selection was started in another window." + (interactive) + (let ((this-window (selected-window))) + (unwind-protect + (progn + (switch-to-buffer-other-window + (marker-buffer reftex-select-return-marker)) + (goto-char (marker-position reftex-select-return-marker)) + (recenter '(4))) + (select-window this-window)))) +(defun reftex-select-callback () + "Show full context in another window." + (interactive) + (if data (funcall call-back data reftex-callback-fwd nil) (ding))) +(defun reftex-select-accept () + "Accept the currently selected item." + (interactive) + (throw 'myexit 'return)) +(defun reftex-select-mouse-accept (ev) + "Accept the item at the mouse click." + (interactive "e") + (mouse-set-point ev) + (setq data (get-text-property (point) :data)) + (setq last-data (or data last-data)) + (throw 'myexit 'return)) +(defun reftex-select-read-label () + "Use minibuffer to read a label to reference, with completion." + (interactive) + (let ((label (completing-read + "Label: " (symbol-value reftex-docstruct-symbol) + nil nil reftex-prefix))) + (unless (or (equal label "") (equal label reftex-prefix)) + (throw 'myexit label)))) +(defun reftex-select-read-cite () + "Use minibuffer to read a citation key with completion." + (interactive) + (let* ((key (completing-read "Citation key: " found-list)) + (entry (assoc key found-list))) + (cond + ((or (null key) (equal key ""))) + (entry + (setq data entry) + (setq last-data data) + (throw 'myexit 'return)) + (t (throw 'myexit key))))) + +(defun reftex-select-mark (&optional separator) + "Mark the entry." + (interactive) + (let* ((data (get-text-property (point) :data)) + boe eoe ovl) + (or data (error "No entry to mark at point")) + (if (assq data reftex-select-marked) + (error "Entry is already marked")) + (setq boe (or (previous-single-property-change (1+ (point)) :data) + (point-min)) + eoe (or (next-single-property-change (point) :data) (point-max))) + (setq ovl (make-overlay boe eoe)) + (push (list data ovl separator) reftex-select-marked) + (overlay-put ovl 'face reftex-select-mark-face) + (if (featurep 'xemacs) + ;; before-string property is broken in Emacs + (overlay-put ovl 'before-string + (if separator + (format "*%c%d* " separator + (length reftex-select-marked)) + (format "*%d* " (length reftex-select-marked))))) + (message "Entry has mark no. %d" (length reftex-select-marked)))) + +(defun reftex-select-mark-comma () + "Mark the entry and store the `comma' separator." + (interactive) + (reftex-select-mark ?,)) +(defun reftex-select-mark-to () + "Mark the entry and store the `to' separator." + (interactive) + (reftex-select-mark ?-)) +(defun reftex-select-mark-and () + "Mark the entry and store `and' to separator." + (interactive) + (reftex-select-mark ?+)) + +(defun reftex-select-unmark () + "Unmark the entry." + (interactive) + (let* ((data (get-text-property (point) :data)) + (cell (assq data reftex-select-marked)) + (ovl (nth 1 cell)) + (cnt 0) + sep) + (unless cell + (error "No marked entry at point")) + (and ovl (delete-overlay ovl)) + (setq reftex-select-marked (delq cell reftex-select-marked)) + (if (featurep 'xemacs) + ;; before-string property is broken in Emacs + (progn + (setq cnt (1+ (length reftex-select-marked))) + (mapcar (lambda (c) + (setq sep (nth 2 c)) + (overlay-put (nth 1 c) 'before-string + (if sep + (format "*%c%d* " sep (decf cnt)) + (format "*%d* " (decf cnt))))) + reftex-select-marked))) + (message "Entry no longer marked"))) + +(defun reftex-select-help () + "Display a summary of the special key bindings." + (interactive) + (with-output-to-temp-buffer "*RefTeX Help*" + (princ help-string)) + (reftex-enlarge-to-fit "*RefTeX Help*" t)) + +;; Common bindings in reftex-select-label-map and reftex-select-bib-map +(let ((map (make-sparse-keymap))) + (substitute-key-definition + 'next-line 'reftex-select-next map global-map) + (substitute-key-definition + 'previous-line 'reftex-select-previous map global-map) + (substitute-key-definition + 'keyboard-quit 'reftex-select-keyboard-quit map global-map) + (substitute-key-definition + 'newline 'reftex-select-accept map global-map) + + (loop for x in + '((" " . reftex-select-callback) + ("n" . reftex-select-next) + ([(down)] . reftex-select-next) + ("p" . reftex-select-previous) + ([(up)] . reftex-select-previous) + ("f" . reftex-select-toggle-follow) + ("\C-m" . reftex-select-accept) + ([(return)] . reftex-select-accept) + ("q" . reftex-select-quit) + ("." . reftex-select-show-insertion-point) + ("?" . reftex-select-help)) + do (define-key map (car x) (cdr x))) + + ;; The mouse-2 binding + (if (featurep 'xemacs) + (define-key map [(button2)] 'reftex-select-mouse-accept) + (define-key map [(mouse-2)] 'reftex-select-mouse-accept)) + + ;; Digit arguments + (loop for key across "0123456789" do + (define-key map (vector (list key)) 'digit-argument)) + (define-key map "-" 'negative-argument) + + ;; Make two maps + (setq reftex-select-label-map map) + (setq reftex-select-bib-map (copy-keymap map))) + +;; Specific bindings in reftex-select-label-map +(loop for key across "aAcgFlrRstx#%" do + (define-key reftex-select-label-map (vector (list key)) + (list 'lambda '() + "Press `?' during selection to find out about this key." + '(interactive) (list 'throw '(quote myexit) key)))) + +(loop for x in + '(("b" . reftex-select-jump-to-previous) + ("v" . reftex-select-toggle-varioref) + ("V" . reftex-select-toggle-fancyref) + ("m" . reftex-select-mark) + ("u" . reftex-select-unmark) + ("," . reftex-select-mark-comma) + ("-" . reftex-select-mark-to) + ("+" . reftex-select-mark-and) + ([(tab)] . reftex-select-read-label) + ("\C-i" . reftex-select-read-label) + ("\C-c\C-n" . reftex-select-next-heading) + ("\C-c\C-p" . reftex-select-previous-heading)) + do + (define-key reftex-select-label-map (car x) (cdr x))) + +;; Specific bindings in reftex-select-bib-map +(loop for key across "grRaA" do + (define-key reftex-select-bib-map (vector (list key)) + (list 'lambda '() + "Press `?' during selection to find out about this key." + '(interactive) (list 'throw '(quote myexit) key)))) + +(loop for x in + '(("\C-i" . reftex-select-read-cite) + ([(tab)] . reftex-select-read-cite) + ("m" . reftex-select-mark) + ("u" . reftex-select-unmark)) + do (define-key reftex-select-bib-map (car x) (cdr x))) + + +;;; reftex-sel.el ends here diff --git a/lisp/textmodes/reftex-toc.el b/lisp/textmodes/reftex-toc.el new file mode 100644 index 00000000000..f9b63916c29 --- /dev/null +++ b/lisp/textmodes/reftex-toc.el @@ -0,0 +1,583 @@ +;;; reftex-toc.el - RefTeX's table of contents mode +;;; Version: 4.5 +;;; +;;; See main file reftex.el for licensing information + +(provide 'reftex-toc) +(require 'reftex) +;;; + +(defvar reftex-toc-map (make-sparse-keymap) + "Keymap used for *toc* buffer.") + +(defvar reftex-toc-menu) + +(defun reftex-toc-mode () + "Major mode for managing Table of Contents for LaTeX files. +This buffer was created with RefTeX. +Press `?' for a summary of important key bindings. + +Here are all local bindings. + +\\{reftex-toc-map}" + (interactive) + (kill-all-local-variables) + (setq major-mode 'reftex-toc-mode + mode-name "TOC") + (use-local-map reftex-toc-map) + (set (make-local-variable 'revert-buffer-function) 'reftex-toc-revert) + (set (make-local-variable 'reftex-toc-include-labels-indicator) "") + (set (make-local-variable 'reftex-toc-include-index-indicator) "") + (setq mode-line-format + (list "---- " 'mode-line-buffer-identification + " " 'global-mode-string " (" mode-name ")" + " L<" 'reftex-toc-include-labels-indicator ">" + " I<" 'reftex-toc-include-index-indicator ">" + " -%-")) + (setq truncate-lines t) + (make-local-hook 'post-command-hook) + (make-local-hook 'pre-command-hook) + (make-local-variable 'reftex-last-follow-point) + (add-hook 'post-command-hook 'reftex-toc-post-command-hook nil t) + (add-hook 'pre-command-hook 'reftex-toc-pre-command-hook nil t) + (easy-menu-add reftex-toc-menu reftex-toc-map) + (run-hooks 'reftex-toc-mode-hook)) + +(defvar reftex-last-toc-file nil + "Stores the file name from which `reftex-toc' was called. For redo command.") + +(defvar reftex-last-window-height nil) +(defvar reftex-toc-include-labels-indicator nil) +(defvar reftex-toc-include-index-indicator nil) + +(defvar reftex-toc-return-marker (make-marker) + "Marker which makes it possible to return from toc to old position.") + +(defconst reftex-toc-help +" AVAILABLE KEYS IN TOC BUFFER + ============================ +n / p next-line / previous-line +SPC Show the corresponding location of the LaTeX document. +TAB Goto the location and keep the *toc* window. +RET Goto the location and hide the *toc* window (also on mouse-2). +C-c > Display Index. With prefix arg, restrict index to current section. +q / k Hide/Kill *toc* buffer, return to position of reftex-toc command. +l i c F Toggle display of [l]abels, [i]ndex, [c]ontext, [F]ile borders. +f / g Toggle follow mode on and off / Refresh *toc* buffer. +r / C-u r Reparse the LaTeX document / Reparse entire LaTeX document. +. In other window, show position from where `reftex-toc' was called. +x Switch to TOC of external document (with LaTeX package `xr').") + +(defun reftex-toc (&optional rebuild) + "Show the table of contents for the current document. +When called with a raw C-u prefix, rescan the document first." + + (interactive) + + (if (or (not (string= reftex-last-toc-master (reftex-TeX-master-file))) + current-prefix-arg) + (reftex-erase-buffer "*toc*")) + + (setq reftex-last-toc-file (buffer-file-name)) + (setq reftex-last-toc-master (reftex-TeX-master-file)) + + (set-marker reftex-toc-return-marker (point)) + + ;; If follow mode is active, arrange to delay it one command + (if reftex-toc-follow-mode + (setq reftex-toc-follow-mode 1)) + + (and reftex-toc-include-index-entries + (reftex-ensure-index-support)) + (or reftex-support-index + (setq reftex-toc-include-index-entries nil)) + + ;; Ensure access to scanning info and rescan buffer if prefix are is '(4) + (reftex-access-scan-info current-prefix-arg) + + (let* ((this-buf (current-buffer)) + (docstruct-symbol reftex-docstruct-symbol) + (xr-data (assq 'xr (symbol-value reftex-docstruct-symbol))) + (xr-alist (cons (cons "" (buffer-file-name)) (nth 1 xr-data))) + (here-I-am (if rebuild + (get 'reftex-toc :reftex-data) + (car (reftex-where-am-I)))) + offset) + + (if (get-buffer-window "*toc*") + (select-window (get-buffer-window "*toc*")) + (when (or (not reftex-toc-keep-other-windows) + (< (window-height) (* 2 window-min-height))) + (delete-other-windows)) + (setq reftex-last-window-height (window-height)) ; remember + (split-window) + (let ((default-major-mode 'reftex-toc-mode)) + (switch-to-buffer "*toc*"))) + + (or (eq major-mode 'reftex-toc-mode) (reftex-toc-mode)) + (set (make-local-variable 'reftex-docstruct-symbol) docstruct-symbol) + (setq reftex-toc-include-labels-indicator + (if (eq reftex-toc-include-labels t) + "ALL" + reftex-toc-include-labels)) + (setq reftex-toc-include-index-indicator + (if (eq reftex-toc-include-index-entries t) + "ALL" + reftex-toc-include-index-entries)) + + (cond + ((= (buffer-size) 0) + ;; buffer is empty - fill it with the table of contents + (message "Building *toc* buffer...") + + (setq buffer-read-only nil) + (insert (format +"TABLE-OF-CONTENTS on %s +SPC=view TAB=goto RET=goto+hide [q]uit [r]escan [l]abels [f]ollow [x]r [?]Help +------------------------------------------------------------------------------ +" (abbreviate-file-name reftex-last-toc-master))) + + (if (reftex-use-fonts) + (put-text-property 1 (point) 'face reftex-toc-header-face)) + (put-text-property 1 (point) 'intangible t) + (put-text-property 1 2 'xr-alist xr-alist) + + (setq offset + (reftex-insert-docstruct + this-buf + t ; include toc + reftex-toc-include-labels + reftex-toc-include-index-entries + reftex-toc-include-file-boundaries + reftex-toc-include-context + nil ; counter + nil ; commented + here-I-am + "" ; xr-prefix + t ; a toc buffer + )) + + (run-hooks 'reftex-display-copied-context-hook) + (message "Building *toc* buffer...done.") + (setq buffer-read-only t)) + (t + ;; Only compute the offset + (setq offset + (or (reftex-get-offset this-buf here-I-am + (if reftex-toc-include-labels " " nil) + t + reftex-toc-include-index-entries + reftex-toc-include-file-boundaries) + (reftex-last-assoc-before-elt + 'toc here-I-am + (symbol-value reftex-docstruct-symbol)))) + (put 'reftex-toc :reftex-line 3) + (goto-line 3) + (beginning-of-line))) + + ;; Find the correct starting point + (reftex-find-start-point (point) offset (get 'reftex-toc :reftex-line)) + (setq reftex-last-follow-point (point)))) + +(defun reftex-toc-pre-command-hook () + ;; used as pre command hook in *toc* buffer + (reftex-unhighlight 0) + (reftex-unhighlight 1)) + +(defun reftex-toc-post-command-hook () + ;; used in the post-command-hook for the *toc* buffer + (when (get-text-property (point) :data) + (put 'reftex-toc :reftex-data (get-text-property (point) :data)) + (and (> (point) 1) + (not (get-text-property (point) 'intangible)) + (memq reftex-highlight-selection '(cursor both)) + (reftex-highlight 1 + (or (previous-single-property-change (1+ (point)) :data) + (point-min)) + (or (next-single-property-change (point) :data) + (point-max))))) + (if (integerp reftex-toc-follow-mode) + ;; remove delayed action + (setq reftex-toc-follow-mode t) + (and reftex-toc-follow-mode + (not (equal reftex-last-follow-point (point))) + ;; show context in other window + (setq reftex-last-follow-point (point)) + (condition-case nil + (reftex-toc-visit-location nil (not reftex-revisit-to-follow)) + (error t))))) + +(defun reftex-re-enlarge () + ;; Enlarge windiw to a remembered size + (enlarge-window + (max 0 (- (or reftex-last-window-height (window-height)) + (window-height))))) + +(defun reftex-toc-show-help () + "Show a summary of special key bindings." + (interactive) + (with-output-to-temp-buffer "*RefTeX Help*" + (princ reftex-toc-help)) + (reftex-enlarge-to-fit "*RefTeX Help*" t) + ;; If follow mode is active, arrange to delay it one command + (if reftex-toc-follow-mode + (setq reftex-toc-follow-mode 1))) + +(defun reftex-toc-next (&optional arg) + "Move to next selectable item." + (interactive "p") + (setq reftex-callback-fwd t) + (or (eobp) (forward-char 1)) + (goto-char (or (next-single-property-change (point) :data) + (point)))) +(defun reftex-toc-previous (&optional arg) + "Move to previous selectable item." + (interactive "p") + (setq reftex-callback-fwd nil) + (goto-char (or (previous-single-property-change (point) :data) + (point)))) +(defun reftex-toc-next-heading (&optional arg) + "Move to next table of contentes line." + (interactive "p") + (end-of-line) + (re-search-forward "^ " nil t arg) + (beginning-of-line)) +(defun reftex-toc-previous-heading (&optional arg) + "Move to previous table of contentes line." + (interactive "p") + (re-search-backward "^ " nil t arg)) +(defun reftex-toc-toggle-follow () + "Toggle follow (other window follows with context)." + (interactive) + (setq reftex-last-follow-point -1) + (setq reftex-toc-follow-mode (not reftex-toc-follow-mode))) +(defun reftex-toc-toggle-file-boundary () + "Toggle inclusion of file boundaries in *toc* buffer." + (interactive) + (setq reftex-toc-include-file-boundaries + (not reftex-toc-include-file-boundaries)) + (reftex-toc-revert)) +(defun reftex-toc-toggle-labels (arg) + "Toggle inclusion of labels in *toc* buffer. +With prefix ARG, prompt for a label type and include only labels of +that specific type." + (interactive "P") + (setq reftex-toc-include-labels + (if arg (reftex-query-label-type) + (not reftex-toc-include-labels))) + (reftex-toc-revert)) +(defun reftex-toc-toggle-index (arg) + "Toggle inclusion of index in *toc* buffer. +With prefix arg, prompt for an index tag and include only entries of that +specific index." + (interactive "P") + (setq reftex-toc-include-index-entries + (if arg (reftex-index-select-tag) + (not reftex-toc-include-index-entries))) + (reftex-toc-revert)) +(defun reftex-toc-toggle-context () + "Toggle inclusion of label context in *toc* buffer. +Label context is only displayed when the labels are there as well." + (interactive) + (setq reftex-toc-include-context (not reftex-toc-include-context)) + (reftex-toc-revert)) +(defun reftex-toc-view-line () + "View document location in other window." + (interactive) + (reftex-toc-visit-location)) +(defun reftex-toc-goto-line-and-hide () + "Go to document location in other window. Hide the *toc* window." + (interactive) + (reftex-toc-visit-location 'hide)) +(defun reftex-toc-goto-line () + "Go to document location in other window. *toc* window stays." + (interactive) + (reftex-toc-visit-location t)) +(defun reftex-toc-mouse-goto-line-and-hide (ev) + "Go to document location in other window. Hide the *toc* window." + (interactive "e") + (mouse-set-point ev) + (reftex-toc-visit-location 'hide)) +(defun reftex-toc-show-calling-point () + "Show point where reftex-toc was called from." + (interactive) + (let ((this-window (selected-window))) + (unwind-protect + (progn + (switch-to-buffer-other-window + (marker-buffer reftex-toc-return-marker)) + (goto-char (marker-position reftex-toc-return-marker)) + (recenter '(4))) + (select-window this-window)))) +(defun reftex-toc-quit () + "Hide the *toc* window and do not move point." + (interactive) + (or (one-window-p) (delete-window)) + (switch-to-buffer (marker-buffer reftex-toc-return-marker)) + (reftex-re-enlarge) + (goto-char (or (marker-position reftex-toc-return-marker) (point)))) +(defun reftex-toc-quit-and-kill () + "Kill the *toc* buffer." + (interactive) + (kill-buffer "*toc*") + (or (one-window-p) (delete-window)) + (switch-to-buffer (marker-buffer reftex-toc-return-marker)) + (reftex-re-enlarge) + (goto-char (marker-position reftex-toc-return-marker))) +(defun reftex-toc-display-index (&optional arg) + "Display the index buffer for the current document. +This works just like `reftex-display-index' from a LaTeX buffer. +With prefix arg 1, restrict index to the section at point." + (interactive "P") + (let ((data (get-text-property (point) :data)) + (docstruct (symbol-value reftex-docstruct-symbol)) + bor eor restr) + (when (equal arg 2) + (setq bor (reftex-last-assoc-before-elt 'toc data docstruct) + eor (assoc 'toc (cdr (memq bor docstruct))) + restr (list (nth 6 bor) bor eor))) + (reftex-toc-goto-line) + (reftex-display-index (if restr nil arg) restr))) +(defun reftex-toc-rescan (&rest ignore) + "Regenerate the *toc* buffer by reparsing file of section at point." + (interactive) + (if (and reftex-enable-partial-scans + (null current-prefix-arg)) + (let* ((data (get-text-property (point) :data)) + (what (car data)) + (file (cond ((eq what 'toc) (nth 3 data)) + ((memq what '(eof bof file-error)) (nth 1 data)) + ((stringp what) (nth 3 data)) + ((eq what 'index) (nth 3 data)))) + (line (+ (count-lines (point-min) (point)) (if (bolp) 1 0)))) + (if (not file) + (error "Don't know which file to rescan. Try `C-u r'") + (put 'reftex-toc :reftex-line line) + (switch-to-buffer-other-window + (reftex-get-file-buffer-force file)) + (setq current-prefix-arg '(4)) + (reftex-toc t))) + (reftex-toc-Rescan)) + (reftex-kill-temporary-buffers)) +(defun reftex-toc-Rescan (&rest ignore) + "Regenerate the *toc* buffer by reparsing the entire document." + (interactive) + (switch-to-buffer-other-window + (reftex-get-file-buffer-force reftex-last-toc-file)) + (setq current-prefix-arg '(16)) + (reftex-toc t)) +(defun reftex-toc-revert (&rest ignore) + "Regenerate the *toc* from the internal lists." + (interactive) + (switch-to-buffer-other-window + (reftex-get-file-buffer-force reftex-last-toc-file)) + (reftex-erase-buffer "*toc*") + (setq current-prefix-arg nil) + (reftex-toc t)) +(defun reftex-toc-external (&rest ignore) + "Switch to table of contents of an external document." + (interactive) + (let* ((old-buf (current-buffer)) + (xr-alist (get-text-property 1 'xr-alist)) + (xr-index (reftex-select-external-document + xr-alist 0))) + (switch-to-buffer-other-window (or (reftex-get-file-buffer-force + (cdr (nth xr-index xr-alist))) + (error "Cannot switch document"))) + (reftex-toc) + (if (equal old-buf (current-buffer)) + (message "") + (message "Switched document")))) + +(defun reftex-toc-visit-location (&optional final no-revisit) + ;; Visit the tex file corresponding to the toc entry on the current line. + ;; If FINAL is t, stay there + ;; If FINAL is 'hide, hide the *toc* window. + ;; Otherwise, move cursor back into *toc* window. + ;; NO-REVISIT means don't visit files, just use live biffers. + ;; This function is pretty clever about finding back a section heading, + ;; even if the buffer is not live, or things like outline, x-symbol etc. + ;; have been active. + + (let* ((toc (get-text-property (point) :data)) + (toc-window (selected-window)) + show-window show-buffer match) + + (unless toc (error "Don't know which toc line to visit")) + + (cond + + ((eq (car toc) 'toc) + ;; a toc entry + (setq match (reftex-toc-find-section toc no-revisit))) + + ((eq (car toc) 'index) + ;; an index entry + (setq match (reftex-index-show-entry toc no-revisit))) + + ((memq (car toc) '(bof eof)) + ;; A file entry + (setq match + (let ((where (car toc)) + (file (nth 1 toc))) + (if (or (not no-revisit) (reftex-get-buffer-visiting file)) + (progn + (switch-to-buffer-other-window + (reftex-get-file-buffer-force file nil)) + (goto-char (if (eq where 'bof) (point-min) (point-max)))) + (message reftex-no-follow-message) nil)))) + + ((stringp (car toc)) + ;; a label + (setq match (reftex-show-label-location toc reftex-callback-fwd + no-revisit t)))) + + (setq show-window (selected-window) + show-buffer (current-buffer)) + + (unless match + (select-window toc-window) + (error "Cannot find location")) + + (select-window toc-window) + + ;; use the `final' parameter to decide what to do next + (cond + ((eq final t) + (reftex-unhighlight 0) + (select-window show-window)) + ((eq final 'hide) + (reftex-unhighlight 0) + (or (one-window-p) (delete-window)) + (switch-to-buffer show-buffer) + (reftex-re-enlarge)) + (t nil)))) + +(defun reftex-toc-find-section (toc &optional no-revisit) + (let* ((file (nth 3 toc)) + (marker (nth 4 toc)) + (level (nth 5 toc)) + (literal (nth 7 toc)) + (emergency-point (nth 8 toc)) + (match + (cond + ((and (markerp marker) (marker-buffer marker)) + ;; Buffer is still live and we have the marker. Should be easy. + (switch-to-buffer-other-window (marker-buffer marker)) + (goto-char (marker-position marker)) + (or (looking-at (regexp-quote literal)) + (looking-at (reftex-make-regexp-allow-for-ctrl-m literal)) + (looking-at (reftex-make-desperate-section-regexp literal)) + (looking-at (concat "\\\\" + (regexp-quote + (car + (rassq level + reftex-section-levels-all))) + "[[{]")))) + ((or (not no-revisit) + (reftex-get-buffer-visiting file)) + ;; Marker is lost. Use the backup method. + (switch-to-buffer-other-window + (reftex-get-file-buffer-force file nil)) + (goto-char (or emergency-point (point-min))) + (or (looking-at (regexp-quote literal)) + (let ((len (length literal))) + (or (reftex-nearest-match (regexp-quote literal) len) + (reftex-nearest-match + (reftex-make-regexp-allow-for-ctrl-m literal) len) + (reftex-nearest-match + (reftex-make-desperate-section-regexp literal) len))))) + (t (message reftex-no-follow-message) nil)))) + (when match + (goto-char (match-beginning 0)) + (if (not (= (point) (point-max))) (recenter 1)) + (reftex-highlight 0 (match-beginning 0) (match-end 0) (current-buffer))) + match)) + +(defun reftex-make-desperate-section-regexp (old) + ;; Return a regexp which will still match a section statement even if + ;; x-symbol or isotex or the like have been at work in the mean time. + (let* ((n (1+ (string-match "[[{]" old))) + (new (regexp-quote (substring old 0 (1+ (string-match "[[{]" old))))) + (old (substring old n))) + (while (string-match + "\\([\r\n]\\)\\|\\(\\`\\|[ \t\n\r]\\)\\([a-zA-Z0-9]+\\)\\([ \t\n\r]\\|}\\'\\)" + old) + (if (match-beginning 1) + (setq new (concat new "[^\n\r]*[\n\r]")) + (setq new (concat new "[^\n\r]*" (match-string 3 old)))) + (setq old (substring old (match-end 0)))) + new)) + +;; Table of Contents map +(define-key reftex-toc-map (if (featurep 'xemacs) [(button2)] [(mouse-2)]) + 'reftex-toc-mouse-goto-line-and-hide) + +(substitute-key-definition + 'next-line 'reftex-toc-next reftex-toc-map global-map) +(substitute-key-definition + 'previous-line 'reftex-toc-previous reftex-toc-map global-map) + +(loop for x in + '(("n" . reftex-toc-next) + ("p" . reftex-toc-previous) + ("?" . reftex-toc-show-help) + (" " . reftex-toc-view-line) + ("\C-m" . reftex-toc-goto-line-and-hide) + ("\C-i" . reftex-toc-goto-line) + ("\C-c>". reftex-toc-display-index) + ("r" . reftex-toc-rescan) + ("R" . reftex-toc-Rescan) + ("g" . revert-buffer) + ("q" . reftex-toc-quit) + ("k" . reftex-toc-quit-and-kill) + ("f" . reftex-toc-toggle-follow) + ("F" . reftex-toc-toggle-file-boundary) + ("i" . reftex-toc-toggle-index) + ("l" . reftex-toc-toggle-labels) + ("c" . reftex-toc-toggle-context) + ("%" . reftex-toc-toggle-commented) + ("x" . reftex-toc-external) + ("." . reftex-toc-show-calling-point) + ("\C-c\C-n" . reftex-toc-next-heading) + ("\C-c\C-p" . reftex-toc-previous-heading)) + do (define-key reftex-toc-map (car x) (cdr x))) + +(loop for key across "0123456789" do + (define-key reftex-toc-map (vector (list key)) 'digit-argument)) +(define-key reftex-toc-map "-" 'negative-argument) + +(easy-menu-define + reftex-toc-menu reftex-toc-map + "Menu for Table of Contents buffer" + '("TOC" + ["Show Location" reftex-toc-view-line t] + ["Go To Location" reftex-toc-goto-line t] + ["Exit & Go To Location" reftex-toc-goto-line-and-hide t] + ["Index" reftex-toc-display-index t] + ["Quit" reftex-toc-quit t] + "--" + ["External Document TOC " reftex-toc-external t] + "--" + ("Update" + ["Rebuilt *toc* Buffer" revert-buffer t] + ["Rescan One File" reftex-toc-rescan reftex-enable-partial-scans] + ["Rescan Entire Document" reftex-toc-Rescan t]) + ("Options" + "TOC Items" + ["File Boundaries" reftex-toc-toggle-file-boundary :style toggle + :selected reftex-toc-include-file-boundaries] + ["Labels" reftex-toc-toggle-labels :style toggle + :selected reftex-toc-include-labels] + ["Index Entries" reftex-toc-toggle-index :style toggle + :selected reftex-toc-include-index-entries] + ["Context" reftex-toc-toggle-context :style toggle + :selected reftex-toc-include-context] + "--" + ["Follow Mode" reftex-toc-toggle-follow :style toggle + :selected reftex-toc-follow-mode]) + "--" + ["Help" reftex-toc-show-help t])) + + +;;; reftex-toc.el ends here diff --git a/lisp/textmodes/reftex-vars.el b/lisp/textmodes/reftex-vars.el new file mode 100644 index 00000000000..7243641ea27 --- /dev/null +++ b/lisp/textmodes/reftex-vars.el @@ -0,0 +1,1547 @@ +;;; reftex-vars.el - Configuration variables for RefTeX +;;; Version: 4.5 +;;; +;;; See main file reftex.el for licensing information + +(provide 'reftex-vars) + +;; Define the two constants which are needed during compilation + +(eval-and-compile +(defconst reftex-label-alist-builtin + '( + ;; Some aliases, mostly for backward compatibility + (Sideways "Alias for -->rotating" (rotating)) + (AMSTeX "amsmath with eqref macro" + ((nil ?e nil "~\\eqref{%s}") + amsmath)) + + ;; Individual package defaults + (amsmath "AMS-LaTeX math environments" + (("align" ?e nil nil eqnarray-like) + ("gather" ?e nil nil eqnarray-like) + ("multline" ?e nil nil t) + ("flalign" ?e nil nil eqnarray-like) + ("alignat" ?e nil nil alignat-like) + ("xalignat" ?e nil nil alignat-like) + ("xxalignat" ?e nil nil alignat-like) + ("subequations" ?e nil nil t))) + + (endnotes "The \\endnote macro" + (("\\endnote[]{}" ?N "en:" "~\\ref{%s}" 2 + (regexp "endnotes?" "notes?" "Anmerkung\\(en\\)?" "Anm\\.")))) + + (fancybox "The Beqnarray environment" + (("Beqnarray" ?e nil nil eqnarray-like))) + + (floatfig "The floatingfigure environment" + (("floatingfigure" ?f nil nil caption))) + + (longtable "The longtable environment" + (("longtable" ?t nil nil caption))) + + (picinpar "The figwindow and tabwindow environments" + (("figwindow" ?f nil nil 1) + ("tabwindow" ?f nil nil 1))) + + (rotating "Sidewaysfigure and table" + (("sidewaysfigure" ?f nil nil caption) + ("sidewaystable" ?t nil nil caption))) + + (sidecap "CSfigure and SCtable" + (("SCfigure" ?f nil nil caption) + ("SCtable" ?t nil nil caption))) + + (subfigure "Subfigure environments/macro" + (("subfigure" ?f nil nil caption) + ("subfigure*" ?f nil nil caption) + ("\\subfigure[]{}" ?f nil nil 1))) + + (supertab "Supertabular environment" + (("supertabular" ?t nil nil "\\tablecaption{"))) + + (wrapfig "The wrapfigure environment" + (("wrapfigure" ?f nil nil caption))) + + ;; The LaTeX core stuff + (LaTeX "LaTeX default environments" + (("section" ?s "%S" "~\\ref{%s}" (nil . t) + (regexp "parts?" "chapters?" "chap\\." "sections?" "sect?\\." + "paragraphs?" "par\\." + "\\\\S" "\247" "Teile?" "Kapitel" "Kap\\." "Abschnitte?" + "appendi\\(x\\|ces\\)" "App\\." "Anh\"?ange?" "Anh\\.")) + + ("enumerate" ?i "item:" "~\\ref{%s}" item + (regexp "items?" "Punkte?")) + + ("equation" ?e "eq:" "~(\\ref{%s})" t + (regexp "equations?" "eqs?\\." "eqn\\." "Gleichung\\(en\\)?" "Gl\\.")) + ("eqnarray" ?e "eq:" nil eqnarray-like) + + ("figure" ?f "fig:" "~\\ref{%s}" caption + (regexp "figure?[sn]?" "figs?\\." "Abbildung\\(en\\)?" "Abb\\.")) + ("figure*" ?f nil nil caption) + + ("table" ?t "tab:" "~\\ref{%s}" caption + (regexp "tables?" "tab\\." "Tabellen?")) + ("table*" ?t nil nil caption) + + ("\\footnote[]{}" ?n "fn:" "~\\ref{%s}" 2 + (regexp "footnotes?" "Fussnoten?")) + + ("any" ?\ " " "~\\ref{%s}" nil) + + ;; The label macro is hard coded, but it *could* be defined like this: + ;;("\\label{*}" nil nil nil nil) + )) + + ) + "The default label environment descriptions. +Lower-case symbols correspond to a style file of the same name in the LaTeX +distribution. Mixed-case symbols are convenience aliases.") + +(defconst reftex-cite-format-builtin + '((default "Default macro \\cite{%l}" + "\\cite{%l}") + (natbib "The Natbib package" + ((?\C-m . "\\cite{%l}") + (?t . "\\citet{%l}") + (?T . "\\citet*{%l}") + (?p . "\\citep{%l}") + (?P . "\\citep*{%l}") + (?e . "\\citep[e.g.][]{%l}") + (?s . "\\citep[see][]{%l}") + (?a . "\\citeauthor{%l}") + (?A . "\\citeauthor*{%l}") + (?y . "\\citeyear{%l}"))) + (harvard "The Harvard package" + ((?\C-m . "\\cite{%l}") + (?p . "\\cite{%l}") + (?t . "\\citeasnoun{%l}") + (?n . "\\citeasnoun{%l}") + (?s . "\\possessivecite{%l}") + (?e . "\\citeaffixed{%l}{?}") + (?y . "\\citeyear{%l}") + (?a . "\\citename{%l}"))) + (chicago "The Chicago package" + ((?\C-m . "\\cite{%l}") + (?t . "\\citeN{%l}") + (?T . "\\shortciteN{%l}") + (?p . "\\cite{%l}") + (?P . "\\shortcite{%l}") + (?a . "\\citeA{%l}") + (?A . "\\shortciteA{%l}") + (?y . "\\citeyear{%l}"))) + (astron "The Astron package" + ((?\C-m . "\\cite{%l}") + (?p . "\\cite{%l}" ) + (?t . "%2a (\\cite{%l})"))) + (author-year "Do-it-yourself Author-year" + ((?\C-m . "\\cite{%l}") + (?t . "%2a (%y)\\nocite{%l}") + (?p . "(%2a %y\\nocite{%l})"))) + (locally "Full info in parenthesis" + "(%2a %y, %j %v, %P, %e: %b, %u, %s %<)") + ) + "Builtin versions of the citation format. +The following conventions are valid for all alist entries: +`?\C-m' should always point to a straight \\cite{%l} macro. +`?t' should point to a textual citation (citation as a noun). +`?p' should point to a parenthetical citation.") + +(defconst reftex-index-macros-builtin + '((default "Default \\index and \\glossary macros" + (("\\index{*}" "idx" ?i "" nil) + ("\\glossary{*}" "glo" ?g "" nil))) + (multind "The multind.sty package" + (("\\index{}{*}" 1 ?i "" nil))) + (index "The index.sty package" + (("\\index[]{*}" 1 ?i "" nil) + ("\\index*[]{*}" 1 ?I "" nil))) + (Index-Shortcut "index.sty with \\shortindexingon" + (("\\index[]{*}" 1 ?i "" nil) + ("\\index*[]{*}" 1 ?I "" nil) + ("^[]{*}" 1 ?^ "" texmathp) + ("_[]{*}" 1 ?_ "" texmathp)))) + "Builtin stuff for reftex-index-macros. +Lower-case symbols correspond to a style file of the same name in the LaTeX +distribution. Mixed-case symbols are convenience aliases.") +) + +;; Configuration Variables and User Options for RefTeX ------------------ + +(defgroup reftex nil + "LaTeX label and citation support." + :tag "RefTeX" + :link '(url-link :tag "Home Page" + "http://strw.leidenuniv.nl/~dominik/Tools/") + :link '(emacs-commentary-link :tag "Commentary in reftex.el" "reftex.el") + :link '(custom-manual "(reftex)Top") + :prefix "reftex-" + :group 'tex) + +;; Table of contents configuration -------------------------------------- + +(defgroup reftex-table-of-contents-browser nil + "A multifile table of contents browser." + :group 'reftex) + +(defcustom reftex-toc-keep-other-windows t + "*Non-nil means, split the selected window to display the *toc* buffer. +This helps to keep the window configuration, but makes the *toc* small. +When nil, all other windows except the selected one will be deleted, so +that the *toc* window fills half the frame." + :group 'reftex-table-of-contents-browser + :type 'boolean) + +(defcustom reftex-toc-include-file-boundaries nil + "*Non-nil means, include file boundaries in *toc* buffer. +This flag can be toggled from within the *toc* buffer with the `F' key." + :group 'reftex-table-of-contents-browser + :type 'boolean) + +(defcustom reftex-toc-include-labels nil + "*Non-nil means, include labels in *toc* buffer. +This flag can be toggled from within the *toc* buffer with the `l' key." + :group 'reftex-table-of-contents-browser + :type 'boolean) + +(defcustom reftex-toc-include-index-entries nil + "*Non-nil means, include index entries in *toc* buffer. +This flag can be toggled from within the *toc* buffer with the `i' key." + :group 'reftex-table-of-contents-browser + :type 'boolean) + +(defcustom reftex-toc-include-context nil + "*Non-nil means, include context with labels in the *toc* buffer. +Context will only be shown when labels are visible as well. +This flag can be toggled from within the *toc* buffer with the `c' key." + :group 'reftex-table-of-contents-browser + :type 'boolean) + +(defcustom reftex-toc-follow-mode nil + "*Non-nil means, point in *toc* buffer will cause other window to follow. +The other window will show the corresponding part of the document. +This flag can be toggled from within the *toc* buffer with the `f' key." + :group 'reftex-table-of-contents-browser + :type 'boolean) + +(defcustom reftex-revisit-to-follow nil + "*Non-nil means, follow-mode will revisit files if necessary. +When nil, follow-mode will be suspended for stuff in unvisited files." + :group 'reftex-table-of-contents-browser + :group 'reftex-referencing-labels + :type 'boolean) + +(defcustom reftex-toc-mode-hook nil + "Mode hook for reftex-toc-mode." + :group 'reftex-table-of-contents-browser + :type 'hook) + +;; Label Support Configuration + +(defgroup reftex-label-support nil + "Support for creation, insertion and referencing of labels in LaTeX." + :group 'reftex) + +(defgroup reftex-defining-label-environments nil + "Definition of environments and macros to do with label." + :group 'reftex-label-support) + +(defcustom reftex-default-label-alist-entries + '(amsmath endnotes fancybox floatfig longtable picinpar + rotating sidecap subfigure supertab wrapfig LaTeX) + "Default label alist specifications. LaTeX should always be the last entry. +The value of this variable is a list of symbols with associations in the +constant `reftex-label-alist-builtin'. Check that constant for a full list +of options." + :group 'reftex-defining-label-environments + :set 'reftex-set-dirty + :type `(set + :indent 4 + :inline t + :greedy t + ,@(mapcar + (lambda (x) + (list 'const :tag (concat (symbol-name (nth 0 x)) + ": " (nth 1 x)) + (nth 0 x))) + reftex-label-alist-builtin))) + +(defcustom reftex-label-alist nil + "Alist with information on environments for \\label-\\ref use. + +This docstring is easier to understand after reading the configuration +examples in `reftex.el'. Looking at the builtin defaults in the constant +`reftex-label-alist-builtin' may also be instructive. + +Set this variable to define additions and changes to the default. The only +things you MUST NOT change is that `?s' is the type indicator for section +labels, and SPC for the `any' label type. These are hard-coded at other +places in the code. + +The value of the variable must be a list of items. Each item is a list +itself and has the following structure: + + (ENV-OR-MACRO TYPE-KEY LABEL-PREFIX REFERENCE-FORMAT CONTEXT-METHOD + (MAGIC-WORD ... )) + +Each list entry describes either an environment carrying a counter for use +with \\label and \\ref, or a LaTeX macro defining a label as (or inside) +one of its arguments. The elements of each list entry are: + +ENV-OR-MACRO + Name of the environment (like \"table\") or macro (like \"\\\\myfig\"). + For macros, indicate the macro arguments for best results, as in + \"\\\\myfig[]{}{}{*}{}\". Use square brackets for optional arguments, + a star to mark the label argument, if any. The macro does not have to + have a label argument - you could also use \\label{..} inside one of + its arguments. + Special names: `section' for section labels, `any' to define a group + which contains all labels. + + This may also be a function to do local parsing and identify point + to be in a a non-standard label environment. The function must take + an argument BOUND and limit backward searches to this value. It + should return either nil or a cons cell (FUNCTION . POSITION) with + the function symbol and the position where the special environment + starts. See the Info documentation for an example. + + Finally this may also be nil if the entry is only meant to change + some settings associated with the type indicator character (see below). + +TYPE-KEY + Type indicator character, like `?t', must be a printable ASCII character. + The type indicator is a single character which defines a label type. + Any label inside the environment or macro is assumed to belong to this + type. The same character may occur several times in this list, to cover + cases in which different environments carry the same label type (like + `equation' and `eqnarray'). + If the type indicator is nil and the macro has a label argument {*}, + the macro defines neutral labels just like \label. In this case + the reminder of this entry is ignored. + +LABEL-PREFIX + Label prefix string, like \"tab:\". + The prefix is a short string used as the start of a label. It may be the + empty string. The prefix may contain the following `%' escapes: + %f Current file name with directory and extension stripped. + %F Current file name relative to directory of master file. + %u User login name, on systems which support this. + %S A section prefix derived with variable `reftex-section-prefixes'. + + Example: In a file `intro.tex', \"eq:%f:\" will become \"eq:intro:\"). + +REFERENCE-FORMAT + Format string for reference insert in buffer. `%s' will be replaced by + the label. + When the format starts with `~', the `~' will only be inserted if + there is not already a whitespace before point. + +CONTEXT-METHOD + Indication on how to find the short context. + - If nil, use the text following the \\label{...} macro. + - If t, use + - the section heading for section labels. + - text following the \\begin{...} statement of environments. + (not a good choice for environments like eqnarray or enumerate, + where one has several labels in a single environment). + - text after the macro name (starting with the first arg) for macros. + - If an integer, use the nth argument of the macro. As a special case, + 1000 means to get text after the last macro argument. + - If a string, use as regexp to search *backward* from the label. Context + is then the text following the end of the match. E.g. putting this to + \"\\\\\\\\caption[[{]\" will use the caption in a figure or table + environment. + \"\\\\\\\\begin{eqnarray}\\\\|\\\\\\\\\\\\\\\\\" works for eqnarrays. + - If any of `caption', `item', `eqnarray-like', `alignat-like', this + symbol will internally be translated into an appropriate regexp + (see also the variable `reftex-default-context-regexps'). + - If a function, call this function with the name of the environment/macro + as argument. On call, point will be just after the \\label macro. The + function is expected to return a suitable context string. It should + throw an exception (error) when failing to find context. + As an example, here is a function returning the 10 chars following + the label macro as context: + + (defun my-context-function (env-or-mac) + (if (> (point-max) (+ 10 (point))) + (buffer-substring (point) (+ 10 (point))) + (error \"Buffer too small\"))) + + Label context is used in two ways by RefTeX: For display in the label + menu, and to derive a label string. If you want to use a different + method for each of these, specify them as a dotted pair. + E.g. `(nil . t)' uses the text after the label (nil) for display, and + text from the default position (t) to derive a label string. This is + actually used for section labels. + +MAGIC-WORDS + List of magic words which identify a reference to be of this type. + If the word before point is equal to one of these words when calling + `reftex-reference', the label list offered will be automatically + restricted to labels of the correct type. + If the first element of this wordlist is the symbol `regexp', the + strings are interpreted as regular expressions. RefTeX will add + a \"\\\\W\" to the beginning and other stuff to the end of the regexp. + +If the type indicator characters of two or more entries are the same, RefTeX +will use + - the first non-nil format and prefix + - the magic words of all involved entries. + +Any list entry may also be a symbol. If that has an association in +`reftex-label-alist-builtin', the cddr of that association is spliced into the +list. However, builtin defaults should normally be set with the variable +`reftex-default-label-alist-entries." + :group 'reftex-defining-label-environments + :set 'reftex-set-dirty + :type + `(repeat + (choice :tag "Package or Detailed " + :value ("" ?a nil nil nil nil) + (list :tag "Detailed Entry" + :value ("" ?a nil nil nil nil) + (choice :tag "Environment or \\macro " + (const :tag "Ignore, just use typekey" nil) + (string "") + (symbol :tag "Special parser" my-parser)) + (choice :tag "Type specification " + (const :tag "unspecified, like in \\label" nil) + (character :tag "Char " ?a)) + (choice :tag "Label prefix string " + (const :tag "Default" nil) + (string :tag "String" "lab:")) + (choice :tag "Label reference format" + (const :tag "Default" nil) + (string :tag "String" "~\\ref{%s}")) + (choice :tag "Context method " + (const :tag "Default position" t) + (const :tag "After label" nil) + (number :tag "Macro arg nr" 1) + (regexp :tag "Regexp" "") + (const :tag "Caption in float" caption) + (const :tag "Item in list" item) + (const :tag "Eqnarray-like" eqnarray-like) + (const :tag "Alignat-like" alignat-like) + (symbol :tag "Function" my-func)) + (repeat :tag "Magic words" :extra-offset 2 (string))) + (choice + :tag "Package" + :value AMSTeX + ,@(mapcar + (lambda (x) + (list 'const :tag (concat (symbol-name (nth 0 x))) + (nth 0 x))) + reftex-label-alist-builtin))))) + +;; LaTeX section commands and level numbers +(defcustom reftex-section-levels + '( + ("part" . 0) + ("chapter" . 1) + ("section" . 2) + ("subsection" . 3) + ("subsubsection" . 4) + ("paragraph" . 5) + ("subparagraph" . 6) + ("subsubparagraph" . 7) + ("addchap" . -1) ; KOMA-Script + ("addsec" . -2) ; KOMA-Script +;;; ("minisec" . -7) ; KOMA-Script + ) + "Commands and levels used for defining sections in the document. +This is an alist with each element like (COMMAND-NAME . LEVEL). +The car of each cons cell is the name of the section macro (without +the backslash). The cdr is a number indicating its level. A negative +level means the same level as the positive value, but the section will +never get a number." + :group 'reftex-defining-label-environments + :set 'reftex-set-dirty + :type '(repeat + (cons (string :tag "sectioning macro" "") + (number :tag "level " 0)))) + +(defcustom reftex-section-prefixes '((0 . "part:") (1 . "cha:") (t . "sec:")) + "Prefixes for section labels. +When the label prefix given in an entry in `reftex-label-alist' contains `%S', +this list is used to determine the correct prefix string depending on the +current section level. +The list is an alist, with each entry of the form (KEY . PREFIX) +Possible keys are sectioning macro names like `chapter', section levels +(as given in `reftex-section-levels'), and t for the default." + :group 'reftex-defining-label-environments + :type '(repeat + (cons :value (0 . "") + (choice + (string :tag "macro name") + (integer :tag "section level") + (const :tag "default" t)) + (string :tag "Prefix")))) + +(defcustom reftex-default-context-regexps + '((caption . "\\\\\\(rot\\)?caption\\*?[[{]") + (item . "\\\\item\\(\\[[^]]*\\]\\)?") + (eqnarray-like . "\\\\begin{%s}\\|\\\\\\\\") + (alignat-like . "\\\\begin{%s}{[0-9]*}\\|\\\\\\\\")) +"Alist with default regular expressions for finding context. +The form (format regexp (regexp-quote environment)) is used to calculate +the final regular expression - so %s will be replaced with the environment +or macro." + :group 'reftex-defining-label-environments + :type '(repeat (cons (symbol) (regexp)))) + +(defcustom reftex-special-environment-functions nil + "List of functions to be called when trying to figure out current environment. +These are special functions to detect \"environments\" which do not +start with \\begin and end with \\end. Some LaTeX packages seem to +use such non-standard ways to set up environment-like constructs. The +purpose of each function in this list is to detect if point is +currently inside such a special \"environment\". If the environment +carries a label, you must also set up an entry for it in +`reftex-label-alist'. + +The function should check if point is currently in the special +environment it was written to detect. If so, the function must return +a cons cell (NAME . POSITION). NAME is the name of the environment +detected and POSITION is the buffer position where the environment +starts. The function must return nil on failure to detect the +environment. + +The function must take an argument BOUND. If non-nil, BOUND is a +boundary for backwards searches which should be observed. + +Here is an example. The LaTeX package linguex.sty defines list macros +`\\ex.', `\\a.', etc for lists which are terminated by `\\z.' or an empty +line. + + \\ex. \\label{ex:12} Some text in an exotic language ... + \\a. \\label{ex:13} more stuff + \\b. \\label{ex:14} still more stuff + + ... more text after the empty line terminating all lists + +And here is the setup for RefTeX: + +1. Define a dummy environment for this in `reftex-label-alist'. Dummy means, + make up an environment name even though it is not used with \\begin and + \\end. Here we use \"linguex\" as this name. + + (setq reftex-label-alist + '((\"linguex\" ?x \"ex:\" \"~\\\\ref{%s}\" nil (\"Example\" \"Ex.\")))) + +2. Write a function to detect the list macros and the determinators as well. + + (defun my-detect-linguex-list (bound) + (let ((pos (point)) p1) + (save-excursion + ;; Search for any of the linguex item macros at the beginning of a line + (if (re-search-backward + \"^[ \\t]*\\\\(\\\\\\\\\\\\(ex\\\\|a\\\\|b\\\\|c\\\\|d\\\\|e\\\\|f\\\\)g?\\\\.\\\\)\" bound t) + (progn + (setq p1 (match-beginning 1)) + ;; Make sure no empty line or \\z. is between us and the item macro + (if (re-search-forward \"\\n[ \\t]*\\n\\\\|\\\\\\\\z\\\\.\" pos t) + ;; Return nil because list was already closed + nil + ;; OK, we got it + (cons \"linguex\" p1))) + ;; Return nil for not found + nil)))) + +3. Tell RefTeX to use this function + + (setq reftex-special-environment-functions '(my-detect-linguex-list)) +" + :group 'reftex-defining-label-environments + :type 'hook) + +;; Label insertion + +(defgroup reftex-making-and-inserting-labels nil + "Options on how to create new labels." + :group 'reftex-label-support) + +(defcustom reftex-insert-label-flags '("s" "sft") + "Flags governing label insertion. First flag DERIVE, second flag PROMPT. + +If DERIVE is t, RefTeX will try to derive a sensible label from context. +A section label for example will be derived from the section heading. +The conversion of the context to a legal label is governed by the +specifications given in `reftex-derive-label-parameters'. +If RefTeX fails to derive a label, it will prompt the user. +If DERIVE is nil, the label generated will consist of the prefix and a +unique number, like `eq:23'. + +If PROMPT is t, the user will be prompted for a label string. The prompt will +already contain the prefix, and (if DERIVE is t) a default label derived from +context. When PROMPT is nil, the default label will be inserted without +query. + +So the combination of DERIVE and PROMPT controls label insertion. Here is a +table describing all four possibilities: + +DERIVE PROMPT ACTION +------------------------------------------------------------------------- + nil nil Insert simple label, like eq:22 or sec:13. No query. + nil t Prompt for label. + t nil Derive a label from context and insert without query. + t t Derive a label from context and prompt for confirmation. + +Each flag may be set to t, nil, or a string of label type letters +indicating the label types for which it should be true. The strings work +like character classes. +Thus, the combination may be set differently for each label type. The +default settings \"s\" and \"sft\" mean: Derive section labels from headings +(with confirmation). Prompt for figure and table labels. Use simple labels +without confirmation for everything else. +The available label types are: s (section), f (figure), t (table), i (item), +e (equation), n (footnote), N (endnote), plus any definitions in +`reftex-label-alist'." + :group 'reftex-making-and-inserting-labels + :type '(list (choice :tag "Derive label from context" + (const :tag "always" t) + (const :tag "never" nil) + (string :tag "selected label types" "")) + (choice :tag "Prompt for label string " + :entry-format " %b %v" + (const :tag "always" t) + (const :tag "never" nil) + (string :tag "selected label types" "")))) + +(defcustom reftex-string-to-label-function 'reftex-string-to-label + "Function to turn an arbitrary string into a legal label. +RefTeX's default function uses the variable `reftex-derive-label-parameters'." + :group 'reftex-making-and-inserting-labels + :type 'symbol) + +(defcustom reftex-translate-to-ascii-function 'reftex-latin1-to-ascii + "Filter function which will process a context string before it is used +to derive a label from it. The intended application is to convert ISO or +Mule characters into something legal in labels. The default function +removes the accents from Latin-1 characters. X-Symbol (>=2.6) sets this +variable to the much more general `x-symbol-translate-to-ascii'." + :group 'reftex-making-and-inserting-labels + :type 'symbol) + +(defcustom reftex-derive-label-parameters '(3 20 t 1 "-" + ("the" "on" "in" "off" "a" "for" "by" "of" "and" "is" "to") t) + "Parameters for converting a string into a label. +This variable is a list of the following items. + +NWORDS Number of words to use. +MAXCHAR Maximum number of characters in a label string. +ILLEGAL nil: Throw away any words containing characters illegal in labels. + t: Throw away only the illegal characters, not the whole word. +ABBREV nil: Never abbreviate words. + t: Always abbreviate words (see `reftex-abbrev-parameters'). + not t and not nil: Abbreviate words if necessary to shorten + label string below MAXCHAR. +SEPARATOR String separating different words in the label. +IGNOREWORDS List of words which should not be part of labels. +DOWNCASE t: Downcase words before using them." + :group 'reftex-making-and-inserting-labels + :type '(list (integer :tag "Number of words " 3) + (integer :tag "Maximum label length " 20) + (choice :tag "Illegal characters in words" + (const :tag "throw away entire word" nil) + (const :tag "throw away single chars" t)) + (choice :tag "Abbreviate words " + (const :tag "never" nil) + (const :tag "always" t) + (const :tag "when label is too long" 1)) + (string :tag "Separator between words " "-") + (repeat :tag "Ignore words" + :entry-format " %i %d %v" + (string :tag "")) + (option (boolean :tag "Downcase words ")))) + +(defcustom reftex-label-illegal-re "[^-a-zA-Z0-9_+=:;,.]" + "Regexp matching characters not legal in labels." + :group 'reftex-making-and-inserting-labels + :type '(regexp :tag "Regular Expression")) + +(defcustom reftex-abbrev-parameters '(4 2 "^aeiou" "aeiou") + "Parameters for abbreviation of words. +This variable is a list of the following items. + +MIN-CHARS Minimum number of characters remaining after abbreviation. +MIN-KILL Minimum number of characters to remove when abbreviating words. +BEFORE Character class before abbrev point in word. +AFTER Character class after abbrev point in word." + :group 'reftex-making-and-inserting-labels + :type '(list + (integer :tag "Minimum chars per word" 4) + (integer :tag "Shorten by at least " 2) + (string :tag "cut before char class " "^saeiou") + (string :tag "cut after char class " "aeiou"))) + +(defcustom reftex-format-label-function nil + "Function which produces the string to insert as a label definition. +Normally should be nil, unless you want to do something fancy. +The function will be called with two arguments, the LABEL and the DEFAULT +FORMAT, which usually is `\label{%s}'. The function should return the +string to insert into the buffer." + :group 'reftex-making-and-inserting-labels + :type 'function) + +;; Label referencing + +(defgroup reftex-referencing-labels nil + "Options on how to reference labels." + :group 'reftex-label-support) + +(eval-and-compile + (defconst reftex-tmp + '((const :tag "on" t) + (const :tag "off" nil) + (string :tag "Selected label types")))) + +(defcustom reftex-label-menu-flags '(t t nil nil nil nil t nil) + "List of flags governing the label menu makeup. +The flags are: + +TABLE-OF-CONTENTS Show the labels embedded in a table of context. +SECTION-NUMBERS Include section numbers (like 4.1.3) in table of contents. +COUNTERS Show counters. This just numbers the labels in the menu. +NO-CONTEXT Non-nil means do NOT show the short context. +FOLLOW Follow full context in other window. +SHOW-COMMENTED Show labels from regions which are commented out. +MATCH-IN-TOC Obsolete flag. +SHOW FILES Show begin and end of included files. + +Each of these flags can be set to t or nil, or to a string of type letters +indicating the label types for which it should be true. These strings work +like character classes in regular expressions. Thus, setting one of the +flags to \"sf\" makes the flag true for section and figure labels, nil +for everything else. Setting it to \"^sf\" makes it the other way round. +The available label types are: s (section), f (figure), t (table), i (item), +e (equation), n (footnote), plus any definitions in `reftex-label-alist'. + +Most options can also be switched from the label menu itself - so if you +decide here to not have a table of contents in the label menu, you can still +get one interactively during selection from the label menu." + :group 'reftex-referencing-labels + :type + `(list + (choice :tag "Embed in table of contents " ,@reftex-tmp) + (choice :tag "Show section numbers " ,@reftex-tmp) + (choice :tag "Show individual counters " ,@reftex-tmp) + (choice :tag "Hide short context " ,@reftex-tmp) + (choice :tag "Follow context in other window " ,@reftex-tmp) + (choice :tag "Show commented labels " ,@reftex-tmp) + (choice :tag "Obsolete flag, Don't use. " ,@reftex-tmp) + (choice :tag "Show begin/end of included files" ,@reftex-tmp))) + +(defcustom reftex-multiref-punctuation '((?, . ", ") (?- . "--") (?+ . " and ")) + "Punctuation strings for multiple references. +When marking is used in the selection buffer to select several references, +this variable associates the 3 marking characters `,-+' with prefix strings +to be inserted into the buffer before the corresponding \ref macro. +This is used to string together whole reference sets, like +`eqs. 1,2,3-5,6 and 7' in a single call to `reftex-reference'. See manual." + :group 'reftex-referencing-labels + :type '(repeat (cons (character) (string)))) + +(defcustom reftex-vref-is-default nil + "*Non-nil means, the varioref macro \\vref is used as default. +In the selection buffer, the `v' key toggles the reference macro between +`\\ref' and `\\vref'. The value of this variable determines the default +which is active when entering the selection process. +Instead of nil or t, this may also be a string of type letters indicating +the label types for which it should be true." + :group 'reftex-referencing-labels + :type `(choice :tag "\\vref is default macro" ,@reftex-tmp)) + +(defcustom reftex-fref-is-default nil + "*Non-nil means, the fancyref macro \\fref is used as default. +In the selection buffer, the `V' key toggles the reference macro between +`\\ref', `\\fref' and `\\Fref'. The value of this variable determines +the default which is active when entering the selection process. +Instead of nil or t, this may also be a string of type letters indicating +the label types for which it should be true." + :group 'reftex-referencing-labels + :type `(choice :tag "\\fref is default macro" ,@reftex-tmp)) + +(defcustom reftex-level-indent 2 + "*Number of spaces to be used for indentation per section level." + :group 'reftex-referencing-labels + :type 'integer) + +(defcustom reftex-guess-label-type t + "*Non-nil means, `reftex-reference' will try to guess the label type. +To do that, RefTeX will look at the word before the cursor and compare it with +the words given in `reftex-label-alist'. When it finds a match, RefTeX will +immediately offer the correct label menu - otherwise it will prompt you for +a label type. If you set this variable to nil, RefTeX will always prompt." + :group 'reftex-referencing-labels + :type 'boolean) + +(defcustom reftex-format-ref-function nil + "Function which produces the string to insert as a reference. +Normally should be nil, because the format to insert a reference can +already be specified in `reftex-label-alist'. +This hook also is used by the special commands to insert `\vref' and `\fref' +references, so even if you set this, your setting will be ignored by +the special commands. +The function will be called with two arguments, the LABEL and the DEFAULT +FORMAT, which normally is `~\ref{%s}'. The function should return the +string to insert into the buffer." + :group 'reftex-referencing-labels + :type 'function) + +(defcustom reftex-select-label-mode-hook nil + "Mode hook for reftex-select-label-mode." + :group 'reftex-referencing-labels + :type 'hook) + +;; BibteX citation configuration ---------------------------------------- + +(defgroup reftex-citation-support nil + "Support for referencing bibliographic data with BibTeX." + :group 'reftex) + +(defvar reftex-bibfile-ignore-list nil) ; compatibility +(defcustom reftex-bibfile-ignore-regexps nil + "*List of regular expressions to exclude files in \\bibliography{..}. +File names matched by these regexps will not be parsed by RefTeX. +Intended for files which contain only `@string' macro definitions and the +like, which are ignored by RefTeX anyway." + :group 'reftex-citation-support + :set 'reftex-set-dirty + :type '(repeat (regexp))) + +(defcustom reftex-default-bibliography nil + "*List of BibTeX database files which should be used if none are specified. +When `reftex-citation' is called from a document which has neither a +`\bibliography{..}' statement nor a `thebibliography' environment, +RefTeX will scan these files instead. Intended for using `reftex-citation' +in non-LaTeX files. The files will be searched along the BIBINPUTS or TEXBIB +path." + :group 'reftex-citation-support + :type '(repeat (file))) + +(defcustom reftex-sort-bibtex-matches 'reverse-year + "*Sorting of the entries found in BibTeX databases by reftex-citation. +Possible values: +nil Do not sort entries. +'author Sort entries by author name. +'year Sort entries by increasing year. +'reverse-year Sort entries by decreasing year." + :group 'reftex-citation-support + :type '(choice (const :tag "not" nil) + (const :tag "by author" author) + (const :tag "by year" year) + (const :tag "by year, reversed" reverse-year))) + +(defcustom reftex-cite-format 'default + "*The format of citations to be inserted into the buffer. +It can be a string or an alist or a symbol. In the simplest case this +is just the string \"\\cite{%l}\", which is also the default. See the +definition of `reftex-cite-format-builtin' for more complex examples. + +If `reftex-cite-format' is a string, it will be used as the format. +In the format, the following percent escapes will be expanded. + +%l The BibTeX label of the citation. +%a List of author names, see also `reftex-cite-punctuation. +%2a Like %a, but abbreviate more than 2 authors like Jones et al. +%A First author name only. +%e Works like %a, but on list of editor names. (%2e and %E work a well) + +It is also possible to access all other BibTeX database fields: +%b booktitle %c chapter %d edition %h howpublished +%i institution %j journal %k key %m month +%n number %o organization %p pages %P first page +%r address %s school %u publisher %t title +%v volume %y year +%B booktitle, abbreviated %T title, abbreviated + +Usually, only %l is needed. The other stuff is mainly for the echo area +display, and for (setq reftex-comment-citations t). + +%< as a special operator kills punctuation and space around it after the +string has been formatted. + +Beware that all this only works with BibTeX database files. When +citations are made from the \\bibitems in an explicit thebibliography +environment, only %l is available. + +If `reftex-cite-format' is an alist of characters and strings, the user +will be prompted for a character to select one of the possible format +strings. + In order to configure this variable, you can either set +`reftex-cite-format' directly yourself or set it to the SYMBOL of one of +the predefined styles. The predefined symbols are those which have an +association in the constant `reftex-cite-format-builtin'. +E.g.: (setq reftex-cite-format 'natbib)" + :group 'reftex-citation-support + :type + `(choice + :format "%{%t%}: \n%[Value Menu%] %v" + (radio :tag "Symbolic Builtins" + :indent 4 + :value default + ,@(mapcar + (lambda (x) + (list 'const :tag (concat (symbol-name (nth 0 x)) + ": " (nth 1 x)) + (nth 0 x))) + reftex-cite-format-builtin)) + (string :tag "format string" "\\cite{%l}") + (repeat :tag "key-ed format strings" + :value ((?\r . "\\cite{%l}") + (?t . "\\cite{%l}") (?p . "\\cite{%l}")) + (cons (character :tag "Key character" ?\r) + (string :tag "Format string" ""))))) + +(defcustom reftex-comment-citations nil + "*Non-nil means add a comment for each citation describing the full entry. +The comment is formatted according to `reftex-cite-comment-format'." + :group 'reftex-citation-support + :type 'boolean) + +(defcustom reftex-cite-comment-format + "%% %2a %y, %j %v, %P, %b, %e, %u, %s %<\n" + "Citation format used for commented citations. Must NOT contain %l. +See the variable `reftex-cite-format' for possible percent escapes." + :group 'reftex-citation-support + :type 'string) + +(defcustom reftex-cite-view-format "%2a %y, %T, %B, %j %v:%P, %s %<" + "Citation format used to display citation info in the message area. +Must NOT contain %l. See the variable `reftex-cite-format' for +possible percent escapes." + :group 'reftex-citation-support + :group 'reftex-viewing-cross-references + :type 'string) + +(defcustom reftex-cite-punctuation '(", " " \\& " " {\\it et al.}") + "Punctuation for formatting of name lists in citations. +This is a list of 3 strings. +1. normal names separator, like \", \" in Jones, Brown and Miller +2. final names separator, like \" and \" in Jones, Brown and Miller +3. The \"et al\" string, like \" {\\it et al.}\" in Jones {\\it et al.}" + :group 'reftex-citation-support + :type '(list + (string :tag "Separator for names ") + (string :tag "Separator for last name in list") + (string :tag "string used as et al. "))) + +(defcustom reftex-format-cite-function nil + "Function which produces the string to insert as a citation. +Normally should be nil, because the format to insert a reference can +already be specified in `reftex-cite-format'. +The function will be called with two arguments, the CITATION KEY and the +DEFAULT FORMAT, which is taken from `reftex-cite-format'. The function +should return the string to insert into the buffer." + :group 'reftex-citation-support + :type 'function) + +(defcustom reftex-select-bib-mode-hook nil + "Mode hook for reftex-select-bib-mode." + :group 'reftex-citation-support + :type 'hook) + +;; Index Support Configuration + +(defgroup reftex-index-support nil + "Support for viewing and editing the index." + :group 'reftex) + +(defcustom reftex-support-index t + "*Non-nil means, index entries are parsed as well. +Index support is resource intensive and the internal structure holding the +parsed information can become quite big. Therefore it can be turned off. +When this is nil and you execute a command which requires index support, +you will be asked for confirmation to turn it on and rescan the document." + :group 'reftex-index-support + :type 'boolean) + +(defcustom reftex-index-special-chars '("!" "|" "@" "\"" "\\") + "Special characters in index entries. The value is a list of five strings. +These correspond to the makeindex keywords LEVEL ENCAP ACTUAL QUOTE ESCAPE." + :group 'reftex-index-support + :type '(list + (string :tag "LEVEL separator") + (string :tag "ENCAP char ") + (string :tag "ACTUAL char ") + (string :tag "QUOTE char ") + (string :tag "ESCAPE char "))) + +(defcustom reftex-index-macros nil + "Macros which define index entries. The structure is + +(MACRO INDEX-TAG KEY PREFIX EXCLUDE) + +MACRO is the macro. Arguments should be denoted by empty braces like +\\index[]{*}. Use square brackets to denote optional arguments. The star +marks where the index key is. + +INDEX-TAG is a short name of the index. \"idx\" and \"glo\" are +reserved for the default index and the glossary. Other indices can be +defined as well. If this is an integer, the Nth argument of the macro +holds the index tag. + +KEY is a character which is used to identify the macro for input with +\\[reftex-index]. ?i, ?I, and ?g are reserved for default index and glossary. + +PREFIX can be a prefix which is added to the KEY part of the index entry. +If you have a macro \\newcommand{\\molec}[1]{#1\\index{Molecules!#1}}, this +prefix should be \"Molecules!\". See the manual for details. + +EXCLUDE can be a function. If this function exists and returns a non-nil +value, the index entry at point is ignored. This was implemented to support +the (deprecated) `^' and `_' shortcuts in the LaTeX2e `index' package. + +The final entry may also be a symbol if this entry has a association +in the variable `reftex-index-macros-builtin' to specify the main +indexing package you are using. Legal values are currently +default The LaTeX default - unnecessary to specify this one +multind The multind.sty package +index The index.sty package +index-shortcut The index.sty packages with the ^ and _ shortcuts. + Should not be used - only for old documents. +Note that AUCTeX sets these things internally for RefTeX as well, so +with a sufficiently new version of AUCTeX, you should not set the +package here." + :group 'reftex-index-support + :set 'reftex-set-dirty + :type `(list + (repeat + :inline t + (list :value ("" "idx" ?a "" nil) + (string :tag "Macro with args") + (choice :tag "Index Tag " + (string) + (integer :tag "Macro arg Nr" :value 1)) + (character :tag "Access Key ") + (string :tag "Key Prefix ") + (symbol :tag "Exclusion hook "))) + (option + :tag "Package:" + (choice :tag "Package" + :value index + ,@(mapcar + (lambda (x) + (list 'const :tag (concat (symbol-name (nth 0 x)) + ": " (nth 1 x)) + (nth 0 x))) + reftex-index-macros-builtin))))) + +(defcustom reftex-index-default-macro '(?i "idx" t) + "The default index macro for \\[reftex-index-selection-or-word]. +This is a list with (MACRO-KEY DEFAULT-TAG REPEAT-WORD). + +MACRO-KEY: Character identifying an index macro - see `reftex-index-macros'. +DEFAULT-TAG: This is the tag to be used if the macro requires a TAG argument. + When this is nil and a TAG is needed, RefTeX will ask for it. + When this is the empty string and the TAG argument of the index + macro is optional, the TAG argument will be omitted. +REPEAT-WORD: Non-nil means, the index macro does not typeset the entry in + the text, so that the text has to be repeated outside the index + macro." + :group 'reftex-index-support + :type '(list + (character :tag "Character identifying default macro") + (choice :tag "Default index tag " + (const nil) + (string)) + (boolean :tag "Word needs to be repeated "))) + +(defcustom reftex-index-default-tag "idx" + "Default index tag. +When working with multiple indexes, RefTeX queries for an index tag when +creating index entries or displaying a specific index. This variable controls +the default offered for these queries. The default can be selected with RET +during selection or completion. Legal values of this variable are: + +nil Do not provide a default index +\"tag\" The default index tag given as a string, e.g. \"idx\". +last The last used index tag will be offered as default." + :group 'reftex-index-support + :type '(choice + (const :tag "no default" nil) + (const :tag "last used " 'last) + (string :tag "index tag " "idx"))) + +(defcustom reftex-index-math-format "$%s$" + "Format of index entries when copied from inside math mode. +When `reftex-index-selection-or-word' is executed inside TeX math mode, +the index key copied from the buffer is processed with this format string +through the `format' function. This can be used to add the math delimiters +(e.g. `$') to the string. +Requires the `texmathp.el' library which is part of AUCTeX." + :group 'reftex-index-support + :type 'string) + +(defcustom reftex-index-section-letters "ABCDEFGHIJKLMNOPQRSTUVWXYZ" + "The letters which denote sections in the index. +Usually these are all capital letters. Don't use any downcase letters. +Order is not significant, the index will be sorted by whatever the sort +function thinks is correct. +In addition to these letters, RefTeX will create a group `!' which +contains all entries sorted below the lowest specified letter. +In the index buffer, pressing any of these capital letters or `!' will jump +to that section." + :group 'reftex-index-support + :type '(string :tag "Capital letters")) + +(defcustom reftex-index-include-context nil + "*Non-nil means, display the index definition context in the index buffer. +This flag may also be toggled from the index buffer with the `c' key." + :group 'reftex-index-support + :type 'boolean) + +(defcustom reftex-index-follow-mode nil + "*Non-nil means, point in *Index* buffer will cause other window to follow. +The other window will show the corresponding part of the document. +This flag can be toggled from within the *Index* buffer with the `f' key." + :group 'reftex-table-of-contents-browser + :type 'boolean) + +;; Viewing Cross References + +(defgroup reftex-viewing-cross-references nil + "Displaying cross references and citations." + :group 'reftex) + +(defcustom reftex-view-crossref-extra nil + "Macros which can be used for the display of cross references. +This is used when `reftex-view-crossref' is called with point in an +argument of a macro. Note that crossref viewing for citations, +references (both ways) and index entries is hard-coded. This variable +is only to configure additional structures for which crossreference +viewing can be useful. Each entry has the structure + +(MACRO-RE SEARCH-RE HIGHLIGHT). + +MACRO-RE is matched against the macro. SEARCH-RE is the regexp used +to search for cross references. `%s' in this regexp is replaced with +with the macro argument at point. HIGHLIGHT is an integer indicating +which subgroup of the match should be highlighted." + :group 'reftex-viewing-cross-references + :type '(repeat (group (regexp :tag "Macro Regexp ") + (string :tag "Search Regexp ") + (integer :tag "Highlight Group")))) + +(defcustom reftex-auto-view-crossref t + "*Non-nil means, initially turn automatic viewing of crossref info on. +Automatic viewing of crossref info normally uses the echo area. +Whenever point is on the argument of a \\ref or \\cite macro, and no +other message is being displayed, the echo area will display +information about that cross reference. You can also set the variable +to the symbol `window'. In this case a small temporary window is +used for the display. +This feature can be turned on and of from the menu +(Ref->Options)." + :group 'reftex-viewing-cross-references + :type '(choice (const :tag "off" nil) + (const :tag "in Echo Area" t) + (const :tag "in Other Window" window))) + +(defcustom reftex-idle-time 1.2 + "*Time (secs) Emacs has to be idle before automatic crossref display is done." + :group 'reftex-viewing-cross-references + :type 'number) + +(defcustom reftex-revisit-to-echo nil + "*Non-nil means, automatic citation display will revisit files if necessary. +When nil, citation display in echo area will only be active for cached +entries and for BibTeX database files with live associated buffers." + :group 'reftex-viewing-cross-references + :type 'boolean) + +(defcustom reftex-cache-cite-echo t + "*Non-nil means, the information displayed in the echo area for cite macros +is cached and even saved along with the parsing information. The cache +survives document scans. In order to clear it, use M-x reftex-reset-mode." + :group 'reftex-viewing-cross-references + :type 'boolean) + +(defcustom reftex-display-copied-context-hook nil + "Normal Hook which is run before context is displayed anywhere. Designed +for X-Symbol, but may have other uses as well." + :group 'reftex-viewing-cross-references + :group 'reftex-referencing-labels + :type 'hook) + +;; Finding Files -------------------------------------------------------- + +(defgroup reftex-finding-files nil + "Finding files on search paths." + :group 'reftex) + +(defcustom reftex-texpath-environment-variables '("TEXINPUTS") + "*List of specifications how to retrieve the search path for TeX files. +Several entries are possible. +- If an element is the name of an environment variable, its content is used. +- If an element starts with an exclamation mark, it is used as a command + to retrieve the path. A typical command with the kpathsearch library would + be `!kpsewhich -show-path=.tex'. +- Otherwise the element itself is interpreted as a path. +Multiple directories can be separated by the system dependent `path-separator'. +Directories ending in `//' or `!!' will be expanded recursively. +See also `reftex-use-external-file-finders'." + :group 'reftex-finding-files + :set 'reftex-set-dirty + :type '(repeat (string :tag "Specification"))) + +(defcustom reftex-bibpath-environment-variables '("BIBINPUTS" "TEXBIB") + "*List of specifications how to retrieve search path for .bib database files. +Several entries are possible. +- If an element is the name of an environment variable, its content is used. +- If an element starts with an exclamation mark, it is used as a command + to retrieve the path. A typical command with the kpathsearch library would + be `!kpsewhich -show-path=.bib'. +- Otherwise the element itself is interpreted as a path. +Multiple directories can be separated by the system dependent `path-separator'. +Directories ending in `//' or `!!' will be expanded recursively. +See also `reftex-use-external-file-finders'." + :group 'reftex-citation-support + :group 'reftex-finding-files + :set 'reftex-set-dirty + :type '(repeat (string :tag "Specification"))) + +(defcustom reftex-file-extensions '(("tex" . (".tex" ".ltx")) + ("bib" . (".bib"))) + "*Association list with file extensions for different file types. +This is a list of items, each item is like: (TYPE . (DEF-EXT OTHER-EXT ...)) + +TYPE: File type like \"bib\" or \"tex\". +DEF-EXT: The default extension for that file type, like \".tex\" or \".bib\". +OTHER-EXT: Any number of other legal extensions for this file type. + +When a files is searched and it does not have any of the legal extensions, +we try the default extension first, and then the naked file name." + :group 'reftex-finding-files + :type '(repeat (cons (string :tag "File type") + (repeat (string :tag "Extension"))))) + +(defcustom reftex-search-unrecursed-path-first t + "*Non-nil means, search all specified directories before trying recursion. +Thus, in a path \".//:/tex/\", search first \"./\", then \"/tex/\" and then +all subdirectories of \"./\". If this option is nil, the subdirectories of +\"./\" are searched before \"/tex/\". This is mainly for speed - most of the +time the recursive path is for the system files and not for the user files. +Set this to nil if the default makes RefTeX finding files with equal names +in wrong sequence." + :group 'reftex-finding-files + :type 'boolean) + +(defcustom reftex-use-external-file-finders nil + "*Non-nil means, use external programs to find files. +Normally, RefTeX searches the paths given in the environment variables +TEXINPUTS and BIBINPUTS to find TeX files and BibTeX database files. +With this option turned on, it calls an external program specified in the +option `reftex-external-file-finders' instead. As a side effect, +the variables `reftex-texpath-environment-variables' and +`reftex-bibpath-environment-variables' will be ignored." + :group 'reftex-finding-files + :type 'boolean) + +(defcustom reftex-external-file-finders '(("tex" . "kpsewhich -format=.tex %f") + ("bib" . "kpsewhich -format=.bib %f")) + "*Association list with external programs to call for finding files. +Each entry is a cons cell (TYPE . PROGRAM). +TYPE is either \"tex\" or \"bib\". PROGRAM is the external program to use with +any arguments. %f will be replaced by the name of the file to be found. +Note that these commands will be executed directly, not via a shell. +Only relevant when `reftex-use-external-file-finders' is non-nil." + :group 'reftex-finding-files + :type '(repeat (cons (string :tag "File type") + (string :tag "Program ")))) + +;; Tuning the parser ---------------------------------------------------- + +(defgroup reftex-optimizations-for-large-documents nil + "Configuration of parser speed and memory usage." + :group 'reftex) + +(defcustom reftex-keep-temporary-buffers 1 + "*Non-nil means, keep buffers created for parsing and lookup. +RefTeX sometimes needs to visit files related to the current document. +We distinguish files visited for +PARSING: Parts of a multifile document loaded when (re)-parsing the document. +LOOKUP: BibTeX database files and TeX files loaded to find a reference, + to display label context, etc. +The created buffers can be kept for later use, or be thrown away immediately +after use, depending on the value of this variable: + +nil Throw away as much as possible. +t Keep everything. +1 Throw away buffers created for parsing, but keep the ones created + for lookup. + +If a buffer is to be kept, the file is visited normally (which is potentially +slow but will happen only once). +If a buffer is to be thrown away, the initialization of the buffer depends +upon the variable `reftex-initialize-temporary-buffers'." + :group 'reftex-optimizations-for-large-documents + :type '(choice + (const :tag "Throw away everything" nil) + (const :tag "Keep everything" t) + (const :tag "Keep lookup buffers only" 1))) + +(defcustom reftex-initialize-temporary-buffers nil + "*Non-nil means do initializations even when visiting file temporarily. +When nil, RefTeX may turn off find-file hooks and other stuff to briefly +visit a file. +When t, the full default initializations are done (find-file-hook etc.). +Instead of t or nil, this variable may also be a list of hook functions to +do a minimal initialization." + :group 'reftex-optimizations-for-large-documents + :type '(choice + (const :tag "Read files literally" nil) + (const :tag "Fully initialize buffers" t) + (repeat :tag "Hook functions" :value (nil) + (function-item)))) + +(defcustom reftex-no-include-regexps '("\\.pstex_t\\'") + "*List of regular expressions to exclude certain input files from parsing. +If the name of a file included via \\include or \\input is matched by any +of the regular expressions in this list, that file is not parsed by RefTeX." + :group 'reftex-optimizations-for-large-documents + :type '(repeat (regexp))) + +(defcustom reftex-enable-partial-scans nil + "*Non-nil means, re-parse only 1 file when asked to re-parse. +Re-parsing is normally requested with a `C-u' prefix to many RefTeX commands, +or with the `r' key in menus. When this option is t in a multifile document, +we will only parse the current buffer, or the file associated with the label +or section heading near point in a menu. Requesting re-parsing of an entire +multifile document then requires a `C-u C-u' prefix or the capital `R' key +in menus." + :group 'reftex-optimizations-for-large-documents + :type 'boolean) + +(defcustom reftex-allow-automatic-rescan t + "*Non-nil means, RefTeX may rescan the document when this seems necessary. +Currently this applies only to rescanning after label insertion, when +the new label cannot be inserted correctly into the internal label +list." + :group 'reftex-optimizations-for-large-documents + :type 'boolean) + +(defcustom reftex-save-parse-info nil + "*Non-nil means, save information gathered with parsing in a file. +The file MASTER.rel in the same directory as MASTER.tex is used to save the +information. When this variable is t, +- accessing the parsing information for the first time in an editing session + will read that file (if available) instead of parsing the document. +- exiting Emacs or killing a buffer in reftex-mode will cause a new version + of the file to be written." + :group 'reftex-optimizations-for-large-documents + :type 'boolean) + +(defcustom reftex-use-multiple-selection-buffers nil + "*Non-nil means use a separate selection buffer for each label type. +These buffers are kept from one selection to the next and need not to be +created for each use - so the menu generally comes up faster. The +selection buffers will be erased (and therefore updated) automatically +when new labels in its category are added. See the variable +`reftex-auto-update-selection-buffers'." + :group 'reftex-optimizations-for-large-documents + :group 'reftex-referencing-labels + :type 'boolean) + +(defcustom reftex-auto-update-selection-buffers t + "*Non-nil means, selection buffers will be updated automatically. +When a new label is defined with `reftex-label', all selection buffers +associated with that label category are emptied, in order to force an +update upon next use. When nil, the buffers are left alone and have to be +updated by hand, with the `g' key from the label selection process. +The value of this variable will only have any effect when +`reftex-use-multiple-selection-buffers' is non-nil." + :group 'reftex-optimizations-for-large-documents + :group 'reftex-referencing-labels + :type 'boolean) + +;; Fontification and Faces ---------------------------------------------- + +(defgroup reftex-fontification-configurations nil + "Options concerning the faces used in RefTeX." + :group 'reftex) + +(defcustom reftex-use-fonts t + "*Non-nil means, use fonts in *toc* and selection buffers. +Font-lock must be loaded as well to actually get fontified display. +When changing this option, a rescan may be necessary to activate the change." + :group 'reftex-fontification-configurations + :type 'boolean) + +(defcustom reftex-refontify-context 1 + "*Non-nil means, re-fontify the context in the label menu with font-lock. +This slightly slows down the creation of the label menu. It is only necessary +when you definitely want the context fontified. + +This option may have 3 different values: +nil Never refontify. +t Always refontify. +1 Refontify when absolutely necessary, e.g. when old versions of X-Symbol. +The option is ignored when `reftex-use-fonts' is nil." + :group 'reftex-fontification-configurations + :group 'reftex-referencing-labels + :type '(choice + (const :tag "Never" nil) + (const :tag "Always" t) + (const :tag "When necessary" 1))) + +(defcustom reftex-highlight-selection 'cursor + "*Non-nil mean, highlight selected text in selection and *toc* buffers. +Normally, the text near the cursor is the selected text, and it is +highlighted. This is the entry most keys in the selction and *toc* +buffers act on. However, if you mainly use the mouse to select an +item, you may find it nice to have mouse-triggered highlighting +instead or as well. The variable may have one of these values: + + nil No highlighting. + cursor Highlighting is cursor driven. + mouse Highlighting is mouse driven. + both Both cursor and mouse trigger highlighting. + +Changing this variable requires to rebuild the selection and *toc* buffers +to become effective (keys `g' or `r')." + :group 'reftex-fontification-configurations + :type '(choice + (const :tag "Never" nil) + (const :tag "Cursor driven" cursor) + (const :tag "Mouse driven" mouse) + (const :tag "Mouse and Cursor driven." both))) + +(defcustom reftex-cursor-selected-face 'highlight + "Face name to highlight cursor selected item in toc and selection buffers. +See also the variable `reftex-highlight-selection'." + :group 'reftex-fontification-configurations + :type 'symbol) +(defcustom reftex-mouse-selected-face 'secondary-selection + "Face name to highlight mouse selected item in toc and selection buffers. +See also the variable `reftex-highlight-selection'." + :group 'reftex-fontification-configurations + :type 'symbol) +(defcustom reftex-file-boundary-face 'font-lock-comment-face + "Face name for file boundaries in selection buffer." + :group 'reftex-fontification-configurations + :type 'symbol) +(defcustom reftex-label-face 'font-lock-constant-face + "Face name for labels in selection buffer." + :group 'reftex-fontification-configurations + :type 'symbol) +(defcustom reftex-section-heading-face 'font-lock-function-name-face + "Face name for section headings in toc and selection buffers." + :group 'reftex-fontification-configurations + :type 'symbol) +(defcustom reftex-toc-header-face 'font-lock-comment-face + "Face name for the header of a toc buffer." + :group 'reftex-fontification-configurations + :type 'symbol) +(defcustom reftex-bib-author-face 'font-lock-keyword-face + "Face name for author names in bib selection buffer." + :group 'reftex-fontification-configurations + :type 'symbol) +(defcustom reftex-bib-year-face 'font-lock-comment-face + "Face name for year in bib selection buffer." + :group 'reftex-fontification-configurations + :type 'symbol) +(defcustom reftex-bib-title-face 'font-lock-function-name-face + "Face name for article title in bib selection buffer." + :group 'reftex-fontification-configurations + :type 'symbol) +(defcustom reftex-bib-extra-face 'font-lock-comment-face + "Face name for bibliographic information in bib selection buffer." + :group 'reftex-fontification-configurations + :type 'symbol) +(defcustom reftex-select-mark-face 'bold + "Face name for marked entries in the selection buffers." + :group 'reftex-fontification-configurations + :type 'symbol) +(defcustom reftex-index-header-face 'font-lock-comment-face + "Face name for the header of an index buffer." + :group 'reftex-fontification-configurations + :type 'symbol) +(defcustom reftex-index-section-face 'font-lock-function-name-face + "Face name for the start of a new letter section in the index." + :group 'reftex-fontification-configurations + :type 'symbol) +(defcustom reftex-index-tag-face 'font-lock-keyword-face + "Face name for index names (for multiple indices)." + :group 'reftex-fontification-configurations + :type 'symbol) +(defcustom reftex-index-face 'font-lock-constant-face + "Face name for index entries." + :group 'reftex-fontification-configurations + :type 'symbol) + +(defcustom reftex-pre-refontification-functions nil + "X-Symbol specific hook. +Functions get two arguments, the buffer from where the command started and a +symbol indicating in what context the hook is called." + :group 'reftex-fontification-configurations + :type 'hook) + +;; Miscellaneous configurations ----------------------------------------- + +(defgroup reftex-miscellaneous-configurations nil + "Collection of further configurations." + :group 'reftex) + +(defcustom reftex-extra-bindings nil + "Non-nil means, make additional key bindings on startup. +These extra bindings are located in the users `C-c letter' map." + :group 'reftex-miscellaneous-configurations + :type 'boolean) + +(defcustom reftex-plug-into-AUCTeX nil + "*Plug-in flags for AUCTeX interface. +This variable is a list of 4 boolean flags. When a flag is non-nil, +RefTeX will + + - supply labels in new sections and environments (flag 1) + - supply arguments for macros like `\\label'. (flag 2) + - supply arguments for macros like `\\ref'. (flag 3) + - supply arguments for macros like `\\cite'. (flag 4) + - supply arguments for macros like `\\index'. (flag 5) + +You may also set the variable itself to t or nil in order to turn all +plug-ins on or off, respectively. +\\Supplying labels in new sections and environments applies when creating +sections with \\[LaTeX-section] and environments with \\[LaTeX-environment]. +Supplying macro arguments applies when you insert such a macro interactively +with \\[TeX-insert-macro]. +See the AUCTeX documentation for more information. +RefTeX uses `fset' to take over the function calls. Changing the variable +may require a restart of Emacs in order to become effective." + :group 'reftex-miscellaneous-configurations + :group 'LaTeX + :type '(choice + (const :tag "No plug-ins" nil) + (const :tag "All possible plug-ins" t) + (list + :tag "Individual choice" + :value (t t t t t) + (boolean :tag "supply label in new sections and environments") + (boolean :tag "supply argument for macros like `\\label' ") + (boolean :tag "supply argument for macros like `\\ref' ") + (boolean :tag "supply argument for macros like `\\cite' ") + (boolean :tag "supply argument for macros like `\\index' ") + ))) + +(defcustom reftex-allow-detached-macro-args nil + "*Non-nil means, allow arguments of macros to be detached by whitespace. +When this is t, `aaa' will be considered as argument of \\bb in the following +construct: \\bbb [xxx] {aaa}." + :group 'reftex-miscellaneous-configurations + :type 'boolean) + + +(defcustom reftex-load-hook nil + "Hook which is being run when loading reftex.el." + :group 'reftex-miscellaneous-configurations + :type 'hook) + +(defcustom reftex-mode-hook nil + "Hook which is being run when turning on RefTeX mode." + :group 'reftex-miscellaneous-configurations + :type 'hook) + +;;; reftex-vars.el ends here diff --git a/lisp/textmodes/reftex-vcr.el b/lisp/textmodes/reftex-vcr.el new file mode 100644 index 00000000000..d8b1637c3da --- /dev/null +++ b/lisp/textmodes/reftex-vcr.el @@ -0,0 +1,452 @@ +;;; reftex-vcr.el - Viewing cross references and citations with RefTeX +;;; Version: 4.5 +;;; +;;; See main file reftex.el for licensing information + +(provide 'reftex-vcr) +(require 'reftex) +;;; + +(defun reftex-view-crossref (&optional arg auto-how) + "View cross reference of macro at point. Point must be on the KEY +argument. When at at `\ref' macro, show corresponding `\label' +definition, also in external documents (`xr'). When on a label, show +a locations where KEY is referenced. Subsequent calls find additional +locations. When on a `\cite', show the associated `\bibitem' macro or +the BibTeX database entry. When on a `\bibitem', show a `\cite' macro +which uses this KEY. When on an `\index', show other locations marked +by the same index entry. +To define additional cross referencing items, use the option +`reftex-view-crossref-extra'. See also `reftex-view-crossref-from-bibtex'. +With one or two C-u prefixes, enforce rescanning of the document. +With argument 2, select the window showing the cross reference. +AUTO-HOW is only for the automatic crossref display and is handed through +to the functions `reftex-view-cr-cite' and `reftex-view-cr-ref'." + + (interactive "P") + ;; See where we are. + (let* ((macro (car (reftex-what-macro-safe 1))) + (key (reftex-this-word "^{}%\n\r,")) + dw) + + (if (or (null macro) (reftex-in-comment)) + (error "Not on a crossref macro argument")) + + (setq reftex-call-back-to-this-buffer (current-buffer)) + + (cond + ((string-match "\\`\\\\cite\\|cite\\*?\\'" macro) + ;; A citation macro: search for bibitems or BibTeX entries + (setq dw (reftex-view-cr-cite arg key auto-how))) + ((string-match "\\`\\\\ref\\|ref\\(range\\)?\\*?\\'" macro) + ;; A reference macro: search for labels + (setq dw (reftex-view-cr-ref arg key auto-how))) + (auto-how nil) ;; No further action for automatic display (speed) + ((or (equal macro "\\label") + (member macro reftex-macros-with-labels)) + ;; A label macro: search for reference macros + (reftex-access-scan-info arg) + (setq dw (reftex-view-regexp-match + (format reftex-find-reference-format (regexp-quote key)) + 4 nil nil))) + ((equal macro "\\bibitem") + ;; A bibitem macro: search for citations + (reftex-access-scan-info arg) + (setq dw (reftex-view-regexp-match + (format reftex-find-citation-regexp-format (regexp-quote key)) + 3 nil nil))) + ((member macro reftex-macros-with-index) + (reftex-access-scan-info arg) + (setq dw (reftex-view-regexp-match + (format reftex-find-index-entry-regexp-format + (regexp-quote key)) + 3 nil nil))) + (t + (reftex-access-scan-info arg) + (catch 'exit + (let ((list reftex-view-crossref-extra) + entry mre action group) + (while (setq entry (pop list)) + (setq mre (car entry) + action (nth 1 entry) + group (nth 2 entry)) + (when (string-match mre macro) + (setq dw (reftex-view-regexp-match + (format action key) group nil nil)) + (throw 'exit t)))) + (error "Not on a crossref macro argument")))) + (if (and (eq arg 2) (windowp dw)) (select-window dw)))) + +(defun reftex-view-cr-cite (arg key how) + ;; View crossreference of a ref cite. HOW can have the values + ;; nil: Show in another window. + ;; echo: Show one-line info in echo area. + ;; tmp-window: Show in small window and arrange for window to disappear. + + ;; Ensure access to scanning info + (reftex-access-scan-info (or arg current-prefix-arg)) + + (if (eq how 'tmp-window) + ;; Remember the window configuration + (put 'reftex-auto-view-crossref 'last-window-conf + (current-window-configuration))) + + (let (files size item (pos (point)) (win (selected-window)) pop-win) + ;; Find the citation mode and the file list + (cond + ((assq 'bib (symbol-value reftex-docstruct-symbol)) + (setq item nil + files (reftex-get-bibfile-list))) + ((assq 'thebib (symbol-value reftex-docstruct-symbol)) + (setq item t + files (list (cdr (assq 'thebib + (symbol-value reftex-docstruct-symbol)))))) + (reftex-default-bibliography + (setq item nil + files (reftex-default-bibliography))) + (how) ;; don't throw for special display + (t (error "Cannot display crossref"))) + + (if (eq how 'echo) + ;; Display in Echo area + (reftex-echo-cite key files item) + ;; Display in a window + (if (not (eq how 'tmp-window)) + ;; Normal display + (reftex-pop-to-bibtex-entry key files nil t item) + ;; A temporary window + (condition-case nil + (reftex-pop-to-bibtex-entry key files nil t item) + (error (goto-char pos) + (message "cite: no such citation key %s" key) + (error ""))) + ;; Resize the window + (setq size (max 1 (count-lines (point) + (reftex-end-of-bib-entry item)))) + (let ((window-min-height 2)) + (shrink-window (1- (- (window-height) size))) + (recenter 0)) + ;; Arrange restoration + (add-hook 'pre-command-hook 'reftex-restore-window-conf)) + + ;; Normal display in other window + (add-hook 'pre-command-hook 'reftex-highlight-shall-die) + (setq pop-win (selected-window)) + (select-window win) + (goto-char pos) + (when (equal arg 2) + (select-window pop-win))))) + +(defun reftex-view-cr-ref (arg label how) + ;; View crossreference of a ref macro. HOW can have the values + ;; nil: Show in another window. + ;; echo: Show one-line info in echo area. + ;; tmp-window: Show in small window and arrange for window to disappear. + + ;; Ensure access to scanning info + (reftex-access-scan-info (or arg current-prefix-arg)) + + (if (eq how 'tmp-window) + ;; Remember the window configuration + (put 'reftex-auto-view-crossref 'last-window-conf + (current-window-configuration))) + + (let* ((xr-data (assoc 'xr (symbol-value reftex-docstruct-symbol))) + (xr-re (nth 2 xr-data)) + (entry (assoc label (symbol-value reftex-docstruct-symbol))) + (win (selected-window)) pop-win (pos (point))) + + (if (and (not entry) (stringp label) xr-re (string-match xr-re label)) + ;; Label is defined in external document + (save-excursion + (save-match-data + (set-buffer + (or (reftex-get-file-buffer-force + (cdr (assoc (match-string 1 label) (nth 1 + xr-data)))) + (error "Problem with external label %s" label)))) + (setq label (substring label (match-end 1))) + (reftex-access-scan-info) + (setq entry + (assoc label (symbol-value reftex-docstruct-symbol))))) + (if (eq how 'echo) + ;; Display in echo area + (reftex-echo-ref label entry (symbol-value reftex-docstruct-symbol)) + (let ((window-conf (current-window-configuration))) + (condition-case nil + (reftex-show-label-location entry t nil t t) + (error (set-window-configuration window-conf) + (message "ref: Label %s not found" label) + (error "ref: Label %s not found" label)))) ;; 2nd is line OK + (add-hook 'pre-command-hook 'reftex-highlight-shall-die) + + (when (eq how 'tmp-window) + ;; Resize window and arrange restauration + (shrink-window (1- (- (window-height) 9))) + (recenter '(4)) + (add-hook 'pre-command-hook 'reftex-restore-window-conf)) + (setq pop-win (selected-window)) + (select-window win) + (goto-char pos) + (when (equal arg 2) + (select-window pop-win))))) + +(defun reftex-mouse-view-crossref (ev) + "View cross reference of \\ref or \\cite macro where you click. +If the macro at point is a \\ref, show the corresponding label definition. +If it is a \\cite, show the BibTeX database entry. +If there is no such macro at point, search forward to find one. +With argument, actually select the window showing the cross reference." + (interactive "e") + (mouse-set-point ev) + (reftex-view-crossref current-prefix-arg)) + +(defun reftex-view-crossref-when-idle () + ;; Display info about crossref at point in echo area or a window. + ;; This function was desigend to work with an idle timer. + ;; We try to get out of here as quickly as possible if the call is useless. + (and reftex-mode + ;; Make sure message area is free if we need it. + (or (eq reftex-auto-view-crossref 'window) (not (current-message))) + ;; Make sure we are not already displaying this one + (not (memq last-command '(reftex-view-crossref + reftex-mouse-view-crossref))) + ;; Quick precheck if this might be a relevant spot + ;; FIXME: Can fail with backslash in comment + (save-excursion + (search-backward "\\" nil t) + (looking-at "\\\\[a-zA-Z]*\\(cite\\|ref\\)")) + + (condition-case nil + (let ((current-prefix-arg nil)) + (cond + ((eq reftex-auto-view-crossref t) + (reftex-view-crossref -1 'echo)) + ((eq reftex-auto-view-crossref 'window) + (reftex-view-crossref -1 'tmp-window)) + (t nil))) + (error nil)))) + +(defun reftex-restore-window-conf () + (set-window-configuration (get 'reftex-auto-view-crossref 'last-window-conf)) + (put 'reftex-auto-view-crossref 'last-window-conf nil) + (remove-hook 'pre-command-hook 'reftex-restore-window-conf)) + +(defun reftex-echo-ref (label entry docstruct) + ;; Display crossref info in echo area. + (cond + ((null docstruct) + (message (substitute-command-keys (format reftex-no-info-message "ref")))) + ((null entry) + (message "ref: unknown label: %s" label)) + (t + (when (stringp (nth 2 entry)) + (message "ref(%s): %s" (nth 1 entry) (nth 2 entry))) + (let ((buf (get-buffer " *Echo Area*"))) + (when buf + (save-excursion + (set-buffer buf) + (run-hooks 'reftex-display-copied-context-hook))))))) + +(defun reftex-echo-cite (key files item) + ;; Display citation info in echo area. + (let* ((cache (assq 'bibview-cache (symbol-value reftex-docstruct-symbol))) + (cache-entry (assoc key (cdr cache))) + entry string buf (all-files files)) + + (if (and reftex-cache-cite-echo cache-entry) + ;; We can just use the cache + (setq string (cdr cache-entry)) + + ;; Need to look in the database + (unless reftex-revisit-to-echo + (setq files (reftex-visited-files files))) + + (setq entry + (condition-case nil + (save-excursion + (reftex-pop-to-bibtex-entry key files nil nil item t)) + (error + (if (and files (= (length all-files) (length files))) + (message "cite: no such database entry: %s" key) + (message (substitute-command-keys + (format reftex-no-info-message "cite")))) + nil))) + (when entry + (if item + (setq string (reftex-nicify-text entry)) + (setq string (reftex-make-cite-echo-string + (reftex-parse-bibtex-entry entry) + reftex-docstruct-symbol))))) + (unless (or (null string) (equal string "")) + (message "cite: %s" string)) + (when (setq buf (get-buffer " *Echo Area*")) + (save-excursion + (set-buffer buf) + (run-hooks 'reftex-display-copied-context-hook))))) + +(defvar reftex-use-itimer-in-xemacs nil + "*Non-nil means use the idle timers in XEmacs for crossref display. +Currently, idle timer restart is broken and we use the post-command-hook.") + +(defun reftex-toggle-auto-view-crossref () + "Toggle the automatic display of crossref information in the echo area. +When active, leaving point idle in the argument of a \\ref or \\cite macro +will display info in the echo area." + (interactive) + (if reftex-auto-view-crossref-timer + (progn + (if (featurep 'xemacs) + (if reftex-use-itimer-in-xemacs + (delete-itimer reftex-auto-view-crossref-timer) + (remove-hook 'post-command-hook 'reftex-start-itimer-once)) + (cancel-timer reftex-auto-view-crossref-timer)) + (setq reftex-auto-view-crossref-timer nil) + (message "Automatic display of crossref information was turned off")) + (setq reftex-auto-view-crossref-timer + (if (featurep 'xemacs) + (if reftex-use-itimer-in-xemacs + (start-itimer "RefTeX Idle Timer" + 'reftex-view-crossref-when-idle + reftex-idle-time reftex-idle-time t) + (add-hook 'post-command-hook 'reftex-start-itimer-once) + t) + (run-with-idle-timer + reftex-idle-time t 'reftex-view-crossref-when-idle))) + (unless reftex-auto-view-crossref + (setq reftex-auto-view-crossref t)) + (message "Automatic display of crossref information was turned on"))) + +(defun reftex-start-itimer-once () + (and reftex-mode + (not (itimer-live-p reftex-auto-view-crossref-timer)) + (setq reftex-auto-view-crossref-timer + (start-itimer "RefTeX Idle Timer" + 'reftex-view-crossref-when-idle + reftex-idle-time nil t)))) + +(defun reftex-view-crossref-from-bibtex (&optional arg) + "View location in a LaTeX document which cites the BibTeX entry at point. +Since BibTeX files can be used by many LaTeX documents, this function +prompts upon first use for a buffer in RefTeX mode. To reset this +link to a document, call the function with with a prefix arg. +Calling this function several times find successive citation locations." + (interactive "P") + (when arg + ;; Break connection to reference buffer + (remprop 'reftex-bibtex-view-cite-locations :ref-buffer)) + (let ((ref-buffer (get 'reftex-bibtex-view-cite-locations :ref-buffer))) + ;; Establish connection to reference buffer + (unless ref-buffer + (setq ref-buffer + (save-excursion + (completing-read + "Reference buffer: " + (delq nil + (mapcar + (lambda (b) + (set-buffer b) + (if reftex-mode (list (buffer-name b)) nil)) + (buffer-list))) + nil t))) + (put 'reftex-bibtex-view-cite-locations :ref-buffer ref-buffer)) + ;; Search for citations + (bibtex-beginning-of-entry) + (if (looking-at + "@[a-zA-Z]+[ \t\n\r]*[{(][ \t\n\r]*\\([^, \t\r\n}]+\\)") + (progn + (goto-char (match-beginning 1)) + (reftex-view-regexp-match + (format reftex-find-citation-regexp-format + (regexp-quote (match-string 1))) + 3 arg ref-buffer)) + (error "Cannot find citation key in BibTeX entry")))) + +(defun reftex-view-regexp-match (re &optional highlight-group new ref-buffer) + ;; Search for RE in current document or in the document of REF-BUFFER. + ;; Continue the search, if the same re was searched last. + ;; Highlight the group HIGHLIGHT-GROUP of the match. + ;; When NEW is non-nil, start a new search regardless. + ;; Match point is displayed in another window. + ;; Upon success, returns the window which displays the match. + + ;;; Decide if new search or continued search + (let* ((oldprop (get 'reftex-view-regexp-match :props)) + (newprop (list (current-buffer) re)) + (cont (and (not new) (equal oldprop newprop))) + (cnt (if cont (get 'reftex-view-regexp-match :cnt) 0)) + (current-window (selected-window)) + (window-conf (current-window-configuration)) + match pop-window) + (switch-to-buffer-other-window (or ref-buffer (current-buffer))) + ;; Search + (condition-case nil + (if cont + (setq match (reftex-global-search-continue)) + (reftex-access-scan-info) + (setq match (reftex-global-search re (reftex-all-document-files)))) + (error nil)) + ;; Evaluate the match. + (if match + (progn + (put 'reftex-view-regexp-match :props newprop) + (put 'reftex-view-regexp-match :cnt (incf cnt)) + (reftex-highlight 0 (match-beginning highlight-group) + (match-end highlight-group)) + (add-hook 'pre-command-hook 'reftex-highlight-shall-die) + (setq pop-window (selected-window))) + (remprop 'reftex-view-regexp-match :props) + (or cont (set-window-configuration window-conf))) + (select-window current-window) + (if match + (progn + (message "Match Nr. %s" cnt) + pop-window) + (if cont + (error "No further matches (total number of matches: %d)" cnt) + (error "No matches"))))) + +(defvar reftex-global-search-marker (make-marker)) +(defun reftex-global-search (regexp file-list) + ;; Start a search for REGEXP in all files of FILE-LIST + (put 'reftex-global-search :file-list file-list) + (put 'reftex-global-search :regexp regexp) + (move-marker reftex-global-search-marker nil) + (reftex-global-search-continue)) + +(defun reftex-global-search-continue () + ;; Continue a global search started with `reftex-global-search' + (unless (get 'reftex-global-search :file-list) + (error "No global search to continue")) + (let* ((file-list (get 'reftex-global-search :file-list)) + (regexp (get 'reftex-global-search :regexp)) + (buf (or (marker-buffer reftex-global-search-marker) + (reftex-get-file-buffer-force (car file-list)))) + (pos (or (marker-position reftex-global-search-marker) 1)) + file) + ;; Take up starting position + (unless buf (error "No such buffer %s" buf)) + (switch-to-buffer buf) + (widen) + (goto-char pos) + ;; Search and switch file if necessary + (if (catch 'exit + (while t + (when (re-search-forward regexp nil t) + (move-marker reftex-global-search-marker (point)) + (throw 'exit t)) + ;; No match - goto next file + (pop file-list) + (or file-list (throw 'exit nil)) + (setq file (car file-list) + buf (reftex-get-file-buffer-force file)) + (unless buf (error "Cannot access file %s" file)) + (put 'reftex-global-search :file-list file-list) + (switch-to-buffer buf) + (widen) + (goto-char 1))) + t + (move-marker reftex-global-search-marker nil) + (error "All files processed")))) + +;;; reftex-vcr.el ends here -- 2.39.5