e72ec608f4
#95 #95 #95 #95 #95 Merge remote-tracking branch 'origin/main' into feature/095 #95 #95 #95 Co-authored-by: miteruzo <miteruzo@naver.com> Reviewed-on: #311
149 lines
5.1 KiB
TypeScript
149 lines
5.1 KiB
TypeScript
import { AnimatePresence, LayoutGroup, motion } from 'framer-motion'
|
|
import { useEffect, useState } from 'react'
|
|
import { BrowserRouter,
|
|
Navigate,
|
|
Route,
|
|
Routes,
|
|
useLocation } from 'react-router-dom'
|
|
|
|
import RouteBlockerOverlay from '@/components/RouteBlockerOverlay'
|
|
import TopNav from '@/components/TopNav'
|
|
import { Toaster } from '@/components/ui/toaster'
|
|
import { apiPost, isApiError } from '@/lib/api'
|
|
import MaterialBasePage from '@/pages/materials/MaterialBasePage'
|
|
import MaterialDetailPage from '@/pages/materials/MaterialDetailPage'
|
|
import MaterialListPage from '@/pages/materials/MaterialListPage'
|
|
import MaterialNewPage from '@/pages/materials/MaterialNewPage'
|
|
// import MaterialSearchPage from '@/pages/materials/MaterialSearchPage'
|
|
import MorePage from '@/pages/MorePage'
|
|
import NicoTagListPage from '@/pages/tags/NicoTagListPage'
|
|
import NotFound from '@/pages/NotFound'
|
|
import TOSPage from '@/pages/TOSPage.mdx'
|
|
import PostDetailPage from '@/pages/posts/PostDetailPage'
|
|
import PostHistoryPage from '@/pages/posts/PostHistoryPage'
|
|
import PostListPage from '@/pages/posts/PostListPage'
|
|
import PostNewPage from '@/pages/posts/PostNewPage'
|
|
import PostSearchPage from '@/pages/posts/PostSearchPage'
|
|
import ServiceUnavailable from '@/pages/ServiceUnavailable'
|
|
import SettingPage from '@/pages/users/SettingPage'
|
|
import TagListPage from '@/pages/tags/TagListPage'
|
|
import TheatreDetailPage from '@/pages/theatres/TheatreDetailPage'
|
|
import WikiDetailPage from '@/pages/wiki/WikiDetailPage'
|
|
import WikiDiffPage from '@/pages/wiki/WikiDiffPage'
|
|
import WikiEditPage from '@/pages/wiki/WikiEditPage'
|
|
import WikiHistoryPage from '@/pages/wiki/WikiHistoryPage'
|
|
import WikiNewPage from '@/pages/wiki/WikiNewPage'
|
|
import WikiSearchPage from '@/pages/wiki/WikiSearchPage'
|
|
|
|
import type { Dispatch, FC, SetStateAction } from 'react'
|
|
|
|
import type { User } from '@/types'
|
|
|
|
|
|
const RouteTransitionWrapper = ({ user, setUser }: {
|
|
user: User | null
|
|
setUser: Dispatch<SetStateAction<User | null>> }) => {
|
|
const location = useLocation ()
|
|
|
|
return (
|
|
<AnimatePresence mode="wait">
|
|
<Routes location={location}>
|
|
<Route path="/" element={<Navigate to="/posts" replace/>}/>
|
|
<Route path="/posts" element={<PostListPage/>}/>
|
|
<Route path="/posts/new" element={<PostNewPage user={user}/>}/>
|
|
<Route path="/posts/search" element={<PostSearchPage/>}/>
|
|
<Route path="/posts/:id" element={<PostDetailRoute user={user}/>}/>
|
|
<Route path="/posts/changes" element={<PostHistoryPage/>}/>
|
|
<Route path="/tags" element={<TagListPage/>}/>
|
|
<Route path="/tags/nico" element={<NicoTagListPage user={user}/>}/>
|
|
<Route path="/theatres/:id" element={<TheatreDetailPage/>}/>
|
|
<Route path="/materials" element={<MaterialBasePage/>}>
|
|
<Route index element={<MaterialListPage/>}/>
|
|
<Route path="new" element={<MaterialNewPage/>}/>
|
|
<Route path=":id" element ={<MaterialDetailPage/>}/>
|
|
</Route>
|
|
{/* <Route path="/materials/search" element={<MaterialSearchPage/>}/> */}
|
|
<Route path="/wiki" element={<WikiSearchPage/>}/>
|
|
<Route path="/wiki/:title" element={<WikiDetailPage/>}/>
|
|
<Route path="/wiki/new" element={<WikiNewPage user={user}/>}/>
|
|
<Route path="/wiki/:id/edit" element={<WikiEditPage user={user}/>}/>
|
|
<Route path="/wiki/:id/diff" element={<WikiDiffPage/>}/>
|
|
<Route path="/wiki/changes" element={<WikiHistoryPage/>}/>
|
|
<Route path="/users/settings" element={<SettingPage user={user} setUser={setUser}/>}/>
|
|
<Route path="/settings" element={<Navigate to="/users/settings" replace/>}/>
|
|
<Route path="/tos" element={<TOSPage/>}/>
|
|
<Route path="/more" element={<MorePage/>}/>
|
|
<Route path="*" element={<NotFound/>}/>
|
|
</Routes>
|
|
</AnimatePresence>)
|
|
}
|
|
|
|
|
|
const PostDetailRoute = ({ user }: { user: User | null }) => {
|
|
const location = useLocation ()
|
|
const key = location.pathname
|
|
return <PostDetailPage key={key} user={user}/>
|
|
}
|
|
|
|
|
|
export default (() => {
|
|
const [user, setUser] = useState<User | null> (null)
|
|
const [status, setStatus] = useState (200)
|
|
|
|
useEffect (() => {
|
|
const createUser = async () => {
|
|
const data = await apiPost<{ code: string; user: User }> ('/users')
|
|
if (data.code)
|
|
{
|
|
localStorage.setItem ('user_code', data.code)
|
|
setUser (data.user)
|
|
}
|
|
}
|
|
|
|
const code = localStorage.getItem ('user_code')
|
|
if (code)
|
|
{
|
|
void (async () => {
|
|
try
|
|
{
|
|
const data = await apiPost<{ valid: boolean; user: User }> ('/users/verify', { code })
|
|
if (data.valid)
|
|
setUser (data.user)
|
|
else
|
|
await createUser ()
|
|
}
|
|
catch (err)
|
|
{
|
|
if (isApiError (err))
|
|
setStatus (err.response?.status ?? 200)
|
|
}
|
|
}) ()
|
|
}
|
|
else
|
|
createUser ()
|
|
}, [])
|
|
|
|
switch (status)
|
|
{
|
|
case 503:
|
|
return <ServiceUnavailable/>
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<RouteBlockerOverlay/>
|
|
<BrowserRouter>
|
|
<LayoutGroup>
|
|
<motion.div
|
|
layout="position"
|
|
transition={{ layout: { duration: .2, ease: 'easeOut' } }}
|
|
className="flex flex-col h-dvh w-full overflow-y-hidden">
|
|
<TopNav user={user}/>
|
|
<RouteTransitionWrapper user={user} setUser={setUser}/>
|
|
</motion.div>
|
|
</LayoutGroup>
|
|
<Toaster/>
|
|
</BrowserRouter>
|
|
</>)
|
|
}) satisfies FC
|