1
0
Fork 0
mirror of https://github.com/sussy-code/smov.git synced 2025-01-08 17:27:40 +01:00
smov/src/setup/App.tsx

225 lines
7.4 KiB
TypeScript
Raw Normal View History

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";
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;
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">
<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>
<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>
<Trans
i18nKey="downtimeNotice.timeFrame"
components={{
bold: (
<span
className="font-bold"
style={{ color: "#cfcfcf" }}
/>
),
}}
/>
<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;