elm-pages-v3-beta/index.js

257 lines
7.9 KiB
JavaScript
Raw Normal View History

const elmPagesVersion = require("./package.json").version;
2020-01-21 17:17:28 +03:00
let prefetchedPages;
let initialLocationHash;
let elmViewRendered = false;
let headTagsAdded = false;
2020-01-21 17:17:28 +03:00
module.exports = function pagesInit(
2020-01-21 17:17:28 +03:00
/** @type { mainElmModule: { init: any } } */ { mainElmModule }
) {
2020-01-21 17:17:28 +03:00
prefetchedPages = [window.location.pathname];
initialLocationHash = document.location.hash.replace(/^#/, "");
return new Promise(function (resolve, reject) {
2020-01-21 17:17:28 +03:00
document.addEventListener("DOMContentLoaded", _ => {
new MutationObserver(function () {
elmViewRendered = true;
if (headTagsAdded) {
document.dispatchEvent(new Event("prerender-trigger"));
}
}).observe(
document.body,
{ attributes: true, childList: true, subtree: true }
);
2020-01-21 17:17:28 +03:00
loadContentAndInitializeApp(mainElmModule).then(resolve, reject);
});
})
};
function loadContentAndInitializeApp(/** @type { init: any } */ mainElmModule) {
const isPrerendering = navigator.userAgent.indexOf("Headless") >= 0
const path = window.location.pathname.replace(/(\w)$/, "$1/")
return Promise.all([
getConfig(),
httpGet(`${window.location.origin}${path}content.json`)]).then(function (/** @type {[DevServerConfig?, JSON]} */[devServerConfig, contentJson]) {
const app = mainElmModule.init({
flags: {
secrets: null,
baseUrl: isPrerendering
? window.location.origin
: document.baseURI,
isPrerendering: isPrerendering,
isDevServer: !!module.hot,
isElmDebugMode: devServerConfig ? devServerConfig.elmDebugger : false,
contentJson,
}
});
app.ports.toJsPort.subscribe((
/** @type { { head: SeoTag[], allRoutes: string[] } } */ fromElm
) => {
appendTag({
type: 'head',
name: "meta",
attributes: [
["name", "generator"],
["content", `elm-pages v${elmPagesVersion}`]
]
});
window.allRoutes = fromElm.allRoutes.map(route => new URL(route, document.baseURI).href);
2020-01-21 17:17:28 +03:00
if (navigator.userAgent.indexOf("Headless") >= 0) {
fromElm.head.forEach(headTag => {
if (headTag.type === 'head') {
appendTag(headTag);
} else if (headTag.type === 'json-ld') {
appendJsonLdTag(headTag);
} else {
throw new Error(`Unknown tag type #{headTag}`)
}
});
headTagsAdded = true;
if (elmViewRendered) {
document.dispatchEvent(new Event("prerender-trigger"));
}
} else {
setupLinkPrefetching();
2020-04-14 01:12:16 +03:00
}
});
2020-01-21 17:17:28 +03:00
if (module.hot) {
// found this trick in the next.js source code
// https://github.com/zeit/next.js/blob/886037b1bac4bdbfeb689b032c1612750fb593f7/packages/next/client/dev/error-overlay/eventsource.js
// https://github.com/zeit/next.js/blob/886037b1bac4bdbfeb689b032c1612750fb593f7/packages/next/client/dev/dev-build-watcher.js
// more details about this API at https://www.html5rocks.com/en/tutorials/eventsource/basics/
let source = new window.EventSource('/__webpack_hmr')
source.addEventListener('message', (e) => {
// console.log('message!!!!!', e)
// console.log(e.data.action)
// console.log('ACTION', e.data.action);
// if (e.data && e.data.action)
2020-05-03 21:40:47 +03:00
if (event.data === '\uD83D\uDC93') {
// heartbeat
} else {
const obj = JSON.parse(event.data)
// console.log('obj.action', obj.action);
2020-05-03 21:40:47 +03:00
if (obj.action === 'building') {
2020-05-12 02:43:09 +03:00
app.ports.fromJsPort.send({ action: 'hmr-check' });
} else if (obj.action === 'built') {
2020-05-03 21:40:47 +03:00
let currentPath = window.location.pathname.replace(/(\w)$/, "$1/")
httpGet(`${window.location.origin}${currentPath}content.json`).then(function (/** @type JSON */ contentJson) {
2020-05-03 21:40:47 +03:00
app.ports.fromJsPort.send({ contentJson: contentJson });
});
}
}
})
}
return app
});
2020-01-21 17:17:28 +03:00
}
2020-02-01 19:21:02 +03:00
function setupLinkPrefetching() {
2020-01-21 17:17:28 +03:00
new MutationObserver(observeFirstRender).observe(document.body, {
attributes: true,
childList: true,
subtree: true
});
}
function loadNamedAnchor() {
if (initialLocationHash !== "") {
const namedAnchor = document.querySelector(
`[name=${initialLocationHash}]`
);
namedAnchor && namedAnchor.scrollIntoView();
}
2020-01-21 17:17:28 +03:00
}
2020-01-21 17:17:28 +03:00
function observeFirstRender(
/** @type {MutationRecord[]} */ mutationList,
/** @type {MutationObserver} */ firstRenderObserver
) {
loadNamedAnchor();
for (let mutation of mutationList) {
if (mutation.type === "childList") {
setupLinkPrefetchingHelp();
}
}
2020-01-21 17:17:28 +03:00
firstRenderObserver.disconnect();
new MutationObserver(observeUrlChanges).observe(document.body.children[0], {
attributes: true,
childList: false,
subtree: false
});
}
2020-01-21 17:17:28 +03:00
function observeUrlChanges(
/** @type {MutationRecord[]} */ mutationList,
/** @type {MutationObserver} */ _theObserver
) {
for (let mutation of mutationList) {
if (
mutation.type === "attributes" &&
mutation.attributeName === "data-url"
) {
setupLinkPrefetchingHelp();
}
}
2020-01-21 17:17:28 +03:00
}
function setupLinkPrefetchingHelp(
/** @type {MutationObserver} */ _mutationList,
/** @type {MutationObserver} */ _theObserver
) {
const links = document.querySelectorAll("a");
links.forEach(link => {
2020-02-01 19:21:02 +03:00
// console.log(link.pathname);
link.addEventListener("mouseenter", function (event) {
2019-09-17 19:40:35 +03:00
if (
2020-01-21 17:17:28 +03:00
event &&
event.target &&
event.target instanceof HTMLAnchorElement
2019-09-17 19:40:35 +03:00
) {
2020-01-21 17:17:28 +03:00
prefetchIfNeeded(event.target);
} else {
2020-02-01 19:21:02 +03:00
// console.log("Couldn't prefetch with event", event);
}
});
2020-01-21 17:17:28 +03:00
});
}
2020-01-21 17:17:28 +03:00
function prefetchIfNeeded(/** @type {HTMLAnchorElement} */ target) {
if (target.host === window.location.host) {
if (prefetchedPages.includes(target.pathname)) {
2020-02-01 19:21:02 +03:00
// console.log("Already preloaded", target.href);
// console.log("Not a known route, skipping preload", target.pathname);
} else if (!allRoutes.includes(new URL(target.pathname, document.baseURI).href)) {
2020-01-28 08:40:30 +03:00
}
else {
2020-01-21 17:17:28 +03:00
prefetchedPages.push(target.pathname);
2020-02-01 19:21:02 +03:00
// console.log("Preloading...", target.pathname);
2020-01-21 17:17:28 +03:00
const link = document.createElement("link");
link.setAttribute("as", "fetch");
link.setAttribute("rel", "prefetch");
link.setAttribute("href", origin + target.pathname + "/content.json");
document.head.appendChild(link);
}
}
2020-01-21 17:17:28 +03:00
}
/** @typedef {HeadTag | JsonLdTag} SeoTag */
/** @typedef {{ name: string; attributes: string[][]; type: 'head' }} HeadTag */
2020-01-21 17:17:28 +03:00
function appendTag(/** @type {HeadTag} */ tagDetails) {
const meta = document.createElement(tagDetails.name);
tagDetails.attributes.forEach(([name, value]) => {
meta.setAttribute(name, value);
});
document.getElementsByTagName("head")[0].appendChild(meta);
}
/** @typedef {{ contents: Object; type: 'json-ld' }} JsonLdTag */
function appendJsonLdTag(/** @type {JsonLdTag} */ tagDetails) {
let jsonLdScript = document.createElement('script');
jsonLdScript.type = "application/ld+json";
jsonLdScript.innerHTML = JSON.stringify(tagDetails.contents);
document.getElementsByTagName("head")[0].appendChild(jsonLdScript);
}
2020-01-21 17:17:28 +03:00
function httpGet(/** @type string */ theUrl) {
return new Promise(function (resolve, reject) {
2020-01-21 17:17:28 +03:00
const xmlHttp = new XMLHttpRequest();
xmlHttp.onreadystatechange = function () {
if (xmlHttp.readyState == 4 && xmlHttp.status == 200)
resolve(JSON.parse(xmlHttp.responseText));
}
2020-01-21 17:17:28 +03:00
xmlHttp.onerror = reject;
xmlHttp.open("GET", theUrl, true); // true for asynchronous
xmlHttp.send(null);
2020-01-21 17:17:28 +03:00
})
}
/**
* @returns { Promise<DevServerConfig?>}
*/
function getConfig() {
if (module.hot) {
return httpGet(`/elm-pages-dev-server-options`)
} else {
return Promise.resolve(null)
}
}
/** @typedef { { elmDebugger : boolean } } DevServerConfig */