138 lines
4.5 KiB
TypeScript
138 lines
4.5 KiB
TypeScript
import * as loader from "./loader/loader";
|
|
import {Stage} from "./loader/loader";
|
|
import {getUrlParameter} from "./loader/utils";
|
|
|
|
let overlay: HTMLDivElement;
|
|
let setupContainer: HTMLDivElement;
|
|
let idleContainer: HTMLDivElement;
|
|
let idleSteamContainer: HTMLDivElement;
|
|
let loaderStageContainer: HTMLDivElement;
|
|
|
|
let finalizing = false;
|
|
let initializeTimestamp;
|
|
|
|
let verbose = false;
|
|
let apngSupport = undefined;
|
|
|
|
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")]) {
|
|
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";
|
|
|
|
idleContainer = overlay.getElementsByClassName("idle")[0] as HTMLDivElement;
|
|
if(!idleContainer)
|
|
throw "missing idle container";
|
|
|
|
idleSteamContainer = idleContainer.getElementsByClassName("steam")[0] as HTMLDivElement;
|
|
if(!idleSteamContainer)
|
|
throw "missing idle steam container";
|
|
|
|
loaderStageContainer = overlay.getElementsByClassName("loader-stage")[0] as HTMLDivElement;
|
|
if(!loaderStageContainer)
|
|
throw "missing loader stage container";
|
|
|
|
setupContainer.onanimationend = setupAnimationFinished;
|
|
idleSteamContainer.onanimationiteration = idleSteamAnimationLooped;
|
|
overlay.onanimationend = overlayAnimationFinished;
|
|
}
|
|
|
|
export async function initialize() {
|
|
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");
|
|
}
|
|
|
|
initializeTimestamp = Date.now();
|
|
return true;
|
|
}
|
|
|
|
export function abort() {
|
|
overlay?.remove();
|
|
}
|
|
|
|
export function finalize() {
|
|
if(getUrlParameter("loader-abort") === "1") {
|
|
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");
|
|
idleContainer.classList.add("visible");
|
|
}
|
|
|
|
function idleSteamAnimationLooped() {
|
|
verbose && console.log("Idle animation looped. Should finalize: %o", finalizing);
|
|
if(!finalizing)
|
|
return;
|
|
|
|
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();
|
|
} |