2024-02-27 00:03:35 +01:00
|
|
|
import { Analytics } from "@vercel/analytics/react";
|
2024-03-03 03:18:27 +01:00
|
|
|
import { ReactElement, Suspense, lazy, useEffect, useState } from "react";
|
2024-03-03 21:28:56 +01:00
|
|
|
import { Helmet } from "react-helmet-async";
|
2024-03-03 03:18:27 +01:00
|
|
|
import { Trans, useTranslation } from "react-i18next";
|
2024-02-26 16:54:56 +01:00
|
|
|
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";
|
2024-03-03 03:18:27 +01:00
|
|
|
import { Button } from "@/components/buttons/Button";
|
2024-03-04 17:44:47 +01:00
|
|
|
import { Icons } from "@/components/Icon";
|
|
|
|
import { IconPill } from "@/components/layout/IconPill";
|
2024-03-03 03:18:27 +01:00
|
|
|
import { Navigation } from "@/components/layout/Navigation";
|
|
|
|
import { Title } from "@/components/text/Title";
|
|
|
|
import { Paragraph } from "@/components/utils/Text";
|
2024-02-26 16:54:56 +01:00
|
|
|
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";
|
2024-03-03 03:18:27 +01:00
|
|
|
import { ErrorContainer, ErrorLayout } from "@/pages/layouts/ErrorLayout";
|
2024-02-26 16:54:56 +01:00
|
|
|
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 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();
|
2024-03-03 03:18:27 +01:00
|
|
|
const { t } = useTranslation();
|
2024-03-05 03:51:02 +01:00
|
|
|
const maintenance = false;
|
2024-03-03 06:47:37 +01:00
|
|
|
const [showDowntime, setShowDowntime] = useState(maintenance);
|
2024-03-03 03:18:27 +01:00
|
|
|
|
|
|
|
const handleButtonClick = () => {
|
|
|
|
setShowDowntime(false);
|
|
|
|
};
|
|
|
|
|
|
|
|
useEffect(() => {
|
2024-03-03 06:03:34 +01:00
|
|
|
const sessionToken = sessionStorage.getItem("downtimeToken");
|
2024-03-05 05:24:33 +01:00
|
|
|
if (!sessionToken && maintenance) {
|
2024-03-03 03:18:27 +01:00
|
|
|
setShowDowntime(true);
|
2024-03-03 06:03:34 +01:00
|
|
|
sessionStorage.setItem("downtimeToken", "true");
|
2024-03-03 03:18:27 +01:00
|
|
|
}
|
2024-03-05 05:24:33 +01:00
|
|
|
}, [setShowDowntime, maintenance]);
|
2024-02-26 16:54:56 +01:00
|
|
|
|
|
|
|
return (
|
|
|
|
<Layout>
|
|
|
|
<LanguageProvider />
|
2024-03-03 03:18:27 +01:00
|
|
|
{!showDowntime && (
|
|
|
|
<Routes>
|
|
|
|
{/* functional routes */}
|
|
|
|
<Route path="/s/:query" element={<QuickSearch />} />
|
|
|
|
<Route path="/search/:type" element={<Navigate to="/browse" />} />
|
|
|
|
<Route path="/search/:type/:query?" element={<QueryView />} />
|
|
|
|
{/* pages */}
|
|
|
|
<Route
|
|
|
|
path="/media/:media"
|
|
|
|
element={
|
|
|
|
<LegacyUrlView>
|
|
|
|
<Suspense fallback={null}>
|
|
|
|
<PlayerView />
|
|
|
|
</Suspense>
|
|
|
|
</LegacyUrlView>
|
|
|
|
}
|
|
|
|
/>
|
|
|
|
<Route
|
|
|
|
path="/media/:media/:season/:episode"
|
|
|
|
element={
|
|
|
|
<LegacyUrlView>
|
|
|
|
<Suspense fallback={null}>
|
|
|
|
<PlayerView />
|
|
|
|
</Suspense>
|
|
|
|
</LegacyUrlView>
|
|
|
|
}
|
|
|
|
/>
|
|
|
|
<Route path="/browse/:query?" element={<HomePage />} />
|
|
|
|
<Route path="/" element={<HomePage />} />
|
|
|
|
<Route path="/register" element={<RegisterPage />} />
|
|
|
|
<Route path="/login" element={<LoginPage />} />
|
|
|
|
<Route path="/about" element={<AboutPage />} />
|
|
|
|
<Route path="/onboarding" element={<OnboardingPage />} />
|
|
|
|
<Route
|
|
|
|
path="/onboarding/extension"
|
|
|
|
element={<OnboardingExtensionPage />}
|
|
|
|
/>
|
|
|
|
<Route path="/onboarding/proxy" element={<OnboardingProxyPage />} />
|
|
|
|
{shouldHaveDmcaPage() ? (
|
|
|
|
<Route path="/dmca" element={<DmcaPage />} />
|
|
|
|
) : null}
|
|
|
|
{/* Settings page */}
|
|
|
|
<Route
|
|
|
|
path="/settings"
|
|
|
|
element={
|
2024-02-26 16:54:56 +01:00
|
|
|
<Suspense fallback={null}>
|
2024-03-03 03:18:27 +01:00
|
|
|
<SettingsPage />
|
2024-02-26 16:54:56 +01:00
|
|
|
</Suspense>
|
2024-03-03 03:18:27 +01:00
|
|
|
}
|
|
|
|
/>
|
|
|
|
{/* admin routes */}
|
|
|
|
<Route path="/admin" element={<AdminPage />} />
|
|
|
|
{/* other */}
|
|
|
|
<Route path="/dev" element={<DeveloperPage />} />
|
|
|
|
<Route path="/dev/video" element={<VideoTesterView />} />
|
|
|
|
{/* developer routes that can abuse workers are disabled in production */}
|
|
|
|
{process.env.NODE_ENV === "development" ? (
|
|
|
|
<Route path="/dev/test" element={<TestView />} />
|
|
|
|
) : null}
|
|
|
|
<Route path="*" element={<NotFoundPage />} />
|
|
|
|
</Routes>
|
|
|
|
)}
|
|
|
|
{showDowntime && (
|
|
|
|
<div className="relative flex flex-1 flex-col">
|
2024-03-04 20:10:16 +01:00
|
|
|
<Navigation />
|
2024-03-03 21:28:56 +01:00
|
|
|
<Helmet>
|
|
|
|
<title>{t("downtimeNotice.title")}</title>
|
|
|
|
</Helmet>
|
2024-03-03 03:18:27 +01:00
|
|
|
<div className="flex h-full flex-1 flex-col items-center justify-center p-5 text-center">
|
|
|
|
<ErrorLayout>
|
|
|
|
<ErrorContainer>
|
2024-03-04 17:44:47 +01:00
|
|
|
<IconPill icon={Icons.WARNING}>
|
|
|
|
{t("downtimeNotice.badge")}
|
|
|
|
</IconPill>
|
2024-03-03 03:18:27 +01:00
|
|
|
<Title>{t("downtimeNotice.title")}</Title>
|
|
|
|
<Paragraph>{t("downtimeNotice.message")}</Paragraph>
|
2024-03-05 00:05:21 +01:00
|
|
|
<Trans
|
|
|
|
i18nKey="downtimeNotice.timeFrame"
|
|
|
|
components={{
|
|
|
|
bold: (
|
|
|
|
<span
|
|
|
|
className="font-bold"
|
|
|
|
style={{ color: "#cfcfcf" }}
|
|
|
|
/>
|
|
|
|
),
|
|
|
|
}}
|
|
|
|
/>
|
2024-03-04 20:10:16 +01:00
|
|
|
<div className="flex gap-3">
|
|
|
|
<Button
|
|
|
|
onClick={handleButtonClick}
|
|
|
|
theme="purple"
|
|
|
|
className="mt-6"
|
|
|
|
>
|
|
|
|
{t("downtimeNotice.goHome")}
|
|
|
|
</Button>
|
|
|
|
</div>
|
2024-03-03 03:18:27 +01:00
|
|
|
</ErrorContainer>
|
|
|
|
</ErrorLayout>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
)}
|
2024-02-27 00:03:35 +01:00
|
|
|
<Analytics />
|
2024-02-26 16:54:56 +01:00
|
|
|
</Layout>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export default App;
|