Skip to content

Commit 271bbba

Browse files
authored
Merge pull request #2414 from notriddle/on2
Load the sidebar toc from a shared JS file or iframe
2 parents 86ff2e1 + 203685e commit 271bbba

File tree

9 files changed

+272
-122
lines changed

9 files changed

+272
-122
lines changed

src/renderer/html_handlebars/hbs_renderer.rs

+17
Original file line numberDiff line numberDiff line change
@@ -528,6 +528,11 @@ impl Renderer for HtmlHandlebars {
528528
debug!("Register the header handlebars template");
529529
handlebars.register_partial("header", String::from_utf8(theme.header.clone())?)?;
530530

531+
debug!("Register the toc handlebars template");
532+
handlebars.register_template_string("toc_js", String::from_utf8(theme.toc_js.clone())?)?;
533+
handlebars
534+
.register_template_string("toc_html", String::from_utf8(theme.toc_html.clone())?)?;
535+
531536
debug!("Register handlebars helpers");
532537
self.register_hbs_helpers(&mut handlebars, &html_config);
533538

@@ -583,6 +588,18 @@ impl Renderer for HtmlHandlebars {
583588
debug!("Creating print.html ✓");
584589
}
585590

591+
debug!("Render toc");
592+
{
593+
let rendered_toc = handlebars.render("toc_js", &data)?;
594+
utils::fs::write_file(destination, "toc.js", rendered_toc.as_bytes())?;
595+
debug!("Creating toc.js ✓");
596+
data.insert("is_toc_html".to_owned(), json!(true));
597+
let rendered_toc = handlebars.render("toc_html", &data)?;
598+
utils::fs::write_file(destination, "toc.html", rendered_toc.as_bytes())?;
599+
debug!("Creating toc.html ✓");
600+
data.remove("is_toc_html");
601+
}
602+
586603
debug!("Copy static files");
587604
self.copy_static_files(destination, &theme, &html_config)
588605
.with_context(|| "Unable to copy across static files")?;

src/renderer/html_handlebars/helpers/toc.rs

+19-43
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,7 @@
11
use std::path::Path;
22
use std::{cmp::Ordering, collections::BTreeMap};
33

4-
use crate::utils;
5-
use crate::utils::bracket_escape;
4+
use crate::utils::special_escape;
65

76
use handlebars::{
87
Context, Handlebars, Helper, HelperDef, Output, RenderContext, RenderError, RenderErrorReason,
@@ -32,21 +31,6 @@ impl HelperDef for RenderToc {
3231
RenderErrorReason::Other("Could not decode the JSON data".to_owned()).into()
3332
})
3433
})?;
35-
let current_path = rc
36-
.evaluate(ctx, "@root/path")?
37-
.as_json()
38-
.as_str()
39-
.ok_or_else(|| {
40-
RenderErrorReason::Other("Type error for `path`, string expected".to_owned())
41-
})?
42-
.replace('\"', "");
43-
44-
let current_section = rc
45-
.evaluate(ctx, "@root/section")?
46-
.as_json()
47-
.as_str()
48-
.map(str::to_owned)
49-
.unwrap_or_default();
5034

5135
let fold_enable = rc
5236
.evaluate(ctx, "@root/fold_enable")?
@@ -64,31 +48,27 @@ impl HelperDef for RenderToc {
6448
RenderErrorReason::Other("Type error for `fold_level`, u64 expected".to_owned())
6549
})?;
6650

51+
// If true, then this is the iframe and we need target="_parent"
52+
let is_toc_html = rc
53+
.evaluate(ctx, "@root/is_toc_html")?
54+
.as_json()
55+
.as_bool()
56+
.unwrap_or(false);
57+
6758
out.write("<ol class=\"chapter\">")?;
6859

6960
let mut current_level = 1;
70-
// The "index" page, which has this attribute set, is supposed to alias the first chapter in
71-
// the book, i.e. the first link. There seems to be no easy way to determine which chapter
72-
// the "index" is aliasing from within the renderer, so this is used instead to force the
73-
// first link to be active. See further below.
74-
let mut is_first_chapter = ctx.data().get("is_index").is_some();
7561

7662
for item in chapters {
77-
let (section, level) = if let Some(s) = item.get("section") {
63+
let (_section, level) = if let Some(s) = item.get("section") {
7864
(s.as_str(), s.matches('.').count())
7965
} else {
8066
("", 1)
8167
};
8268

83-
let is_expanded =
84-
if !fold_enable || (!section.is_empty() && current_section.starts_with(section)) {
85-
// Expand if folding is disabled, or if the section is an
86-
// ancestor or the current section itself.
87-
true
88-
} else {
89-
// Levels that are larger than this would be folded.
90-
level - 1 < fold_level as usize
91-
};
69+
// Expand if folding is disabled, or if levels that are larger than this would not
70+
// be folded.
71+
let is_expanded = !fold_enable || level - 1 < (fold_level as usize);
9272

9373
match level.cmp(&current_level) {
9474
Ordering::Greater => {
@@ -121,7 +101,7 @@ impl HelperDef for RenderToc {
121101
// Part title
122102
if let Some(title) = item.get("part") {
123103
out.write("<li class=\"part-title\">")?;
124-
out.write(&bracket_escape(title))?;
104+
out.write(&special_escape(title))?;
125105
out.write("</li>")?;
126106
continue;
127107
}
@@ -139,16 +119,12 @@ impl HelperDef for RenderToc {
139119
.replace('\\', "/");
140120

141121
// Add link
142-
out.write(&utils::fs::path_to_root(&current_path))?;
143122
out.write(&tmp)?;
144-
out.write("\"")?;
145-
146-
if path == &current_path || is_first_chapter {
147-
is_first_chapter = false;
148-
out.write(" class=\"active\"")?;
149-
}
150-
151-
out.write(">")?;
123+
out.write(if is_toc_html {
124+
"\" target=\"_parent\">"
125+
} else {
126+
"\">"
127+
})?;
152128
path_exists = true;
153129
}
154130
_ => {
@@ -167,7 +143,7 @@ impl HelperDef for RenderToc {
167143
}
168144

169145
if let Some(name) = item.get("name") {
170-
out.write(&bracket_escape(name))?
146+
out.write(&special_escape(name))?
171147
}
172148

173149
if path_exists {

src/theme/css/chrome.css

+16
Original file line numberDiff line numberDiff line change
@@ -420,6 +420,22 @@ ul#searchresults span.teaser em {
420420
background-color: var(--sidebar-bg);
421421
color: var(--sidebar-fg);
422422
}
423+
.sidebar-iframe-inner {
424+
background-color: var(--sidebar-bg);
425+
color: var(--sidebar-fg);
426+
padding: 10px 10px;
427+
margin: 0;
428+
font-size: 1.4rem;
429+
}
430+
.sidebar-iframe-outer {
431+
border: none;
432+
height: 100%;
433+
position: absolute;
434+
top: 0;
435+
bottom: 0;
436+
left: 0;
437+
right: 0;
438+
}
423439
[dir=rtl] .sidebar { left: unset; right: 0; }
424440
.sidebar-resizing {
425441
-moz-user-select: none;

src/theme/index.hbs

+6-24
Original file line numberDiff line numberDiff line change
@@ -106,35 +106,17 @@
106106
</script>
107107

108108
<nav id="sidebar" class="sidebar" aria-label="Table of contents">
109-
<div class="sidebar-scrollbox">
110-
{{#toc}}{{/toc}}
111-
</div>
109+
<!-- populated by js -->
110+
<div class="sidebar-scrollbox"></div>
111+
<noscript>
112+
<iframe class="sidebar-iframe-outer" src="{{ path_to_root }}toc.html"></iframe>
113+
</noscript>
112114
<div id="sidebar-resize-handle" class="sidebar-resize-handle">
113115
<div class="sidebar-resize-indicator"></div>
114116
</div>
115117
</nav>
116118

117-
<!-- Track and set sidebar scroll position -->
118-
<script>
119-
var sidebarScrollbox = document.querySelector('#sidebar .sidebar-scrollbox');
120-
sidebarScrollbox.addEventListener('click', function(e) {
121-
if (e.target.tagName === 'A') {
122-
sessionStorage.setItem('sidebar-scroll', sidebarScrollbox.scrollTop);
123-
}
124-
}, { passive: true });
125-
var sidebarScrollTop = sessionStorage.getItem('sidebar-scroll');
126-
sessionStorage.removeItem('sidebar-scroll');
127-
if (sidebarScrollTop) {
128-
// preserve sidebar scroll position when navigating via links within sidebar
129-
sidebarScrollbox.scrollTop = sidebarScrollTop;
130-
} else {
131-
// scroll sidebar to current active section when navigating via "next/previous chapter" buttons
132-
var activeSection = document.querySelector('#sidebar .active');
133-
if (activeSection) {
134-
activeSection.scrollIntoView({ block: 'center' });
135-
}
136-
}
137-
</script>
119+
<script async src="{{ path_to_root }}toc.js"></script>
138120

139121
<div id="page-wrapper" class="page-wrapper">
140122

src/theme/mod.rs

+12
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,8 @@ pub static INDEX: &[u8] = include_bytes!("index.hbs");
1717
pub static HEAD: &[u8] = include_bytes!("head.hbs");
1818
pub static REDIRECT: &[u8] = include_bytes!("redirect.hbs");
1919
pub static HEADER: &[u8] = include_bytes!("header.hbs");
20+
pub static TOC_JS: &[u8] = include_bytes!("toc.js.hbs");
21+
pub static TOC_HTML: &[u8] = include_bytes!("toc.html.hbs");
2022
pub static CHROME_CSS: &[u8] = include_bytes!("css/chrome.css");
2123
pub static GENERAL_CSS: &[u8] = include_bytes!("css/general.css");
2224
pub static PRINT_CSS: &[u8] = include_bytes!("css/print.css");
@@ -50,6 +52,8 @@ pub struct Theme {
5052
pub head: Vec<u8>,
5153
pub redirect: Vec<u8>,
5254
pub header: Vec<u8>,
55+
pub toc_js: Vec<u8>,
56+
pub toc_html: Vec<u8>,
5357
pub chrome_css: Vec<u8>,
5458
pub general_css: Vec<u8>,
5559
pub print_css: Vec<u8>,
@@ -85,6 +89,8 @@ impl Theme {
8589
(theme_dir.join("head.hbs"), &mut theme.head),
8690
(theme_dir.join("redirect.hbs"), &mut theme.redirect),
8791
(theme_dir.join("header.hbs"), &mut theme.header),
92+
(theme_dir.join("toc.js.hbs"), &mut theme.toc_js),
93+
(theme_dir.join("toc.html.hbs"), &mut theme.toc_html),
8894
(theme_dir.join("book.js"), &mut theme.js),
8995
(theme_dir.join("css/chrome.css"), &mut theme.chrome_css),
9096
(theme_dir.join("css/general.css"), &mut theme.general_css),
@@ -174,6 +180,8 @@ impl Default for Theme {
174180
head: HEAD.to_owned(),
175181
redirect: REDIRECT.to_owned(),
176182
header: HEADER.to_owned(),
183+
toc_js: TOC_JS.to_owned(),
184+
toc_html: TOC_HTML.to_owned(),
177185
chrome_css: CHROME_CSS.to_owned(),
178186
general_css: GENERAL_CSS.to_owned(),
179187
print_css: PRINT_CSS.to_owned(),
@@ -232,6 +240,8 @@ mod tests {
232240
"head.hbs",
233241
"redirect.hbs",
234242
"header.hbs",
243+
"toc.js.hbs",
244+
"toc.html.hbs",
235245
"favicon.png",
236246
"favicon.svg",
237247
"css/chrome.css",
@@ -263,6 +273,8 @@ mod tests {
263273
head: Vec::new(),
264274
redirect: Vec::new(),
265275
header: Vec::new(),
276+
toc_js: Vec::new(),
277+
toc_html: Vec::new(),
266278
chrome_css: Vec::new(),
267279
general_css: Vec::new(),
268280
print_css: Vec::new(),

src/theme/toc.html.hbs

+43
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
<!DOCTYPE HTML>
2+
<html lang="{{ language }}" class="{{ default_theme }}" dir="{{ text_direction }}">
3+
<head>
4+
<!-- sidebar iframe generated using mdBook
5+
6+
This is a frame, and not included directly in the page, to control the total size of the
7+
book. The TOC contains an entry for each page, so if each page includes a copy of the TOC,
8+
the total size of the page becomes O(n**2).
9+
10+
The frame is only used as a fallback when JS is turned off. When it's on, the sidebar is
11+
instead added to the main page by `toc.js` instead. The JavaScript mode is better
12+
because, when running in a `file:///` URL, the iframed page would not be Same-Origin as
13+
the rest of the page, so the sidebar and the main page theme would fall out of sync.
14+
-->
15+
<meta charset="UTF-8">
16+
<meta name="robots" content="noindex">
17+
{{#if base_url}}
18+
<base href="{{ base_url }}">
19+
{{/if}}
20+
<!-- Custom HTML head -->
21+
{{> head}}
22+
<meta name="viewport" content="width=device-width, initial-scale=1">
23+
<meta name="theme-color" content="#ffffff">
24+
<link rel="stylesheet" href="{{ path_to_root }}css/variables.css">
25+
<link rel="stylesheet" href="{{ path_to_root }}css/general.css">
26+
<link rel="stylesheet" href="{{ path_to_root }}css/chrome.css">
27+
{{#if print_enable}}
28+
<link rel="stylesheet" href="{{ path_to_root }}css/print.css" media="print">
29+
{{/if}}
30+
<!-- Fonts -->
31+
<link rel="stylesheet" href="{{ path_to_root }}FontAwesome/css/font-awesome.css">
32+
{{#if copy_fonts}}
33+
<link rel="stylesheet" href="{{ path_to_root }}fonts/fonts.css">
34+
{{/if}}
35+
<!-- Custom theme stylesheets -->
36+
{{#each additional_css}}
37+
<link rel="stylesheet" href="{{ ../path_to_root }}{{ this }}">
38+
{{/each}}
39+
</head>
40+
<body class="sidebar-iframe-inner">
41+
{{#toc}}{{/toc}}
42+
</body>
43+
</html>

src/theme/toc.js.hbs

+54
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
// Populate the sidebar
2+
//
3+
// This is a script, and not included directly in the page, to control the total size of the book.
4+
// The TOC contains an entry for each page, so if each page includes a copy of the TOC,
5+
// the total size of the page becomes O(n**2).
6+
var sidebarScrollbox = document.querySelector("#sidebar .sidebar-scrollbox");
7+
sidebarScrollbox.innerHTML = '{{#toc}}{{/toc}}';
8+
(function() {
9+
let current_page = document.location.href.toString();
10+
if (current_page.endsWith("/")) {
11+
current_page += "index.html";
12+
}
13+
var links = sidebarScrollbox.querySelectorAll("a");
14+
var l = links.length;
15+
for (var i = 0; i < l; ++i) {
16+
var link = links[i];
17+
var href = link.getAttribute("href");
18+
if (href && !href.startsWith("#") && !/^(?:[a-z+]+:)?\/\//.test(href)) {
19+
link.href = path_to_root + href;
20+
}
21+
// The "index" page is supposed to alias the first chapter in the book.
22+
if (link.href === current_page || (i === 0 && path_to_root === "" && current_page.endsWith("/index.html"))) {
23+
link.classList.add("active");
24+
var parent = link.parentElement;
25+
while (parent) {
26+
if (parent.tagName === "LI" && parent.previousElementSibling) {
27+
if (parent.previousElementSibling.classList.contains("chapter-item")) {
28+
parent.previousElementSibling.classList.add("expanded");
29+
}
30+
}
31+
parent = parent.parentElement;
32+
}
33+
}
34+
}
35+
})();
36+
37+
// Track and set sidebar scroll position
38+
sidebarScrollbox.addEventListener('click', function(e) {
39+
if (e.target.tagName === 'A') {
40+
sessionStorage.setItem('sidebar-scroll', sidebarScrollbox.scrollTop);
41+
}
42+
}, { passive: true });
43+
var sidebarScrollTop = sessionStorage.getItem('sidebar-scroll');
44+
sessionStorage.removeItem('sidebar-scroll');
45+
if (sidebarScrollTop) {
46+
// preserve sidebar scroll position when navigating via links within sidebar
47+
sidebarScrollbox.scrollTop = sidebarScrollTop;
48+
} else {
49+
// scroll sidebar to current active section when navigating via "next/previous chapter" buttons
50+
var activeSection = document.querySelector('#sidebar .active');
51+
if (activeSection) {
52+
activeSection.scrollIntoView({ block: 'center' });
53+
}
54+
}

0 commit comments

Comments
 (0)