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