diff options
author | Jose A. Ortega Ruiz <jao@gnu.org> | 2022-09-30 15:08:40 +0200 |
---|---|---|
committer | Lars Ingebrigtsen <larsi@gnus.org> | 2022-09-30 15:08:40 +0200 |
commit | b23e062d7463b76d25dfd9ba4a80c1848a448e42 (patch) | |
tree | 468460b2d0b64f07d5d835356e603792d6ac1b25 /lisp/doc-view.el | |
parent | 0332142e8e78b49b4f98438be21d2868e738986b (diff) | |
download | emacs-b23e062d7463b76d25dfd9ba4a80c1848a448e42.tar.gz emacs-b23e062d7463b76d25dfd9ba4a80c1848a448e42.tar.bz2 emacs-b23e062d7463b76d25dfd9ba4a80c1848a448e42.zip |
docview: fixes for imenu generation
* lisp/doc-view.el: (doc-view--pdf-outline):
(doc-view-imenu-index):
(doc-view-imenu-setup): Fix multiple empty index generation for
documents without an outline, caching the result (see discussion in
bug#58103).
(doc-view--imenu-subtree): Fix for nested imenus (bug introduced in
commit fe002cc8ce) (bug#58180).
Diffstat (limited to 'lisp/doc-view.el')
-rw-r--r-- | lisp/doc-view.el | 43 |
1 files changed, 25 insertions, 18 deletions
diff --git a/lisp/doc-view.el b/lisp/doc-view.el index 80c4fd21deb..b1ea90c212b 100644 --- a/lisp/doc-view.el +++ b/lisp/doc-view.el @@ -1900,6 +1900,9 @@ If BACKWARD is non-nil, jump to the previous match." (defconst doc-view--outline-rx "[^\t]+\\(\t+\\)\"\\(.+\\)\"\t#\\(?:page=\\)?\\([0-9]+\\)") +(defvar-local doc-view--outline nil + "Cached PDF outline, so that it is only computed once per document.") + (defun doc-view--pdf-outline (&optional file-name) "Return a list describing the outline of FILE-NAME. Return a list describing the current file if FILE-NAME is nil. @@ -1907,19 +1910,20 @@ Return a list describing the current file if FILE-NAME is nil. Each element in the returned list contains information about a section's title, nesting level and page number. The list is flat: its tree structure is extracted by `doc-view--imenu-subtree'." - (let* ((outline nil) - (fn (or file-name (buffer-file-name))) - (fn (shell-quote-argument (expand-file-name fn)))) - (with-temp-buffer - (insert (shell-command-to-string (format "mutool show %s outline" fn))) - (goto-char (point-min)) - (while (re-search-forward doc-view--outline-rx nil t) - (push `((level . ,(length (match-string 1))) - (title . ,(replace-regexp-in-string "\\\\[rt]" " " - (match-string 2))) - (page . ,(string-to-number (match-string 3)))) - outline))) - (nreverse outline))) + (let ((fn (or file-name (buffer-file-name)))) + (when fn + (let ((outline nil) + (fn (shell-quote-argument (expand-file-name fn)))) + (with-temp-buffer + (insert (shell-command-to-string (format "mutool show %s outline" fn))) + (goto-char (point-min)) + (while (re-search-forward doc-view--outline-rx nil t) + (push `((level . ,(length (match-string 1))) + (title . ,(replace-regexp-in-string "\\\\[rt]" " " + (match-string 2))) + (page . ,(string-to-number (match-string 3)))) + outline))) + (nreverse outline))))) (defun doc-view--imenu-subtree (outline act) "Construct a tree of imenu items for the given outline list and action. @@ -1932,7 +1936,8 @@ entries at an upper level." (nested (not doc-view-imenu-flatten)) (index nil)) (while (and (car outline) - (or nested (<= level (alist-get 'level (car outline))))) + (or (not nested) + (<= level (alist-get 'level (car outline))))) (let-alist (car outline) (let ((title (format-spec doc-view-imenu-title-format `((?t . ,.title) (?p . ,.page))))) @@ -1953,16 +1958,18 @@ For extensibility, callers can specify a FILE-NAME to indicate the buffer other than the current buffer, and a jumping function GOTO-PAGE-FN other than `doc-view-goto-page'." (let* ((goto (or goto-page-fn 'doc-view-goto-page)) - (act (lambda (_name _pos page) (funcall goto page)))) - (car (doc-view--imenu-subtree (doc-view--pdf-outline file-name) act)))) + (act (lambda (_name _pos page) (funcall goto page))) + (outline (or doc-view--outline (doc-view--pdf-outline file-name)))) + (car (doc-view--imenu-subtree outline act)))) (defun doc-view-imenu-setup () "Set up local state in the current buffer for imenu, if needed." (when (and doc-view-imenu-enabled (executable-find "mutool")) (setq-local imenu-create-index-function #'doc-view-imenu-index imenu-submenus-on-top nil - imenu-sort-function nil) - (imenu-add-to-menubar "Outline"))) + imenu-sort-function nil + doc-view--outline (doc-view--pdf-outline)) + (when doc-view--outline (imenu-add-to-menubar "Outline")))) ;;;; User interface commands and the mode |