emacs-orgmode@gnu.org archives
 help / color / mirror / code / Atom feed
From: "Rudolf Adamkovič" <rudolf@adamkovic.org>
To: emacs-orgmode@gnu.org
Cc: yantar92@posteo.net
Subject: [PATCH v3] Improvements to Texinfo exporter (was: [PATCH] Re: Texinfo: Custom navigation and links in headings)
Date: Sat, 24 Aug 2024 21:54:51 +0200	[thread overview]
Message-ID: <m2cylxlmr8.fsf@adamkovic.org> (raw)
In-Reply-To: <m2y14m46ru.fsf@adamkovic.org>

[-- Attachment #1: Type: text/plain, Size: 580 bytes --]

Rudolf Adamkovič <rudolf@adamkovic.org> writes:

> I will fix the bug today.

And fixed.

CHANGE 1/3

  I split `ALT_NAVIGATION' to
  
    `ALT_NEXT'
    `ALT_PREVIOUS'
    `ALT_UP'
  
  to support commas in node names.
  
  (I am open to renaming to `TEXINFO_UP', etc.)

CHANGE 2/3

  I added a 3rd patch that makes
  
    numeric values of `toc' in `#+OPTIONS'
  
  work out of the box, like in e.g. HTML exports.

CHANGE 3/3

  Further, I

    improved commit messages, documentation,

  and the like.

Ready for review ... again!

Rudy


[-- Warning: decoded text below may be mangled, UTF-8 assumed --]
[-- Attachment #2: 0001-ox-texinfo-Support-alternative-navigation.patch --]
[-- Type: text/x-patch, Size: 19270 bytes --]

From 7a8692ec67360414143524d49a098fd99fce03a3 Mon Sep 17 00:00:00 2001
From: Rudolf Adamkovic <rudolf@adamkovic.org>
Date: Sat, 24 Aug 2024 21:35:44 +0200
Subject: [PATCH 1/3] ox-texinfo: Support alternative navigation

* doc/org-manual.org (Headings and sectioning structure): Describe the
new feature.
* etc/ORG-NEWS (Texinfo exports can use alternative navigation):
Announce the new feature.
* lisp/ox-texinfo.el (org-texinfo-headline, org-texinfo-template): Use
the `ALT_NEXT', `ALT_PREVIOUS', and `ALT_UP' Org properties when
generating Texinfo @node lines for Next, Previous, and Up pointers.
* testing/lisp/test-ox-texinfo.el (test-ox-texinfo/alt-title,
test-ox-texinfo/alt-navigation/all-directions,
test-ox-texinfo/alt-navigation/one-direction,
test-ox-texinfo/alt-navigation/no-directions,
test-ox-texinfo/alt-navigation/with-alt-title,
test-ox-texinfo/alt-navigation/top/default,
test-ox-texinfo/alt-navigation/top/default): Test the new feature to
avoid regressions in the future.
---
 doc/org-manual.org              |  20 +++
 etc/ORG-NEWS                    |   8 +
 lisp/ox-texinfo.el              | 103 +++++++-----
 testing/lisp/test-ox-texinfo.el | 290 ++++++++++++++++++++++++++++++++
 4 files changed, 382 insertions(+), 39 deletions(-)

diff --git a/doc/org-manual.org b/doc/org-manual.org
index 9365c66b1..049454624 100644
--- a/doc/org-manual.org
+++ b/doc/org-manual.org
@@ -15999,6 +15999,26 @@ the node in which a reader enters an Info manual.  As such, it is
 expected not to appear in printed output generated from the =.texi=
 file.  See [[info:texinfo::The Top Node]], for more information.
 
+#+cindex: @samp{ALT_NEXT}, property
+#+cindex: @samp{ALT_PREVIOUS}, property
+#+cindex: @samp{ALT_UP}, property
+Texinfo automatically sets the /Next/, /Previous/, and /Up/ pointers,
+reflecting the hierarchy of your document.  If you want to use a
+different hierarchy, or no hierarchy at all, set the =ALT_NEXT=,
+=ALT_PREVIOUS=, and =ALT_UP= properties to the relevant titles.  For
+example:
+
+#+begin_example
+,* Mathematical Logic
+  :PROPERTIES:
+  :ALT_TITLE: Logic
+  :END:
+,* Proposition
+  :PROPERTIES:
+  :ALT_UP: Logic
+  :END:
+#+end_example
+
 *** Indices
 :PROPERTIES:
 :DESCRIPTION: Creating indices.
diff --git a/etc/ORG-NEWS b/etc/ORG-NEWS
index 392788055..49dafcd5d 100644
--- a/etc/ORG-NEWS
+++ b/etc/ORG-NEWS
@@ -77,6 +77,14 @@ You can now create links to =shortdoc= documentation groups for Emacs
 Lisp functions (see =M-x shortdoc-display-group=).  Requires Emacs 28
 or newer.
 
+*** Texinfo exports can use alternative navigation
+
+You can now alter the Texinfo navigation hierarchy by specifying the
+/Next/, /Previous/, and /Up/ pointers in the =ALT_NEXT=,
+=ALT_PREVIOUS=, and =ALT_UP= properties, respectively.  For more
+information, see "13.14.6 Headings and sectioning structure" section
+of the Org manual.
+
 ** New and changed options
 
 # Chanes deadling with changing default values of customizations,
diff --git a/lisp/ox-texinfo.el b/lisp/ox-texinfo.el
index 6adee9fca..808dc22f6 100644
--- a/lisp/ox-texinfo.el
+++ b/lisp/ox-texinfo.el
@@ -512,46 +512,63 @@ INFO is a plist used as a communication channel.  See
     ;; Else use format string.
     (fmt (format fmt text))))
 
+(defun org-texinfo--node-line (headline info)
+  "Return node line for HEADLINE.
+
+INFO is a plist used as a communication channel."
+  (let ((next (org-element-property :ALT_NEXT headline))
+        (previous (org-element-property :ALT_PREVIOUS headline))
+        (up (org-element-property :ALT_UP headline)))
+    (string-trim-right
+     (concat (format "@node %s" (org-texinfo--get-node headline info))
+             (if (or next previous up) ", ")
+             (and next (org-texinfo--sanitize-node next))
+             (if (or previous up) ", ")
+             (and previous (org-texinfo--sanitize-node previous))
+             (if (or up) ", ")
+             (and up (org-texinfo--sanitize-node up))))))
+
 (defun org-texinfo--get-node (datum info)
   "Return node or anchor associated to DATUM.
-DATUM is a headline, a radio-target or a target.  INFO is a plist
-used as a communication channel.  The function guarantees the
+DATUM is org-data (root), a headline, a radio-target or a target.  INFO
+is a plist used as a communication channel.  The function guarantees the
 node or anchor name is unique."
-  (let ((cache (plist-get info :texinfo-node-cache)))
-    (or (cdr (assq datum cache))
-	(let* ((salt 0)
-	       (basename
-		(org-texinfo--sanitize-node
-		 (pcase (org-element-type datum)
-		   (`headline
-		    (org-texinfo--sanitize-title
-		     (org-export-get-alt-title datum info) info))
-		   (`radio-target
-		    (org-export-data (org-element-contents datum) info))
-		   (`target
-		    (org-element-property :value datum))
-		   (_
-		    (or (org-element-property :name datum)
-			(org-export-get-reference datum info))))))
-	       (name basename))
-	  ;; Org exports deeper elements before their parents.  If two
-	  ;; node names collide -- e.g., they have the same title --
-	  ;; within the same hierarchy, the second one would get the
-	  ;; smaller node name.  This is counter-intuitive.
-	  ;; Consequently, we ensure that every parent headline gets
-	  ;; its node beforehand.  As a recursive operation, this
-	  ;; achieves the desired effect.
-	  (let ((parent (org-element-lineage datum 'headline)))
-	    (when (and parent (not (assq parent cache)))
-	      (org-texinfo--get-node parent info)
-	      (setq cache (plist-get info :texinfo-node-cache))))
-	  ;; Ensure NAME is unique and not reserved node name "Top",
-          ;; no matter what case is used.
-	  (while (or (string-equal "Top" (capitalize name))
-                     (rassoc name cache))
-	    (setq name (concat basename (format " (%d)" (cl-incf salt)))))
-	  (plist-put info :texinfo-node-cache (cons (cons datum name) cache))
-	  name))))
+  (if (eq (org-element-type datum) 'org-data) "Top"
+    (let ((cache (plist-get info :texinfo-node-cache)))
+      (or (cdr (assq datum cache))
+	  (let* ((salt 0)
+	         (basename
+		  (org-texinfo--sanitize-node
+		   (pcase (org-element-type datum)
+		     (`headline
+		      (org-texinfo--sanitize-title
+		       (org-export-get-alt-title datum info) info))
+		     (`radio-target
+		      (org-export-data (org-element-contents datum) info))
+		     (`target
+		      (org-element-property :value datum))
+		     (_
+		      (or (org-element-property :name datum)
+			  (org-export-get-reference datum info))))))
+	         (name basename))
+	    ;; Org exports deeper elements before their parents.  If
+	    ;; two node names collide -- e.g., they have the same
+	    ;; title -- within the same hierarchy, the second one
+	    ;; would get the smaller node name.  This is
+	    ;; counter-intuitive.  Consequently, we ensure that every
+	    ;; parent headline gets its node beforehand.  As a
+	    ;; recursive operation, this achieves the desired effect.
+	    (let ((parent (org-element-lineage datum 'headline)))
+	      (when (and parent (not (assq parent cache)))
+	        (org-texinfo--get-node parent info)
+	        (setq cache (plist-get info :texinfo-node-cache))))
+	    ;; Ensure NAME is unique and not reserved node name "Top",
+            ;; no matter what case is used.
+	    (while (or (string-equal "Top" (capitalize name))
+                       (rassoc name cache))
+	      (setq name (concat basename (format " (%d)" (cl-incf salt)))))
+	    (plist-put info :texinfo-node-cache (cons (cons datum name) cache))
+	    name)))))
 
 (defun org-texinfo--sanitize-node (title)
   "Bend string TITLE to node line requirements.
@@ -875,7 +892,12 @@ holding export options."
      ;; Configure Top Node when not for TeX.  Also include contents
      ;; from the first section of the document.
      "@ifnottex\n"
-     "@node Top\n"
+     ;; Top node.
+     (org-element-map (plist-get info :parse-tree) 'org-data
+       (lambda (root)
+         (org-texinfo--node-line root info))
+       info t)
+     "\n"
      (format "@top %s\n" title)
      (let* ((first-section
 	     (org-element-map (plist-get info :parse-tree) 'section
@@ -1118,7 +1140,10 @@ holding contextual information."
 	(concat
 	 ;; Even if HEADLINE is using @subheading and al., leave an
 	 ;; anchor so cross-references in the Org document still work.
-	 (format (if notoc? "@anchor{%s}\n" "@node %s\n") node)
+         (if notoc?
+             (format "@anchor{%s}" node)
+           (org-texinfo--node-line headline info))
+         "\n"
 	 (format command full-text)
 	 contents))))))
 
diff --git a/testing/lisp/test-ox-texinfo.el b/testing/lisp/test-ox-texinfo.el
index b16a344e7..fb0269649 100644
--- a/testing/lisp/test-ox-texinfo.el
+++ b/testing/lisp/test-ox-texinfo.el
@@ -345,5 +345,295 @@ body
        (should-not (org-element-contents section))
        (should (eq first-heading (org-element-parent section)))))))
 
+\f
+;;; Alternative title and navigation
+
+(ert-deftest test-ox-texinfo/alt-title ()
+  "Test alternative titles."
+  (should
+   (org-test-with-temp-text
+       (string-join
+        (list "* Title 1"
+              ":PROPERTIES:"
+              ":ALT_TITLE: Title 2"
+              ":END:")
+        "\n")
+     (let ((export-buffer "*Test Texinfo Export*")
+           (org-export-show-temporary-export-buffer nil))
+       (org-export-to-buffer 'texinfo export-buffer
+         nil nil nil nil nil
+         #'texinfo-mode)
+       (with-current-buffer export-buffer
+         (goto-char (point-min))
+         (re-search-forward "^@node Title 2$"))))))
+
+(ert-deftest test-ox-texinfo/alt-title/sanitized ()
+  "Test sanitized alternative titles."
+  (should
+   (org-test-with-temp-text
+       (string-join
+        (list "* Title 1"
+              ":PROPERTIES:"
+              ":ALT_TITLE: (Foo:)  Bar, baz."
+              ":END:")
+        "\n")
+     (let ((export-buffer "*Test Texinfo Export*")
+           (org-export-show-temporary-export-buffer nil))
+       (org-export-to-buffer 'texinfo export-buffer
+         nil nil nil nil nil
+         #'texinfo-mode)
+       (with-current-buffer export-buffer
+         (goto-char (point-min))
+         (re-search-forward "^@node \\[Foo) Bar baz$"))))))
+
+(ert-deftest test-ox-texinfo/alt-navigation/next ()
+  "Test alternative navigation to Next nodes."
+  (should
+   (org-test-with-temp-text
+       (string-join
+        (list "* Title"
+              ":PROPERTIES:"
+              ":ALT_NEXT: Next"
+              ":END:")
+        "\n")
+     (let ((export-buffer "*Test Texinfo Export*")
+           (org-export-show-temporary-export-buffer nil))
+       (org-export-to-buffer 'texinfo export-buffer
+         nil nil nil nil nil
+         #'texinfo-mode)
+       (with-current-buffer export-buffer
+         (goto-char (point-min))
+         (re-search-forward "^@node Title, Next$"))))))
+
+(ert-deftest test-ox-texinfo/alt-navigation/next/sanitized ()
+  "Test sanitized alternative navigation to Next nodes."
+  (should
+   (org-test-with-temp-text
+       (string-join
+        (list "* Title 1"
+              ":PROPERTIES:"
+              ":ALT_NEXT: (Foo:)  Bar, baz."
+              ":END:")
+        "\n")
+     (let ((export-buffer "*Test Texinfo Export*")
+           (org-export-show-temporary-export-buffer nil))
+       (org-export-to-buffer 'texinfo export-buffer
+         nil nil nil nil nil
+         #'texinfo-mode)
+       (with-current-buffer export-buffer
+         (goto-char (point-min))
+         (re-search-forward "^@node Title 1, \\[Foo) Bar baz$"))))))
+
+(ert-deftest test-ox-texinfo/alt-navigation/next/with-previous ()
+  "Test alternative navigation to Next and Previous nodes."
+  (should
+   (org-test-with-temp-text
+       (string-join
+        (list "* Title"
+              ":PROPERTIES:"
+              ":ALT_NEXT: Next"
+              ":ALT_PREVIOUS: Previous"
+              ":END:")
+        "\n")
+     (let ((export-buffer "*Test Texinfo Export*")
+           (org-export-show-temporary-export-buffer nil))
+       (org-export-to-buffer 'texinfo export-buffer
+         nil nil nil nil nil
+         #'texinfo-mode)
+       (with-current-buffer export-buffer
+         (goto-char (point-min))
+         (re-search-forward "^@node Title, Next, Previous$"))))))
+
+(ert-deftest test-ox-texinfo/alt-navigation/next/with-previous-and-up ()
+  "Test alternative navigation to Next, Previous, and Up nodes."
+  (should
+   (org-test-with-temp-text
+       (string-join
+        (list "* Title"
+              ":PROPERTIES:"
+              ":ALT_NEXT: Next"
+              ":ALT_PREVIOUS: Previous"
+              ":ALT_UP: Up"
+              ":END:")
+        "\n")
+     (let ((export-buffer "*Test Texinfo Export*")
+           (org-export-show-temporary-export-buffer nil))
+       (org-export-to-buffer 'texinfo export-buffer
+         nil nil nil nil nil
+         #'texinfo-mode)
+       (with-current-buffer export-buffer
+         (goto-char (point-min))
+         (re-search-forward "^@node Title, Next, Previous, Up$"))))))
+
+(ert-deftest test-ox-texinfo/alt-navigation/previous ()
+  "Test alternative navigation to Previous nodes."
+  (should
+   (org-test-with-temp-text
+       (string-join
+        (list "* Title"
+              ":PROPERTIES:"
+              ":ALT_PREVIOUS: Previous"
+              ":END:")
+        "\n")
+     (let ((export-buffer "*Test Texinfo Export*")
+           (org-export-show-temporary-export-buffer nil))
+       (org-export-to-buffer 'texinfo export-buffer
+         nil nil nil nil nil
+         #'texinfo-mode)
+       (with-current-buffer export-buffer
+         (goto-char (point-min))
+         (re-search-forward "^@node Title, , Previous$"))))))
+
+(ert-deftest test-ox-texinfo/alt-navigation/previous/sanitized ()
+  "Test sanitized alternative navigation to Previous nodes."
+  (should
+   (org-test-with-temp-text
+       (string-join
+        (list "* Title 1"
+              ":PROPERTIES:"
+              ":ALT_PREVIOUS: (Foo:)  Bar, baz."
+              ":END:")
+        "\n")
+     (let ((export-buffer "*Test Texinfo Export*")
+           (org-export-show-temporary-export-buffer nil))
+       (org-export-to-buffer 'texinfo export-buffer
+         nil nil nil nil nil
+         #'texinfo-mode)
+       (with-current-buffer export-buffer
+         (goto-char (point-min))
+         (re-search-forward "^@node Title 1, , \\[Foo) Bar baz$"))))))
+
+(ert-deftest test-ox-texinfo/alt-navigation/previous/with-up ()
+  "Test alternative navigation to Previous and Up nodes."
+  (should
+   (org-test-with-temp-text
+       (string-join
+        (list "* Title"
+              ":PROPERTIES:"
+              ":ALT_PREVIOUS: Previous"
+              ":ALT_UP: Up"
+              ":END:")
+        "\n")
+     (let ((export-buffer "*Test Texinfo Export*")
+           (org-export-show-temporary-export-buffer nil))
+       (org-export-to-buffer 'texinfo export-buffer
+         nil nil nil nil nil
+         #'texinfo-mode)
+       (with-current-buffer export-buffer
+         (goto-char (point-min))
+         (re-search-forward "^@node Title, , Previous, Up$"))))))
+
+(ert-deftest test-ox-texinfo/alt-navigation/up ()
+  "Test alternative navigation to Up nodes."
+  (should
+   (org-test-with-temp-text
+       (string-join
+        (list "* Title"
+              ":PROPERTIES:"
+              ":ALT_UP: Up"
+              ":END:")
+        "\n")
+     (let ((export-buffer "*Test Texinfo Export*")
+           (org-export-show-temporary-export-buffer nil))
+       (org-export-to-buffer 'texinfo export-buffer
+         nil nil nil nil nil
+         #'texinfo-mode)
+       (with-current-buffer export-buffer
+         (goto-char (point-min))
+         (re-search-forward "^@node Title, , , Up$"))))))
+
+(ert-deftest test-ox-texinfo/alt-navigation/up/sanitized ()
+  "Test sanitized alternative navigation to Up nodes."
+  (should
+   (org-test-with-temp-text
+       (string-join
+        (list "* Title 1"
+              ":PROPERTIES:"
+              ":ALT_UP: (Foo:)  Bar, baz."
+              ":END:")
+        "\n")
+     (let ((export-buffer "*Test Texinfo Export*")
+           (org-export-show-temporary-export-buffer nil))
+       (org-export-to-buffer 'texinfo export-buffer
+         nil nil nil nil nil
+         #'texinfo-mode)
+       (with-current-buffer export-buffer
+         (goto-char (point-min))
+         (re-search-forward "^@node Title 1, , , \\[Foo) Bar baz$"))))))
+
+(ert-deftest test-ox-texinfo/alt-navigation/no-pointers ()
+  "Test alternative navigation with no pointers."
+  (should
+   (org-test-with-temp-text
+       (string-join
+        (list "* Title"
+              ":PROPERTIES:"
+              ":ALT_NEXT:"
+              ":END:")
+        "\n")
+     (let ((export-buffer "*Test Texinfo Export*")
+           (org-export-show-temporary-export-buffer nil))
+       (org-export-to-buffer 'texinfo export-buffer
+         nil nil nil nil nil
+         #'texinfo-mode)
+       (with-current-buffer export-buffer
+         (goto-char (point-min))
+         (re-search-forward "^@node Title,$"))))))
+
+(ert-deftest test-ox-texinfo/alt-navigation/top-node/not-set ()
+  "Test alternative navigation for Top nodes, when not set."
+  (should
+   (org-test-with-temp-text "Hello world!"
+     (let ((export-buffer "*Test Texinfo Export*")
+           (org-export-show-temporary-export-buffer nil))
+       (org-export-to-buffer 'texinfo export-buffer
+         nil nil nil nil nil
+         #'texinfo-mode)
+       (with-current-buffer export-buffer
+         (goto-char (point-min))
+         (re-search-forward "^@node Top$"))))))
+
+(ert-deftest test-ox-texinfo/alt-navigation/top-node/set ()
+  "Test alternative navigation for Top nodes, when set."
+  (should
+   (org-test-with-temp-text
+       (string-join
+        (list ":PROPERTIES:"
+              ":ALT_NEXT: Next"
+              ":ALT_PREVIOUS: Previous"
+              ":ALT_UP: Up"
+              ":END:"
+              "#+TITLE: Title"
+              "* Headline")
+        "\n")
+     (let ((export-buffer "*Test Texinfo Export*")
+           (org-export-show-temporary-export-buffer nil))
+       (org-export-to-buffer 'texinfo export-buffer
+         nil nil nil nil nil
+         #'texinfo-mode)
+       (with-current-buffer export-buffer
+         (goto-char (point-min))
+         (re-search-forward "^@node Top, Next, Previous, Up$"))))))
+
+(ert-deftest test-ox-texinfo/alt-navigation/with-alt-title ()
+  "Test alternative navigation combined with alternative titles."
+  (should
+   (org-test-with-temp-text
+       (string-join
+        (list "* Title 1"
+              ":PROPERTIES:"
+              ":ALT_TITLE: Title 2"
+              ":ALT_UP: Title 3"
+              ":END:")
+        "\n")
+     (let ((export-buffer "*Test Texinfo Export*")
+           (org-export-show-temporary-export-buffer nil))
+       (org-export-to-buffer 'texinfo export-buffer
+         nil nil nil nil nil
+         #'texinfo-mode)
+       (with-current-buffer export-buffer
+         (goto-char (point-min))
+         (re-search-forward "^@node Title 2, , , Title 3$"))))))
+
 (provide 'test-ox-texinfo)
 ;;; test-ox-texinfo.el end here
-- 
2.39.3 (Apple Git-146)


[-- Warning: decoded text below may be mangled, UTF-8 assumed --]
[-- Attachment #3: 0002-ox-texinfo-Support-links-in-headings.patch --]
[-- Type: text/x-patch, Size: 4859 bytes --]

From 3b9c3753df9580805ec763c0deaacaac0cfc38e1 Mon Sep 17 00:00:00 2001
From: Rudolf Adamkovic <rudolf@adamkovic.org>
Date: Sat, 24 Aug 2024 21:37:29 +0200
Subject: [PATCH 2/3] ox-texinfo: Support links in headings

* etc/ORG-NEWS (Texinfo exporter now supports links in headings):
Announce the new feature.
* lisp/ox-texinfo.el: (org-texinfo--format-entries,
org-texinfo--get-node, org-texinfo--sanitize-title-reference,
org-texinfo--sanitize-title): A 2-step change: (1) Rename
`--sanitize-title' to `--sanitize-title-reference' and (2) create a
new `--sanitize-title' sanitation function.  The new function is less
strict in that does not remove links, which should be allowed in
sectioning commands, such as `@unnumbered'.  The old function remains
more strict, which is useful for generating `@node' names, for
example.
* testing/lisp/test-ox-texinfo.el:
(test-ox-texinfo/headings-with-links): Test the new functionality to
avoid regressions in the future.
---
 etc/ORG-NEWS                    |  6 ++++++
 lisp/ox-texinfo.el              | 17 +++++++++++++++--
 testing/lisp/test-ox-texinfo.el | 28 ++++++++++++++++++++++++++++
 3 files changed, 49 insertions(+), 2 deletions(-)

diff --git a/etc/ORG-NEWS b/etc/ORG-NEWS
index 49dafcd5d..0ae1e4c81 100644
--- a/etc/ORG-NEWS
+++ b/etc/ORG-NEWS
@@ -231,6 +231,12 @@ This way, attachments will remain accessible when opening symlinked Org file.
 When no attach dir exists, Org mode will still prefer creating it in
 the "default" directory - where the symlink is located.
 
+*** Texinfo exporter now supports links in headings
+
+The Texinfo exporter no longer removes links from headings.  This
+applies to all headings, below and above the =H= and =toc= export
+=#+OPTIONS:=.
+
 * Version 9.7
 
 ** Important announcements and breaking changes
diff --git a/lisp/ox-texinfo.el b/lisp/ox-texinfo.el
index 808dc22f6..2e316d204 100644
--- a/lisp/ox-texinfo.el
+++ b/lisp/ox-texinfo.el
@@ -541,7 +541,7 @@ node or anchor name is unique."
 		  (org-texinfo--sanitize-node
 		   (pcase (org-element-type datum)
 		     (`headline
-		      (org-texinfo--sanitize-title
+		      (org-texinfo--sanitize-title-reference
 		       (org-export-get-alt-title datum info) info))
 		     (`radio-target
 		      (org-export-data (org-element-contents datum) info))
@@ -586,6 +586,19 @@ periods, commas and colons."
 (defun org-texinfo--sanitize-title (title info)
   "Make TITLE suitable as a section name.
 TITLE is a string or a secondary string.  INFO is the current
+export state, as a plist."
+  (org-export-data-with-backend
+   title
+   (org-export-create-backend
+    :parent 'texinfo
+    :transcoders `((footnote-reference . ,#'ignore)
+                   (radio-target . ,(lambda (_r c _) c))
+                   (target . ,#'ignore)))
+   info))
+
+(defun org-texinfo--sanitize-title-reference (title info)
+  "Make TITLE suitable as a section reference.
+TITLE is a string or a secondary string.  INFO is the current
 export state, as a plist."
   (org-export-data-with-backend
    title (org-export-toc-entry-backend 'texinfo) info))
@@ -1493,7 +1506,7 @@ a plist containing contextual information."
 	      ;; name.  Remove them.
 	      (replace-regexp-in-string
 	       "[ \t]*:+" ""
-	       (org-texinfo--sanitize-title
+	       (org-texinfo--sanitize-title-reference
 		(org-export-get-alt-title h info) info)))
 	     (node (org-texinfo--get-node h info))
 	     (entry (concat "* " title ":"
diff --git a/testing/lisp/test-ox-texinfo.el b/testing/lisp/test-ox-texinfo.el
index fb0269649..4a521b3ed 100644
--- a/testing/lisp/test-ox-texinfo.el
+++ b/testing/lisp/test-ox-texinfo.el
@@ -635,5 +635,33 @@ body
          (goto-char (point-min))
          (re-search-forward "^@node Title 2, , , Title 3$"))))))
 
+\f
+;;; Headings with links
+
+(ert-deftest test-ox-texinfo/headings-with-links ()
+  "Test node and chapter names."
+  (should
+   (org-test-with-temp-text
+       (string-join
+        (list "* Heading 1"
+              "  ...."
+              "* Heading 2 ([[* Heading 1][Heading 1]])"
+              "  ....")
+        "\n")
+     (let ((export-buffer "*Test Texinfo Export*")
+           (org-export-show-temporary-export-buffer nil))
+       (org-export-to-buffer 'texinfo export-buffer
+         nil nil nil nil nil
+         #'texinfo-mode)
+       (with-current-buffer export-buffer
+         (goto-char (point-min))
+         (and
+          (re-search-forward "^* Heading 1::$")
+          (re-search-forward "^* Heading 2 (Heading 1)::$")
+          (re-search-forward "^@node Heading 1$")
+          (re-search-forward "^@chapter Heading 1$")
+          (re-search-forward "^@node Heading 2 (Heading 1)$")
+          (re-search-forward "^@chapter Heading 2 (@ref{Heading 1})$")))))))
+
 (provide 'test-ox-texinfo)
 ;;; test-ox-texinfo.el end here
-- 
2.39.3 (Apple Git-146)


[-- Warning: decoded text below may be mangled, UTF-8 assumed --]
[-- Attachment #4: 0003-ox-texinfo-Support-numeric-values-of-toc-in-OPTIONS.patch --]
[-- Type: text/x-patch, Size: 5719 bytes --]

From a45ed0eaa84144a5a34324d54aa326d5476ac5c2 Mon Sep 17 00:00:00 2001
From: Rudolf Adamkovic <rudolf@adamkovic.org>
Date: Sat, 24 Aug 2024 15:26:18 +0200
Subject: [PATCH 3/3] ox-texinfo: Support numeric values of `toc' in
 `#+OPTIONS'

* etc/ORG-NEWS (Texinfo exporter now supports numeric =toc= values in
=#+OPTIONS:=): Announce the new feature.
* lisp/ox.el (org-export-excluded-from-toc-p): Return nil for all
headlines that are above the `toc' value in export `#+OPTIONS'.
* testing/lisp/test-ox-texinfo.el
(test-ox-texinfo/headings-and-table-of-contents): Add a test.
---
 etc/ORG-NEWS                    |  9 +++++
 lisp/ox.el                      |  6 ++-
 testing/lisp/test-ox-texinfo.el | 66 +++++++++++++++++++++++++++++++++
 3 files changed, 80 insertions(+), 1 deletion(-)

diff --git a/etc/ORG-NEWS b/etc/ORG-NEWS
index 0ae1e4c81..3c0bfc173 100644
--- a/etc/ORG-NEWS
+++ b/etc/ORG-NEWS
@@ -85,6 +85,15 @@ You can now alter the Texinfo navigation hierarchy by specifying the
 information, see "13.14.6 Headings and sectioning structure" section
 of the Org manual.
 
+*** Texinfo exporter now supports numeric =toc= values in =#+OPTIONS:=
+
+For example, given =H:3= and =toc:2= in =#+OPTIONS:=, all headings at
+the 1st and 2nd level appear in the table of contents and those at the
+3rd level do not.  The latter will be instead exported as if they had
+the =UNNUMBERED= property set to =notoc=, that is using the Texinfo
+command given in the =unnumbered-no-toc-3= list position of the
+=org-texinfo-classes= variable, as documented.
+
 ** New and changed options
 
 # Chanes deadling with changing default values of customizations,
diff --git a/lisp/ox.el b/lisp/ox.el
index 7a0ab4dc7..acc535f72 100644
--- a/lisp/ox.el
+++ b/lisp/ox.el
@@ -5625,7 +5625,11 @@ contents.  However, it is useful if some additional processing is
 required on headlines excluded from table of contents."
   (or (org-element-property :footnote-section-p headline)
       (org-export-low-level-p headline info)
-      (equal "notoc" (org-export-get-node-property :UNNUMBERED headline t))))
+      (equal "notoc" (org-export-get-node-property :UNNUMBERED headline t))
+      (let ((toc-depth (plist-get info :with-toc)))
+        (and (wholenump toc-depth)
+             (> (org-element-property :level headline)
+                toc-depth)))))
 
 (defun org-export-toc-entry-backend (parent &rest transcoders)
   "Return an export backend appropriate for table of contents entries.
diff --git a/testing/lisp/test-ox-texinfo.el b/testing/lisp/test-ox-texinfo.el
index 4a521b3ed..e6dfaf876 100644
--- a/testing/lisp/test-ox-texinfo.el
+++ b/testing/lisp/test-ox-texinfo.el
@@ -663,5 +663,71 @@ body
           (re-search-forward "^@node Heading 2 (Heading 1)$")
           (re-search-forward "^@chapter Heading 2 (@ref{Heading 1})$")))))))
 
+\f
+;;; Headings and table of contents
+
+(ert-deftest test-ox-texinfo/headings-and-table-of-contents ()
+  "Test headings with regards to table of contents."
+  (should
+   (org-test-with-temp-text
+       (string-join
+        (list "#+OPTIONS: h:2 toc:1 num:nil"
+              "* Level 1"
+              "** Level 1-1"
+              "*** Level 1-1-1"
+              "*** Level 1-1-2"
+              "** Level 1-2"
+              "*** Level 1-2-1"
+              "*** Level 1-2-2"
+              "* Level 2"
+              "** Level 2-1"
+              "*** Level 2-1-1"
+              "*** Level 2-1-2"
+              "** Level 2-2"
+              "*** Level 2-2-1"
+              "*** Level 2-2-2")
+        "\n")
+     (let ((export-buffer "*Test Texinfo Export*")
+           (org-export-show-temporary-export-buffer nil))
+       (org-export-to-buffer 'texinfo export-buffer
+         nil nil nil nil nil
+         #'texinfo-mode)
+       (with-current-buffer export-buffer
+         (goto-char (point-min))
+         (and
+          (re-search-forward "^@menu$")
+          (re-search-forward "^* Level 1::$")
+          (re-search-forward "^* Level 2::$")
+          (re-search-forward "^@detailmenu$")
+          (re-search-forward "^Level 1$")
+          (re-search-forward "^* Level 1-1::$")
+          (re-search-forward "^* Level 1-2::$")
+          (re-search-forward "^Level 2$")
+          (re-search-forward "^* Level 2-1::$")
+          (re-search-forward "^* Level 2-2::$")
+          (re-search-forward "^@node Level 1$")
+          (re-search-forward "^@unnumbered Level 1$")
+          (re-search-forward "^@menu$")
+          (re-search-forward "^* Level 1-1::$")
+          (re-search-forward "^* Level 1-2::$")
+          (re-search-forward "^@heading Level 1-1$")
+          (re-search-forward "^@anchor{Level 1-1-1}Level 1-1-1$")
+          (re-search-forward "^@anchor{Level 1-1-2}Level 1-1-2$")
+          (re-search-forward "^@heading Level 1-2$")
+          (re-search-forward "^@anchor{Level 1-2-1}Level 1-2-1$")
+          (re-search-forward "^@anchor{Level 1-2-2}Level 1-2-2$")
+          (re-search-forward "^@node Level 2$")
+          (re-search-forward "^@unnumbered Level 2$")
+          (re-search-forward "^@menu$")
+          (re-search-forward "^* Level 2-1::$")
+          (re-search-forward "^* Level 2-2::$")
+          (re-search-forward "^@heading Level 2-1$")
+          (re-search-forward "^@anchor{Level 2-1-1}Level 2-1-1$")
+          (re-search-forward "^@anchor{Level 2-1-2}Level 2-1-2$")
+          (re-search-forward "^@anchor{Level 2-2}$")
+          (re-search-forward "^@heading Level 2-2$")
+          (re-search-forward "^@anchor{Level 2-2-1}Level 2-2-1$")
+          (re-search-forward "^@anchor{Level 2-2-2}Level 2-2-2$")))))))
+
 (provide 'test-ox-texinfo)
 ;;; test-ox-texinfo.el end here
-- 
2.39.3 (Apple Git-146)


[-- Attachment #5: Type: text/plain, Size: 173 bytes --]

-- 
"Thinking is a momentary dismissal of irrelevancies."
--- Richard Buckminster Fuller, 1969

Rudolf Adamkovič <rudolf@adamkovic.org> [he/him]
http://adamkovic.org

  reply	other threads:[~2024-08-24 19:55 UTC|newest]

Thread overview: 9+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2024-08-22 18:20 Texinfo: Custom navigation and links in headings Rudolf Adamkovič
2024-08-23  7:08 ` Rudolf Adamkovič
2024-08-23 12:45   ` Rudolf Adamkovič
2024-08-24  6:46     ` [PATCH] Re: Texinfo: Custom navigation and links in headings (was: Texinfo: Custom navigation and links in headings) Rudolf Adamkovič
2024-08-24  9:19       ` [PATCH] Re: Texinfo: Custom navigation and links in headings Rudolf Adamkovič
2024-08-24 19:54         ` Rudolf Adamkovič [this message]
2024-08-30 13:19           ` [PATCH v3] Improvements to Texinfo exporter (was: [PATCH] Re: Texinfo: Custom navigation and links in headings) Rudolf Adamkovič
2024-08-31 13:36             ` Ihor Radchenko
2024-08-31 20:48             ` Rudolf Adamkovič

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

  List information: https://www.orgmode.org/

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=m2cylxlmr8.fsf@adamkovic.org \
    --to=rudolf@adamkovic.org \
    --cc=emacs-orgmode@gnu.org \
    --cc=yantar92@posteo.net \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
Code repositories for project(s) associated with this public inbox

	https://git.savannah.gnu.org/cgit/emacs/org-mode.git

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for read-only IMAP folder(s) and NNTP newsgroup(s).