184 lines
6.0 KiB
TypeScript
184 lines
6.0 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 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();
|
|
} |