From a4e4e312aa3e3f41344e351bb1324ec462911515 Mon Sep 17 00:00:00 2001 From: Peter Mosses <18308236+pdmosses@users.noreply.github.com> Date: Tue, 20 Aug 2024 22:34:11 +0200 Subject: [PATCH] Feature: Allow unlimited multi-level navigation (#1431) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * Allow unlimited multi-level navigation This PR supersedes #462. The only user-level difference from #462 is that disambiguation of parent pages has to use either `grand_parent` or `ancestor` titles: the somewhat unnatural `section_id` and `in_section` fields are not supported. The implementation has been significantly simplified by the changes introduced in v0.7.0 of the theme. * Detect cyclic parenthood A page should not have a parent or ancestor with the same title. If it does, the location of the repeated link is marked by ∞, to facilitate debugging the navigation (and an unbounded loop leading to a build exception is avoided). * Add nav_error_report warning in main navigation When activated by `nav_error_report: true` in `_config.yml`, displays warnings about pages with the same title as their parent page or an ancestral page. * Cache site-nav with links to all pages The extra cached site-nav is used for determining breadcrumbs and children navigation, which may involve pages that are excluded from the main navigation. * Replace code for determining children by inclusion of components/nav/children.html * Update CHANGELOG.md --------- Co-authored-by: Matt Wang --- CHANGELOG.md | 6 +- _config.yml | 3 + _includes/components/breadcrumbs.html | 174 +++---- _includes/components/children_nav.html | 88 +++- _includes/components/nav.html | 75 --- _includes/components/nav/children.html | 48 ++ _includes/components/nav/links.html | 53 ++ _includes/components/nav/pages.html | 23 + .../nav/sorted.html} | 6 +- _includes/components/site_nav.html | 19 +- _includes/css/activation.scss.liquid | 493 +++++++++--------- _layouts/default.html | 2 +- 12 files changed, 526 insertions(+), 464 deletions(-) delete mode 100644 _includes/components/nav.html create mode 100644 _includes/components/nav/children.html create mode 100644 _includes/components/nav/links.html create mode 100644 _includes/components/nav/pages.html rename _includes/{sorted_pages.html => components/nav/sorted.html} (96%) diff --git a/CHANGELOG.md b/CHANGELOG.md index 8f8e121..b1466af 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -17,12 +17,16 @@ This website is built from the `HEAD` of the `main` branch of the theme reposito Code changes to `main` that are *not* in the latest release: -- N/A +### New Features + +- Added: Allow unlimited multi-level navigation by [@pdmosses] in [#1431] Docs changes made since the latest release: - N/A +[#1431]: https://github.com/just-the-docs/just-the-docs/pull/1431 + ## Release v0.9.0 Hi folks! This minor release adds a `nav_enabled` set of variables to enable/disable the navigation at a site, layout, and page level --- and uses that to add search and auxilary links to the `minimal` layout. In addition, it fixes `search-data.json` corruption with default layouts and some minor CSS/SCSS issues. diff --git a/_config.yml b/_config.yml index a63240a..54c9bd1 100644 --- a/_config.yml +++ b/_config.yml @@ -122,6 +122,9 @@ nav_external_links: - title: Just the Docs on GitHub url: https://github.com/just-the-docs/just-the-docs +# Show navigation error report +nav_error_report: true # default is false/nil. + liquid: error_mode: strict strict_filters: true diff --git a/_includes/components/breadcrumbs.html b/_includes/components/breadcrumbs.html index 8668791..c945277 100644 --- a/_includes/components/breadcrumbs.html +++ b/_includes/components/breadcrumbs.html @@ -1,144 +1,96 @@ {%- comment -%} Include as: {%- include components/breadcrumbs.html -%} Depends on: page, site. + Includes: components/site_nav.html. Results in: HTML for the breadcrumbs component. Overwrites: - node, pages_list, parent_page, grandparent_page. + nav_list_link, site_nav, nav_list_simple, nav_list_link_class, nav_category, + nav_anchor_splits, nav_breadcrumbs, nav_split, nav_split_next, nav_split_test, + nav_breadcrumb_link, nav_list_end_less, nav_list_end_count, nav_end_index, nav_breadcrumb. {%- endcomment -%} -{%- if page.url != "/" and page.parent -%} +{%- if page.url != "/" and page.parent and page.title -%} {%- capture nav_list_link -%} {%- endcapture -%} {%- capture site_nav -%} -{%- include_cached components/site_nav.html -%} +{%- include_cached components/site_nav.html all=true -%} {%- endcapture -%} -{%- if site_nav contains nav_list_link -%} +{%- capture nav_list_simple -%} +" -%} - {%- assign nav_list_end_less = nav_split_next | remove: "" -%} - {%- assign nav_list_end_count = - nav_split_next.size | minus: nav_list_end_less.size | divided_by: 5 -%} - {% for nav_end_index in (1..nav_list_end_count) %} - {%- assign nav_breadcrumbs = nav_breadcrumbs | pop -%} - {%- endfor -%} - {%- endif -%} - - {%- endunless -%} - {%- endfor -%} - - {%- assign nav_parent_link = nav_breadcrumbs[-1] -%} - {%- assign nav_grandparent_link = nav_breadcrumbs[-2] -%} - -{%- else -%} - - {%- comment -%} - Pages whose links are excluded from the main navigation may still have - breadcrumbs. Determining them appears to require inspecting the front matter - of all the pages in the same group. For sites with 100s of pages, this is too - inefficient in Jekyll 3 (also when the for-loop is replaced by where-filters). - {%- endcomment -%} - - {%- assign pages_list = site[page.collection] | default: site.html_pages -%} +{%- comment -%} + The ordinary pages (if any) and the collections pages (if any) are separated by + occurrences of nav_category. - {%- assign parent_page = nil -%} - {%- assign grandparent_page = nil -%} - - {%- for node in pages_list -%} + Any ancestor nav-links of the page are contained in the last group of pages, + immediately preceding nav-lists. After splitting at "", the anchor that + was split is a potential ancestor link when the following split starts with + a nav-list. - {%- if node.has_children and page.grand_parent -%} - - {%- if node.title == page.parent and node.parent == page.grand_parent -%} - {%- assign parent_page = node -%} - {%- endif -%} - {%- if node.title == page.grand_parent -%} - {%- assign grandparent_page = node -%} - {%- endif -%} - {%- if parent_page and grandparent_page -%} - {%- break -%} - {%- endif -%} - - {%- elsif node.has_children and node.title == page.parent and node.parent == nil -%} - - {%- assign parent_page = node -%} - {%- break -%} - - {%- endif -%} - - {%- endfor -%} + The array nav_breadcrumbs is the stack of current potential ancestors of the + current page. A split that contains one or more ""s requires that number + of potential ancestors to be popped from the stack. - {%- capture nav_parent_link -%} - {{ page.parent }} - {%- endcapture -%} + The number of occurrences of a string in nav_split_next is computed by removing + them all, then dividing the resulting size difference by the length of the string. +{%- endcomment %} - {%- if page.grand_parent %} - {%- capture nav_grandparent_link -%} - {{ page.grand_parent }} - {%- endcapture -%} - {%- endif -%} +{%- assign nav_breadcrumbs = "" | split: "" -%} +{%- for nav_split in nav_anchor_splits -%} +{%- unless forloop.last -%} + +{%- assign nav_split_next = nav_anchor_splits[forloop.index] | strip -%} + +{%- assign nav_split_test = + nav_split_next | remove_first: nav_list_simple | prepend: nav_list_simple -%} +{%- if nav_split_test == nav_split_next -%} + {%- assign nav_breadcrumb_link = + nav_split | split: "" | append: "" -%} + {%- assign nav_breadcrumbs = nav_breadcrumbs | push: nav_breadcrumb_link -%} {%- endif -%} +{%- if nav_split_next contains "" -%} + {%- assign nav_list_end_less = nav_split_next | remove: "" -%} + {%- assign nav_list_end_count = + nav_split_next.size | minus: nav_list_end_less.size | divided_by: 5 -%} + {% for nav_end_index in (1..nav_list_end_count) %} + {%- assign nav_breadcrumbs = nav_breadcrumbs | pop -%} + {%- endfor -%} +{%- endif -%} + +{%- endunless -%} +{%- endfor -%} + +{% if site.nav_error_report %} +{{ nav_error_report }} +{% endif %} + {%- endif -%} diff --git a/_includes/components/children_nav.html b/_includes/components/children_nav.html index ce2482a..d233ebd 100644 --- a/_includes/components/children_nav.html +++ b/_includes/components/children_nav.html @@ -1,33 +1,89 @@ {%- comment -%} Include as: {%- include components/children_nav.html -%} - Depends on: page, site. + Depends on: page, site, nav_breadcrumbs. Results in: HTML for the children-navigation component. - Includes: - sorted_pages.html - toc_heading_custom.html + Includes: components/nav/sorted.html, toc_heading_custom.html. Overwrites: - child_pages. + nav_ancestor_links, nav_top_node_titles, nav_child_candidates, nav_children, + nav_child, nav_child_ok, nav_child_ancestor, nav_sorted. {%- endcomment -%} -{%- if page.has_children == true and page.has_toc != false -%} - {%- assign child_pages = site[page.collection] - | default: site.html_pages - | where: "parent", page.title - | where: "grand_parent", page.parent -%} +{%- comment -%} + Whether a page has any children is checked efficiently by inspecting the cached + site_nav. If the page has no children, nav_children is set to an empty array; + otherwise nav_children is left unset. +{%- endcomment -%} - {%- include sorted_pages.html pages = child_pages -%} +{%- if page.has_children == false -%} + {%- assign nav_children = "" | split: "" -%} +{%- else -%} + + {%- assign nav_children = nil -%} + + {%- capture nav_list_link -%} + + {%- endcapture -%} + + {%- capture site_nav -%} + {%- include_cached components/site_nav.html all=true -%} + {%- endcapture -%} + + {%- capture nav_list_simple -%} +