mirror of
https://github.com/askama-rs/askama.git
synced 2025-10-02 07:20:55 +00:00
book: add toc
This commit is contained in:
parent
b76d1000b6
commit
a9e0132b28
@ -11,3 +11,5 @@ edition = "2021"
|
|||||||
[output.html]
|
[output.html]
|
||||||
git-repository-url = "https://github.com/rinja-rs/rinja/tree/master/book"
|
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}"
|
edit-url-template = "https://github.com/rinja-rs/rinja/tree/master/book/{path}"
|
||||||
|
additional-css = ["theme/pagetoc.css"]
|
||||||
|
additional-js = ["theme/pagetoc.js"]
|
||||||
|
@ -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.
|
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).
|
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]: #built-in-filters
|
[#built-in-filters]: #built-in-filters
|
||||||
|
|
||||||
|
2
book/theme/.gitignore
vendored
2
book/theme/.gitignore
vendored
@ -1 +1 @@
|
|||||||
*
|
index.hbs
|
||||||
|
7
book/theme/LICENSE.md
Normal file
7
book/theme/LICENSE.md
Normal 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
109
book/theme/pagetoc.css
Normal 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
143
book/theme/pagetoc.js
Normal 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();
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user