import "@/setup/pwa"; import "core-js/stable"; import "./stores/__old/imports"; import "@/setup/ga"; import "@/assets/css/index.css"; import { StrictMode, Suspense, useCallback } from "react"; import type { ReactNode } from "react"; import { createRoot } from "react-dom/client"; import { HelmetProvider } from "react-helmet-async"; import { useTranslation } from "react-i18next"; import { BrowserRouter, HashRouter } from "react-router-dom"; import { useAsync, useAsyncFn } from "react-use"; import { Button } from "@/components/buttons/Button"; import { Icon, Icons } from "@/components/Icon"; import { Loading } from "@/components/layout/Loading"; import { useAuth } from "@/hooks/auth/useAuth"; import { useAuthRestore } from "@/hooks/auth/useAuthRestore"; import { useBackendUrl } from "@/hooks/auth/useBackendUrl"; import { ErrorBoundary } from "@/pages/errors/ErrorBoundary"; import { MigrationPart } from "@/pages/parts/migrations/MigrationPart"; import { LargeTextPart } from "@/pages/parts/util/LargeTextPart"; import App from "@/setup/App"; import { conf } from "@/setup/config"; import { useAuthStore } from "@/stores/auth"; import { BookmarkSyncer } from "@/stores/bookmarks/BookmarkSyncer"; import { changeAppLanguage, useLanguageStore } from "@/stores/language"; import { ProgressSyncer } from "@/stores/progress/ProgressSyncer"; import { SettingsSyncer } from "@/stores/subtitles/SettingsSyncer"; import { ThemeProvider } from "@/stores/theme"; import { extensionInfo, isExtensionActiveCached, } from "./backend/extension/messaging"; import { initializeChromecast } from "./setup/chromecast"; import { initializeOldStores } from "./stores/__old/migrations"; // initialize initializeChromecast(); function LoadingScreen(props: { type: "user" | "lazy" }) { const mapping = { user: "screens.loadingUser", lazy: "screens.loadingApp", }; const { t } = useTranslation(); return ( }> {t(mapping[props.type] ?? "unknown.translation")} ); } function ErrorScreen(props: { children: ReactNode; showResetButton?: boolean; showLogoutButton?: boolean; showReloadButton?: boolean; }) { const { t } = useTranslation(); const { logout } = useAuth(); const setBackendUrl = useAuthStore((s) => s.setBackendUrl); const resetBackend = useCallback(() => { setBackendUrl(null); // eslint-disable-next-line no-restricted-globals location.reload(); }, [setBackendUrl]); const logoutFromBackend = useCallback(() => { logout().then(() => { // eslint-disable-next-line no-restricted-globals location.reload(); }); }, [logout]); return ( } > {props.children} {props.showResetButton ? (
) : null} {props.showLogoutButton ? (
) : null} {props.showReloadButton ? (
) : null}
); } function AuthWrapper() { const status = useAuthRestore(); const backendUrl = conf().BACKEND_URL; const userBackendUrl = useBackendUrl(); const { t } = useTranslation(); const isCustomUrl = backendUrl !== userBackendUrl; if (status.loading) return ; if (status.error) return ( {t( isCustomUrl ? "screens.loadingUserError.textWithReset" : "screens.loadingUserError.text", )} ); return ; } function MigrationRunner() { const status = useAsync(async () => { changeAppLanguage(useLanguageStore.getState().language); await initializeOldStores(); }, []); const { t } = useTranslation(); if (status.loading) return ; if (status.error) return {t("screens.migration.failed")}; return ; } function TheRouter(props: { children: ReactNode }) { const normalRouter = conf().NORMAL_ROUTER; if (normalRouter) return {props.children}; return {props.children}; } // Checks if the extension is installed function ExtensionStatus() { const { t } = useTranslation(); const [state] = useAsyncFn(async () => { if (!isExtensionActiveCached) { return extensionInfo(); } }); if (state.loading) { return ; } if (state.error) { return {t("screens.loadingUserError.reload")}; } return null; } const container = document.getElementById("root"); const root = createRoot(container!); root.render( }> , );