dynamic loading

This commit is contained in:
リキ萌 2023-08-20 15:05:59 +02:00
parent b28a4f5b9a
commit aa91046bc5
7 changed files with 218 additions and 63 deletions

View file

@ -166,4 +166,4 @@
- breaking - breaking
% content.link = "secret" % content.link = "secret"
- this block includes another block's content + this block includes another block's content

View file

@ -18,3 +18,21 @@ impl<'a> Display for EscapeAttribute<'a> {
Ok(()) Ok(())
} }
} }
pub struct EscapeHtml<'a>(&'a str);
impl<'a> Display for EscapeHtml<'a> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
for c in self.0.chars() {
match c {
'<' => f.write_str("&lt;")?,
'>' => f.write_str("&gt;")?,
'&' => f.write_str("&amp;")?,
'\'' => f.write_str("&apos;")?,
'"' => f.write_str("&quot;")?,
_ => f.write_char(c)?,
}
}
Ok(())
}
}

View file

@ -8,76 +8,37 @@ use crate::{
state::{toml_error_to_diagnostic, FileId, TomlError, Treehouse}, state::{toml_error_to_diagnostic, FileId, TomlError, Treehouse},
}; };
use super::{attributes::Attributes, markdown}; use super::{
attributes::{Attributes, Content},
markdown, EscapeHtml,
};
pub fn branch_to_html(s: &mut String, treehouse: &mut Treehouse, file_id: FileId, branch: &Branch) { pub fn branch_to_html(s: &mut String, treehouse: &mut Treehouse, file_id: FileId, branch: &Branch) {
let source = treehouse.get_source(file_id); let attributes = parse_attributes(treehouse, file_id, branch);
let mut successfully_parsed = true;
let mut attributes = if let Some(attributes) = &branch.attributes {
toml_edit::de::from_str(&source[attributes.data.clone()]).unwrap_or_else(|error| {
treehouse
.diagnostics
.push(toml_error_to_diagnostic(TomlError {
message: error.message().to_owned(),
span: error.span(),
file_id,
input_range: attributes.data.clone(),
}));
successfully_parsed = false;
Attributes::default()
})
} else {
Attributes::default()
};
let successfully_parsed = successfully_parsed;
// Only check for attribute validity if the attributes were parsed successfully.
if successfully_parsed {
let attribute_warning_span = branch
.attributes
.as_ref()
.map(|attributes| attributes.percent.clone())
.unwrap_or(branch.kind_span.clone());
if attributes.id.is_empty() {
attributes.id = format!("treehouse-missingno-{}", treehouse.next_missingno());
treehouse.diagnostics.push(Diagnostic {
severity: Severity::Warning,
code: Some("attr".into()),
message: "branch does not have an `id` attribute".into(),
labels: vec![Label {
style: LabelStyle::Primary,
file_id,
range: attribute_warning_span,
message: String::new(),
}],
notes: vec![
format!(
"note: a generated id `{}` will be used, but this id is unstable and will not persist across generations",
attributes.id
),
format!("help: run `treehouse fix {}` to add missing ids to branches", treehouse.get_filename(file_id)),
],
});
}
}
// Reborrow because the closure requires unique access (it adds a new diagnostic.) // Reborrow because the closure requires unique access (it adds a new diagnostic.)
let source = treehouse.get_source(file_id); let source = treehouse.get_source(file_id);
let class = if !branch.children.is_empty() { let has_children =
"branch" !branch.children.is_empty() || matches!(attributes.content, Content::Link(_));
let class = if has_children { "branch" } else { "leaf" };
let linked_branch = if let Content::Link(link) = &attributes.content {
format!(
" is=\"th-linked-branch\" data-th-link=\"{}\"",
EscapeHtml(link)
)
} else { } else {
"leaf" String::new()
}; };
write!( write!(
s, s,
"<li class=\"{class}\" id=\"{}\">", "<li class=\"{class}\" id=\"{}\"{linked_branch}>",
EscapeAttribute(&attributes.id) EscapeAttribute(&attributes.id)
) )
.unwrap(); .unwrap();
{ {
if !branch.children.is_empty() { if has_children {
s.push_str(match branch.kind { s.push_str(match branch.kind {
BranchKind::Expanded => "<details open>", BranchKind::Expanded => "<details open>",
BranchKind::Collapsed => "<details>", BranchKind::Collapsed => "<details>",
@ -110,18 +71,37 @@ pub fn branch_to_html(s: &mut String, treehouse: &mut Treehouse, file_id: FileId
}); });
markdown::push_html(s, markdown_parser); markdown::push_html(s, markdown_parser);
s.push_str("<th-bb>"); if let Content::Link(link) = &attributes.content {
{
write!( write!(
s, s,
"<a class=\"branch-link\" href=\"#{}\" title=\"permalink\"></a>", "<noscript><a class=\"navigate icon-go\" href=\"{}.html\">Go to linked tree: <code>{}</code></a></noscript>",
EscapeAttribute(link),
EscapeHtml(link),
)
.unwrap();
}
s.push_str("<th-bb>");
{
if let Content::Link(link) = &attributes.content {
write!(
s,
"<a class=\"icon icon-go\" href=\"{}.html\" title=\"linked tree\"></a>",
EscapeAttribute(link),
)
.unwrap();
}
write!(
s,
"<a class=\"icon icon-permalink\" href=\"#{}\" title=\"permalink\"></a>",
EscapeAttribute(&attributes.id) EscapeAttribute(&attributes.id)
) )
.unwrap(); .unwrap();
} }
s.push_str("</th-bb>"); s.push_str("</th-bb>");
if !branch.children.is_empty() { if has_children {
s.push_str("</summary>"); s.push_str("</summary>");
branches_to_html(s, treehouse, file_id, &branch.children); branches_to_html(s, treehouse, file_id, &branch.children);
s.push_str("</details>"); s.push_str("</details>");
@ -132,6 +112,82 @@ pub fn branch_to_html(s: &mut String, treehouse: &mut Treehouse, file_id: FileId
s.push_str("</li>"); s.push_str("</li>");
} }
fn parse_attributes(treehouse: &mut Treehouse, file_id: usize, branch: &Branch) -> Attributes {
let source = treehouse.get_source(file_id);
let mut successfully_parsed = true;
let mut attributes = if let Some(attributes) = &branch.attributes {
toml_edit::de::from_str(&source[attributes.data.clone()]).unwrap_or_else(|error| {
treehouse
.diagnostics
.push(toml_error_to_diagnostic(TomlError {
message: error.message().to_owned(),
span: error.span(),
file_id,
input_range: attributes.data.clone(),
}));
successfully_parsed = false;
Attributes::default()
})
} else {
Attributes::default()
};
let successfully_parsed = successfully_parsed;
// Only check for attribute validity if the attributes were parsed successfully.
if successfully_parsed {
let attribute_warning_span = branch
.attributes
.as_ref()
.map(|attributes| attributes.percent.clone())
.unwrap_or(branch.kind_span.clone());
// Check that every block has an ID.
if attributes.id.is_empty() {
attributes.id = format!("treehouse-missingno-{}", treehouse.next_missingno());
treehouse.diagnostics.push(Diagnostic {
severity: Severity::Warning,
code: Some("attr".into()),
message: "branch does not have an `id` attribute".into(),
labels: vec![Label {
style: LabelStyle::Primary,
file_id,
range: attribute_warning_span.clone(),
message: String::new(),
}],
notes: vec![
format!(
"note: a generated id `{}` will be used, but this id is unstable and will not persist across generations",
attributes.id
),
format!("help: run `treehouse fix {}` to add missing ids to branches", treehouse.get_filename(file_id)),
],
});
}
// Check that link-type blocks are `+`-type to facilitate lazy loading.
if let Content::Link(_) = &attributes.content {
if branch.kind == BranchKind::Expanded {
treehouse.diagnostics.push(Diagnostic {
severity: Severity::Warning,
code: Some("attr".into()),
message: "`content.link` branch is expanded by default".into(),
labels: vec![Label {
style: LabelStyle::Primary,
file_id,
range: branch.kind_span.clone(),
message: String::new(),
}],
notes: vec![
"note: `content.link` branches should normally be collapsed to allow for lazy loading".into(),
],
});
}
}
}
attributes
}
pub fn branches_to_html( pub fn branches_to_html(
s: &mut String, s: &mut String,
treehouse: &mut Treehouse, treehouse: &mut Treehouse,

View file

@ -97,8 +97,7 @@
opacity: 100%; opacity: 100%;
} }
.tree .branch-link { .tree .icon {
background-image: url("../svg/link.svg");
background-repeat: no-repeat; background-repeat: no-repeat;
background-position: 50% 50%; background-position: 50% 50%;
opacity: 35%; opacity: 35%;
@ -106,3 +105,24 @@
width: 24px; width: 24px;
height: 24px; height: 24px;
} }
.tree .icon-permalink {
background-image: url("../svg/permalink.svg");
}
.tree .icon-go {
background-image: url("../svg/go.svg");
}
.tree a.navigate {
background-repeat: no-repeat;
background-position: 0 50%;
opacity: 50%;
color: #000;
padding-left: 20px;
}
.tree .link-loading {
padding-left: 24px;
opacity: 50%;
}

60
static/js/tree.js Normal file
View file

@ -0,0 +1,60 @@
class LinkedBranch extends HTMLLIElement {
constructor() {
super();
this.linkedTree = this.getAttribute("data-th-link");
this.details = this.childNodes[0];
this.innerUL = this.details.childNodes[1];
this.state = "notloaded";
this.loadingText = document.createElement("p");
{
this.loadingText.className = "link-loading";
let linkedTreeName = document.createElement("code");
linkedTreeName.innerText = this.linkedTree;
this.loadingText.append("Loading ", linkedTreeName, "...");
}
this.innerUL.appendChild(this.loadingText);
// This produces a warning during static generation but we still want to handle that
// correctly. Having an expanded-by-default linked block can be useful in development.
if (this.details.open) {
this.loadTree();
}
this.details.addEventListener("toggle", event => {
if (this.details.open) {
this.loadTree();
}
});
}
loadTree() {
if (this.state == "notloaded") {
this.state = "loading";
fetch(`/${this.linkedTree}.html`)
.then(request => request.text())
.then(text => {
let parser = new DOMParser();
let linkedDocument = parser.parseFromString(text, "text/html");
let main = linkedDocument.getElementsByTagName("main")[0];
let ul /*: Element */ = main.getElementsByTagName("ul")[0];
console.log(ul);
this.loadingText.remove();
for (let i = 0; i < ul.childNodes.length; ++i) {
this.innerUL.appendChild(ul.childNodes[i]);
}
})
.catch(error => {
this.loadingText.innerText = error.toString();
});
}
}
}
customElements.define("th-linked-branch", LinkedBranch, { extends: "li" });

View file

Before

Width:  |  Height:  |  Size: 1.3 KiB

After

Width:  |  Height:  |  Size: 1.3 KiB

View file

@ -11,6 +11,7 @@
<link rel="stylesheet" href="{{ site }}/static/css/tree.css"> <link rel="stylesheet" href="{{ site }}/static/css/tree.css">
<link rel="stylesheet" href="{{ site }}/static/font/font.css"> <link rel="stylesheet" href="{{ site }}/static/font/font.css">
<script type="module" src="{{ site }}/static/js/tree.js"></script>
<script type="module" src="{{ site }}/static/js/usability.js"></script> <script type="module" src="{{ site }}/static/js/usability.js"></script>
</head> </head>