mirror of
https://github.com/dillonkearns/elm-pages-v3-beta.git
synced 2025-01-07 23:17:35 +03:00
113 lines
2.9 KiB
JavaScript
113 lines
2.9 KiB
JavaScript
import userInit from "/index.js";
|
|
|
|
let prefetchedPages;
|
|
let initialLocationHash;
|
|
/**
|
|
* @returns
|
|
*/
|
|
function loadContentAndInitializeApp() {
|
|
let path = window.location.pathname.replace(/(\w)$/, "$1/");
|
|
if (!path.endsWith("/")) {
|
|
path = path + "/";
|
|
}
|
|
const app = Elm.Main.init({
|
|
flags: {
|
|
secrets: null,
|
|
isPrerendering: false,
|
|
isDevServer: false,
|
|
isElmDebugMode: false,
|
|
contentJson: {},
|
|
pageDataBase64: document.getElementById("__ELM_PAGES_BYTES_DATA__")
|
|
.innerHTML,
|
|
userFlags: userInit.flags(),
|
|
},
|
|
});
|
|
|
|
app.ports.toJsPort.subscribe((fromElm) => {
|
|
loadNamedAnchor();
|
|
});
|
|
|
|
app.ports.elmPagesReloadData &&
|
|
app.ports.elmPagesReloadData.subscribe((data) => {
|
|
console.log("RELOAD DATA PORT!");
|
|
app.ports.fromJsPort.send({ tag: "Reload", data });
|
|
});
|
|
|
|
return app;
|
|
}
|
|
|
|
function loadNamedAnchor() {
|
|
if (initialLocationHash !== "") {
|
|
const namedAnchor = document.querySelector(`[name=${initialLocationHash}]`);
|
|
namedAnchor && namedAnchor.scrollIntoView();
|
|
}
|
|
}
|
|
|
|
function prefetchIfNeeded(/** @type {HTMLAnchorElement} */ target) {
|
|
if (
|
|
target.host === window.location.host &&
|
|
!prefetchedPages.includes(target.pathname)
|
|
) {
|
|
prefetchedPages.push(target.pathname);
|
|
const link = document.createElement("link");
|
|
link.setAttribute("as", "fetch");
|
|
|
|
link.setAttribute("rel", "prefetch");
|
|
link.setAttribute("href", origin + target.pathname + "/content.dat");
|
|
document.head.appendChild(link);
|
|
}
|
|
}
|
|
|
|
export function setup() {
|
|
prefetchedPages = [window.location.pathname];
|
|
initialLocationHash = document.location.hash.replace(/^#/, "");
|
|
const appPromise = new Promise(function (resolve, reject) {
|
|
document.addEventListener("DOMContentLoaded", (_) => {
|
|
resolve(loadContentAndInitializeApp());
|
|
});
|
|
});
|
|
userInit.load(appPromise);
|
|
|
|
if (typeof connect === "function") {
|
|
connect(function (bytesData) {
|
|
appPromise.then((app) => {
|
|
app.ports.hotReloadData.send(bytesData);
|
|
});
|
|
});
|
|
}
|
|
|
|
/** @param {MouseEvent} event */
|
|
const trigger_prefetch = (event) => {
|
|
const a = find_anchor(/** @type {Node} */ (event.target));
|
|
if (a && a.href && a.hasAttribute("elm-pages:prefetch")) {
|
|
prefetchIfNeeded(a);
|
|
}
|
|
};
|
|
|
|
/** @type {NodeJS.Timeout} */
|
|
let mousemove_timeout;
|
|
|
|
/** @param {MouseEvent} event */
|
|
const handle_mousemove = (event) => {
|
|
clearTimeout(mousemove_timeout);
|
|
mousemove_timeout = setTimeout(() => {
|
|
trigger_prefetch(event);
|
|
}, 20);
|
|
};
|
|
|
|
addEventListener("touchstart", trigger_prefetch);
|
|
addEventListener("mousemove", handle_mousemove);
|
|
}
|
|
|
|
/**
|
|
* @param {Node} node
|
|
// * @rturns {HTMLAnchorElement | SVGAElement}
|
|
* @returns {HTMLAnchorElement}
|
|
*/
|
|
function find_anchor(node) {
|
|
while (node && node.nodeName.toUpperCase() !== "A") node = node.parentNode; // SVG <a> elements have a lowercase name
|
|
return /** @type {HTMLAnchorElement} */ (node);
|
|
}
|
|
|
|
setup();
|