import { state } from "./state"; import { showControls } from "./controls"; import { updateSubtitleOptions } from "./subtitles"; import { updateQualityOptions } from "./quality"; import { safeLocalStorage } from "./storage"; import { streamUrlForMode } from "./source"; import { loadVideoSource } from "./video"; import type { ModeSource } from "./types"; const isRecord = (v: unknown): v is Record => typeof v === "object" && v !== null && !Array.isArray(v); const isStringArray = (v: unknown): v is string[] => Array.isArray(v) && v.every((item) => typeof item === "string"); const isSubtitleItemArray = (v: unknown): v is { lang: string; token: string }[] => Array.isArray(v) && v.every( (item) => isRecord(item) && typeof item.lang === "string" && typeof item.token === "string", ); const parseModeSources = (v: unknown): Record => { if (!isRecord(v)) return {}; const out: Record = {}; for (const [key, value] of Object.entries(v)) { if (!isRecord(value)) continue; if (typeof value.token !== "string" || value.token === "") continue; const subtitles = value.subtitles == null ? [] : value.subtitles; if (!isSubtitleItemArray(subtitles)) continue; const qualities = value.qualities; out[key] = { token: value.token, type: typeof value.type === "string" ? value.type : undefined, subtitles, qualities: isStringArray(qualities) ? qualities : undefined, }; } return out; }; const alternateModeFor = (mode: string): "sub" | "dub" | null => { if (mode === "sub") return "dub"; if (mode === "dub") return "sub"; return null; }; export const hydrateAlternateMode = async (signal?: AbortSignal): Promise => { const alternateMode = alternateModeFor(state.currentMode); if (!alternateMode) return; if (state.modeSources[alternateMode]?.token) return; try { const res = await fetch( `/api/watch/episode/${state.malID}/${encodeURIComponent(state.currentEpisode)}?mode=${encodeURIComponent(alternateMode)}`, { signal }, ); if (!res.ok) return; const data: unknown = await res.json(); if (!isRecord(data)) return; const sources = parseModeSources(data.mode_sources); const alternateSource = sources[alternateMode]; if (!alternateSource?.token) return; state.modeSources = { ...state.modeSources, [alternateMode]: alternateSource, }; updateSubtitleOptions(); updateQualityOptions(); updateModeButtons(); } catch (error: unknown) { if (error instanceof DOMException && error.name === "AbortError") return; } }; /** * Switches between sub/dub mode. * Saves preference to localStorage, reloads video src. */ export const switchMode = (mode: string): void => { if (!state.availableModes.includes(mode) || mode === state.currentMode) return; state.currentMode = mode; safeLocalStorage.setItem("player-audio-mode", mode); const qualitySelect = state.container.querySelector( "[data-quality-select]", ) as HTMLSelectElement | null; const url = streamUrlForMode(mode, qualitySelect?.value); loadVideoSource(url, state.modeSources[mode]?.type); // Fallback: if the media element doesn't actually switch sources (some browsers can get "stuck"), // reload the page with the desired mode and resume time via sessionStorage. if (url) { const expectedToken = state.modeSources[mode]?.token; const expectedMode = mode; const resumeSeconds = state.video.currentTime; window.setTimeout(() => { if (!expectedToken) return; const currentSrc = state.video.currentSrc || state.video.src || ""; if (currentSrc.includes(`token=${encodeURIComponent(expectedToken)}`)) return; try { sessionStorage.setItem("mal:resume-after-mode-switch", String(resumeSeconds)); const next = new URL(window.location.href); next.searchParams.set("mode", expectedMode); window.location.href = next.toString(); } catch { // no-op } }, 800); } updateSubtitleOptions(); updateQualityOptions(); updateModeButtons(); }; /** * Updates dub/sub button styling based on current mode. * Disables unavailable modes. */ export const updateModeButtons = (): void => { const dub = state.container.querySelector("[data-mode-dub]") as HTMLButtonElement | null; const sub = state.container.querySelector("[data-mode-sub]") as HTMLButtonElement | null; const m = state.currentMode; dub?.classList.toggle("text-accent", m === "dub"); dub?.classList.toggle("text-foreground", m !== "dub"); dub?.classList.toggle("opacity-50", !state.availableModes.includes("dub")); dub?.classList.toggle("cursor-not-allowed", !state.availableModes.includes("dub")); if (dub) { dub.disabled = !state.availableModes.includes("dub"); } sub?.classList.toggle("text-accent", m === "sub"); sub?.classList.toggle("text-foreground", m !== "sub"); sub?.classList.toggle("opacity-50", !state.availableModes.includes("sub")); sub?.classList.toggle("cursor-not-allowed", !state.availableModes.includes("sub")); if (sub) { sub.disabled = !state.availableModes.includes("sub"); } }; /** * Binds click handlers for mode buttons and autoplay toggle. */ export const setupMode = (): void => { const dub = state.container.querySelector("[data-mode-dub]") as HTMLButtonElement | null; const sub = state.container.querySelector("[data-mode-sub]") as HTMLButtonElement | null; dub?.addEventListener("click", () => { if (state.availableModes.includes("dub")) { switchMode("dub"); showControls(); } }); sub?.addEventListener("click", () => { if (state.availableModes.includes("sub")) { switchMode("sub"); showControls(); } }); const autoplayBtn = document.querySelector("[data-autoplay]") as HTMLInputElement | null; autoplayBtn?.addEventListener("change", (e) => { safeLocalStorage.setItem( "mal:autoplay-enabled", (e.target as HTMLInputElement).checked ? "true" : "false", ); showControls(); }); };