book: add toc

This commit is contained in:
René Kijewski 2024-11-14 11:06:40 +01:00
parent b76d1000b6
commit a9e0132b28
6 changed files with 262 additions and 37 deletions

View File

@ -11,3 +11,5 @@ edition = "2021"
[output.html]
git-repository-url = "https://github.com/rinja-rs/rinja/tree/master/book"
edit-url-template = "https://github.com/rinja-rs/rinja/tree/master/book/{path}"
additional-css = ["theme/pagetoc.css"]
additional-js = ["theme/pagetoc.js"]

View File

@ -17,42 +17,6 @@ Rinja has a collection of built-in filters, documented below, but can also inclu
Additionally, the `json` filter is included in the built-in filters, but is disabled by default.
Enable it with Cargo features (see below for more information).
**Table of contents**
* **[Built-in filters][#built-in-filters]:**
* [`capitalize`][#capitalize]
* [`center`][#center]
* [`deref`][#deref]
* [`escape|e`][#escape]
* [`filesizeformat`][#filesizeformat]
* [`fmt`][#fmt]
* [`format`][#format]
* [`indent`][#indent]
* [`join`][#join]
* [`linebreaks`][#linebreaks]
* [`linebreaksbr`][#linebreaksbr]
* [`lower|lowercase`][#lower]
* [`paragraphbreaks`][#paragraphbreaks]
* [`pluralize`][#pluralize]
* [`ref`][#ref]
* [`safe`][#safe]
* [`title`][#title]
* [`trim`][#trim]
* [`truncate`][#truncate]
* [`upper|uppercase`][#upper]
* [`urlencode`][#urlencode]
* [`wordcount`][#wordcount]
* **[Optional / feature gated filters][#optional-filters]:**
[`json|tojson`][#json],
* **[Custom filters][#custom-filters]**
* **[HTML-safe types][#html-safe-types]**
* **[Safe output of custom filters][#safe-output-of-custom-filters]**
## Built-In Filters
[#built-in-filters]: #built-in-filters

View File

@ -1 +1 @@
*
index.hbs

7
book/theme/LICENSE.md Normal file
View File

@ -0,0 +1,7 @@
`pagetoc.css` and `pagetoc.js` where copied from the [Trunk] project in revision [4c9d85f6f04a31a272c71db8df12e142c76311fb].
Author: Jens Reimann ([@ctron]) and the Trunk developers.
SPDX-License-Identifier: MIT/Apache-2.0.
[Trunk]: <https://github.com/trunk-rs/trunk/>
[@ctron]: <https://github.com/ctron>
[4c9d85f6f04a31a272c71db8df12e142c76311fb]: <https://github.com/trunk-rs/trunk/commits/4c9d85f6f04a31a272c71db8df12e142c76311fb/guide/theme>

109
book/theme/pagetoc.css Normal file
View File

@ -0,0 +1,109 @@
main {
display: none;
}
:root {
--toc-width: 270px;
--center-content-toc-shift: calc(-1 * var(--toc-width) / 2);
}
.nav-chapters {
/* adjust width of buttons that bring to the previous or the next page */
min-width: 50px;
}
.previous {
/*
adjust the space between the left sidebar or the left side of the screen
and the button that leads to the previous page
*/
margin-left: var(--page-padding);
}
@media only screen {
main.wrapped {
display: flex;
}
@media (max-width: 1179px) {
.sidebar-hidden .sidetoc {
display: none;
}
}
@media (max-width: 1439px) {
.sidebar-visible .sidetoc {
display: none;
}
}
@media (1180px <= width <= 1439px) {
.sidebar-hidden main {
position: relative;
left: var(--center-content-toc-shift);
}
}
@media (1440px <= width <= 1700px) {
.sidebar-visible main {
position: relative;
left: var(--center-content-toc-shift);
}
}
.content-wrap {
overflow-y: auto;
width: 100%;
}
.sidetoc {
margin-top: 20px;
margin-left: 10px;
margin-right: auto;
}
.pagetoc {
position: fixed;
/* adjust TOC width */
width: var(--toc-width);
height: calc(100vh - var(--menu-bar-height) - 0.67em * 4);
overflow: auto;
}
.pagetoc a {
border-left: 1px solid var(--sidebar-bg);
color: var(--fg) !important;
display: block;
padding-bottom: 5px;
padding-top: 5px;
padding-left: 10px;
text-align: left;
text-decoration: none;
}
.pagetoc a:hover,
.pagetoc a.active {
background: var(--sidebar-bg);
color: var(--sidebar-fg) !important;
}
.pagetoc .active {
background: var(--sidebar-bg);
color: var(--sidebar-fg);
font-weight: bold;
}
.pagetoc .pagetoc-H2 {
padding-left: 20px;
font-size: 90%;
}
.pagetoc .pagetoc-H3 {
padding-left: 40px;
font-size: 90%;
}
.pagetoc .pagetoc-H4 {
padding-left: 60px;
font-size: 90%;
}
}
@media print {
.sidetoc {
display: none;
}
}

143
book/theme/pagetoc.js Normal file
View File

@ -0,0 +1,143 @@
function forEach(elems, fun) {
Array.prototype.forEach.call(elems, fun);
}
function getPagetoc(){
const pagetoc = document.getElementsByClassName("pagetoc")[0];
if (pagetoc) {
return pagetoc;
}
return autoCreatePagetoc();
}
function autoCreatePagetoc() {
const main = document.querySelector("#content > main");
const content = document.createElement("div");
content.classList.add("content-wrap");
content.append(...main.childNodes);
main.appendChild(content);
main.classList.add("wrapped");
main.insertAdjacentHTML("beforeend", `
<div class="sidetoc">
<nav class="pagetoc"></nav>
</div>
`);
return document.getElementsByClassName("pagetoc")[0]
}
function getPagetocElems() {
return getPagetoc().children;
}
function getHeaders(){
return document.getElementsByClassName("header")
}
// Un-active everything when you click it
function forPagetocElem(fun) {
forEach(getPagetocElems(), fun);
}
function getRect(element) {
return element.getBoundingClientRect();
}
function overflowTop(container, element) {
return getRect(container).top - getRect(element).top;
}
function overflowBottom(container, element) {
return getRect(container).bottom - getRect(element).bottom;
}
var activeHref = location.href;
var updateFunction = function (elem = undefined) {
var id = elem;
if (!id && location.href != activeHref) {
activeHref = location.href;
forPagetocElem(function (el) {
if (el.href === activeHref) {
id = el;
}
});
}
if (!id) {
var elements = getHeaders();
let margin = window.innerHeight / 3;
forEach(elements, function (el, i, arr) {
if (!id && getRect(el).top >= 0) {
if (getRect(el).top < margin) {
id = el;
} else {
id = arr[Math.max(0, i - 1)];
}
}
// a very long last section
// its heading is over the screen
if (!id && i == arr.length - 1) {
id = el
}
});
}
forPagetocElem(function (el) {
el.classList.remove("active");
});
if (!id) return;
forPagetocElem(function (el) {
if (id.href.localeCompare(el.href) == 0) {
el.classList.add("active");
let pagetoc = getPagetoc();
if (overflowTop(pagetoc, el) > 0) {
pagetoc.scrollTop = el.offsetTop;
}
if (overflowBottom(pagetoc, el) < 0) {
pagetoc.scrollTop -= overflowBottom(pagetoc, el);
}
}
});
};
let elements = getHeaders();
if (elements.length > 1) {
// Populate sidebar on load
window.addEventListener("load", function () {
var pagetoc = getPagetoc();
var elements = getHeaders();
forEach(elements, function (el) {
var link = document.createElement("a");
link.appendChild(document.createTextNode(el.text));
link.href = el.hash;
link.classList.add("pagetoc-" + el.parentElement.tagName);
pagetoc.appendChild(link);
link.onclick = function () {
updateFunction(link);
};
});
updateFunction();
});
// Handle active elements on scroll
window.addEventListener("scroll", function () {
updateFunction();
});
} else {
getPagetoc();
const sidetoc = document.getElementsByClassName("sidetoc");
if (sidetoc.length > 0 ) {
document.getElementsByClassName("sidetoc")[0].remove();
}
}