import * as loader from "./loader/loader"; import {Stage} from "./loader/loader"; import {getUrlParameter} from "./loader/utils"; let overlay: HTMLDivElement; let setupContainer: HTMLDivElement; let idleNormalContainer: HTMLDivElement; let idleNormalSteamContainer: HTMLDivElement; let idleHalloweenContainer: HTMLDivElement; let loaderStageContainer: HTMLDivElement; let finalizing = false; let initializeTimestamp; let verbose = false; let apngSupport = undefined; let loopInterval: number; let animationType: "normal" | "halloween"; async function detectAPNGSupport() { const image = new Image(); const ctx = document.createElement("canvas").getContext("2d"); // frame 1 (skipped on apng-supporting browsers): [0, 0, 0, 255] // frame 2: [0, 0, 0, 0] image.src = ""; await new Promise(resolve => image.onload = resolve); ctx.drawImage(image, 0, 0); apngSupport = ctx.getImageData(0, 0, 1, 1).data[3] === 0; } function initializeElements() { overlay = document.getElementById("loader-overlay") as HTMLDivElement; if(!overlay) { throw "missing loader overlay"; } for(const lazyImage of [...overlay.getElementsByTagName("lazy-img")]) { if(lazyImage.hasAttribute("x-animation-depend") && lazyImage.getAttribute("x-animation-depend") !== animationType) { lazyImage.remove(); continue; } const image = document.createElement("img"); image.alt = lazyImage.getAttribute("alt"); image.src = lazyImage.getAttribute(apngSupport ? "src-apng" : "src-gif") || lazyImage.getAttribute("src"); image.className = lazyImage.className; image.draggable = false; lazyImage.replaceWith(image); } setupContainer = overlay.getElementsByClassName("setup")[0] as HTMLDivElement; if(!setupContainer) { throw "missing setup container"; } idleNormalContainer = overlay.getElementsByClassName("idle animation-normal")[0] as HTMLDivElement; if(!idleNormalContainer) { throw "missing normal idle container"; } idleHalloweenContainer = overlay.getElementsByClassName("idle animation-halloween")[0] as HTMLDivElement; if(!idleHalloweenContainer) { throw "missing halloween idle container"; } idleNormalSteamContainer = idleNormalContainer.getElementsByClassName("steam")[0] as HTMLDivElement; if(!idleNormalSteamContainer) { throw "missing normal idle steam container"; } loaderStageContainer = overlay.getElementsByClassName("loader-stage")[0] as HTMLDivElement; if(!loaderStageContainer) { throw "missing loader stage container"; } setupContainer.onanimationend = setupAnimationFinished; idleHalloweenContainer.onanimationiteration = idleSteamAnimationLooped; idleNormalSteamContainer.onanimationiteration = idleSteamAnimationLooped; overlay.onanimationend = overlayAnimationFinished; } export async function initialize(customLoadingAnimations: boolean) { if(customLoadingAnimations) { const now = new Date(); /* Note, the months start with zero */ if(now.getMonth() === 9) { animationType = "halloween"; } else if(now.getMonth() === 10 && now.getDate() <= 5) { animationType = "halloween"; } } if(!animationType) { animationType = "normal"; } await detectAPNGSupport(); try { initializeElements(); } catch (error) { console.error("Failed to setup animations: %o", error); loader.critical_error("Animation setup failed", error); return false; } StageNames[Stage.SETUP] = "starting app"; StageNames[Stage.TEMPLATES] = "loading templates"; StageNames[Stage.STYLE] = "loading styles"; StageNames[Stage.JAVASCRIPT] = "loading app"; StageNames[Stage.JAVASCRIPT_INITIALIZING] = "initializing"; StageNames[Stage.FINALIZING] = "rounding up"; StageNames[Stage.LOADED] = "starting app"; overlay.classList.add("initialized"); if(parseInt(getUrlParameter("animation-short")) === 1) { setupAnimationFinished(); } else { setupContainer.classList.add("visible", animationType); } initializeTimestamp = Date.now(); return true; } export function abort() { overlay?.remove(); clearInterval(loopInterval); loopInterval = 0; } export function finalize(abortAnimation: boolean) { if(abortAnimation) { abort(); } else { finalizing = true; if(loaderStageContainer) { loaderStageContainer.innerText = "app loaded successfully (" + (Date.now() - initializeTimestamp) + "ms)"; } } } const StageNames = {}; export function updateState(state: Stage, tasks: string[]) { if(loaderStageContainer) loaderStageContainer.innerText = StageNames[state] + (tasks.length === 1 ? " (task: " + tasks[0] + ")" : " (tasks: " + tasks.join(",") + ")"); } function setupAnimationFinished() { verbose && console.log("Entering idle animation"); setupContainer.classList.remove("visible"); if(animationType === "halloween") { loopInterval = setInterval(() => idleSteamAnimationLooped(), 1000); idleHalloweenContainer.classList.add("visible"); } else { idleNormalContainer.classList.add("visible"); } } function idleSteamAnimationLooped() { verbose && console.log("Idle animation looped. Should finalize: %o", finalizing); if(!finalizing) { return; } clearInterval(loopInterval); loopInterval = 0; overlay.classList.add("finishing"); } function overlayAnimationFinished(event: AnimationEvent) { /* the text animation is the last one */ if(event.animationName !== "swipe-out-text") { return; } verbose && console.log("Animation finished"); overlay.remove(); }