import { Analytics } from "@vercel/analytics/react"; import { ReactElement, Suspense, lazy, useEffect, useState } from "react"; import { Helmet } from "react-helmet-async"; import { Trans, useTranslation } from "react-i18next"; import { lazyWithPreload } from "react-lazy-with-preload"; import { Navigate, Route, Routes, useLocation, useNavigate, useParams, } from "react-router-dom"; import { convertLegacyUrl, isLegacyUrl } from "@/backend/metadata/getmeta"; import { generateQuickSearchMediaUrl } from "@/backend/metadata/tmdb"; import { Button } from "@/components/buttons/Button"; import { Navigation } from "@/components/layout/Navigation"; import { Title } from "@/components/text/Title"; import { Paragraph } from "@/components/utils/Text"; import { useOnlineListener } from "@/hooks/usePing"; import { AboutPage } from "@/pages/About"; import { AdminPage } from "@/pages/admin/AdminPage"; import VideoTesterView from "@/pages/developer/VideoTesterView"; import { DmcaPage, shouldHaveDmcaPage } from "@/pages/Dmca"; import { NotFoundPage } from "@/pages/errors/NotFoundPage"; import { HomePage } from "@/pages/HomePage"; import { ErrorContainer, ErrorLayout } from "@/pages/layouts/ErrorLayout"; import { LoginPage } from "@/pages/Login"; import { OnboardingPage } from "@/pages/onboarding/Onboarding"; import { OnboardingExtensionPage } from "@/pages/onboarding/OnboardingExtension"; import { OnboardingProxyPage } from "@/pages/onboarding/OnboardingProxy"; import { RegisterPage } from "@/pages/Register"; import { Layout } from "@/setup/Layout"; import { useHistoryListener } from "@/stores/history"; import { LanguageProvider } from "@/stores/language"; const isDowntime = true; function checkDowntime() { const now = new Date(); const hour = now.getHours(); // Downtime between 12 AM - 1 PM // return isDowntime && hour >= 0 && hour < 1; return true; } const DeveloperPage = lazy(() => import("@/pages/DeveloperPage")); const TestView = lazy(() => import("@/pages/developer/TestView")); const PlayerView = lazyWithPreload(() => import("@/pages/PlayerView")); const SettingsPage = lazyWithPreload(() => import("@/pages/Settings")); PlayerView.preload(); SettingsPage.preload(); function LegacyUrlView({ children }: { children: ReactElement }) { const location = useLocation(); const navigate = useNavigate(); useEffect(() => { const url = location.pathname; if (!isLegacyUrl(url)) return; convertLegacyUrl(location.pathname).then((convertedUrl) => { navigate(convertedUrl ?? "/", { replace: true }); }); }, [location.pathname, navigate]); if (isLegacyUrl(location.pathname)) return null; return children; } function QuickSearch() { const { query } = useParams<{ query: string }>(); const navigate = useNavigate(); useEffect(() => { if (query) { generateQuickSearchMediaUrl(query).then((url) => { navigate(url ?? "/", { replace: true }); }); } else { navigate("/", { replace: true }); } }, [query, navigate]); return null; } function QueryView() { const { query } = useParams<{ query: string }>(); const navigate = useNavigate(); useEffect(() => { if (query) { navigate(`/browse/${query}`, { replace: true }); } else { navigate("/", { replace: true }); } }, [query, navigate]); return null; } function App() { useHistoryListener(); useOnlineListener(); const { t } = useTranslation(); const maintenance = false; const [showDowntime, setShowDowntime] = useState(maintenance); const handleButtonClick = () => { setShowDowntime(false); }; useEffect(() => { const sessionToken = sessionStorage.getItem("downtimeToken"); if (!sessionToken) { setShowDowntime(true); sessionStorage.setItem("downtimeToken", "true"); } }, [setShowDowntime]); return ( {!showDowntime && ( {/* functional routes */} } /> } /> } /> {/* pages */} } /> } /> } /> } /> } /> } /> } /> } /> } /> } /> {shouldHaveDmcaPage() ? ( } /> ) : null} {/* Settings page */} } /> {/* admin routes */} } /> {/* other */} } /> } /> {/* developer routes that can abuse workers are disabled in production */} {process.env.NODE_ENV === "development" ? ( } /> ) : null} } /> )} {showDowntime && (
{t("downtimeNotice.title")}
{t("downtimeNotice.title")} {t("downtimeNotice.message")} , }} />
)}
); } export default App;