2023-08-22 19:23:31 +02:00
|
|
|
const branchStateKey = "treehouse.openBranches";
|
|
|
|
let branchState = JSON.parse(localStorage.getItem(branchStateKey)) || {};
|
|
|
|
|
|
|
|
function saveBranchIsOpen(branchID, state) {
|
|
|
|
branchState[branchID] = state;
|
|
|
|
localStorage.setItem(branchStateKey, JSON.stringify(branchState));
|
|
|
|
}
|
|
|
|
|
|
|
|
function branchIsOpen(branchID) {
|
|
|
|
return branchState[branchID];
|
|
|
|
}
|
|
|
|
|
|
|
|
class Branch extends HTMLLIElement {
|
2023-08-20 15:05:59 +02:00
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
|
|
|
|
this.details = this.childNodes[0];
|
|
|
|
this.innerUL = this.details.childNodes[1];
|
|
|
|
|
2023-08-22 19:23:31 +02:00
|
|
|
let isOpen = branchIsOpen(this.id);
|
|
|
|
if (isOpen !== undefined) {
|
|
|
|
this.details.open = isOpen;
|
|
|
|
}
|
|
|
|
this.details.addEventListener("toggle", _ => {
|
|
|
|
saveBranchIsOpen(this.id, this.details.open);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
customElements.define("th-b", Branch, { extends: "li" });
|
|
|
|
|
|
|
|
class LinkedBranch extends Branch {
|
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
|
|
|
|
this.linkedTree = this.getAttribute("data-th-link");
|
|
|
|
|
|
|
|
this.loadingState = "notloaded";
|
2023-08-20 15:05:59 +02:00
|
|
|
|
|
|
|
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
|
2023-08-22 19:23:31 +02:00
|
|
|
// correctly, as Branch saves the state in localStorage. Having an expanded-by-default
|
|
|
|
// linked block can be useful in development.
|
2023-08-20 15:05:59 +02:00
|
|
|
if (this.details.open) {
|
|
|
|
this.loadTree();
|
|
|
|
}
|
|
|
|
|
2023-08-22 19:23:31 +02:00
|
|
|
this.details.addEventListener("toggle", _ => {
|
2023-08-20 15:05:59 +02:00
|
|
|
if (this.details.open) {
|
|
|
|
this.loadTree();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
loadTree() {
|
2023-08-22 19:23:31 +02:00
|
|
|
if (this.loadingState == "notloaded") {
|
|
|
|
this.loadingState = "loading";
|
2023-08-20 15:05:59 +02:00
|
|
|
|
|
|
|
fetch(`/${this.linkedTree}.html`)
|
2023-08-20 18:37:00 +02:00
|
|
|
.then(response => {
|
|
|
|
if (response.status == 404) {
|
|
|
|
throw `Hmm, seems like the tree "${this.linkedTree}" does not exist.`;
|
|
|
|
}
|
|
|
|
return response.text();
|
|
|
|
})
|
2023-08-20 15:05:59 +02:00
|
|
|
.then(text => {
|
|
|
|
let parser = new DOMParser();
|
|
|
|
let linkedDocument = parser.parseFromString(text, "text/html");
|
|
|
|
let main = linkedDocument.getElementsByTagName("main")[0];
|
2023-08-22 19:23:31 +02:00
|
|
|
let ul = main.getElementsByTagName("ul")[0];
|
2023-08-20 15:05:59 +02:00
|
|
|
|
|
|
|
this.loadingText.remove();
|
2023-08-22 19:23:31 +02:00
|
|
|
this.innerUL.innerHTML = ul.innerHTML;
|
2023-08-20 15:05:59 +02:00
|
|
|
|
2023-08-22 19:23:31 +02:00
|
|
|
this.loadingState = "loaded";
|
2023-08-20 15:05:59 +02:00
|
|
|
})
|
|
|
|
.catch(error => {
|
|
|
|
this.loadingText.innerText = error.toString();
|
2023-08-22 19:23:31 +02:00
|
|
|
this.loadingState = "error";
|
2023-08-20 15:05:59 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-22 19:23:31 +02:00
|
|
|
customElements.define("th-b-linked", LinkedBranch, { extends: "li" });
|
2023-08-21 21:12:06 +02:00
|
|
|
|
|
|
|
function expandDetailsRecursively(element) {
|
|
|
|
while (element && element.tagName != "MAIN") {
|
|
|
|
if (element.tagName == "DETAILS") {
|
|
|
|
element.open = true;
|
|
|
|
}
|
|
|
|
element = element.parentElement;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// When you click on a link, and the destination is within a <details> that is not expanded,
|
|
|
|
// expand the <details> recursively.
|
|
|
|
window.addEventListener("popstate", _ => {
|
|
|
|
let element = document.getElementById(window.location.hash.substring(1));
|
|
|
|
if (element !== undefined) {
|
|
|
|
// If the element is already loaded on the page, we're good.
|
|
|
|
expandDetailsRecursively(element);
|
|
|
|
window.location.hash = window.location.hash;
|
|
|
|
}
|
|
|
|
})
|