TeaWeb/loader/app/animation.ts

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();
}