48f823a7c8
Merge branch 'main' into feature/308 #308 #308 #308 #308 Co-authored-by: miteruzo <miteruzo@naver.com> Reviewed-on: #315
162 lines
4.8 KiB
TypeScript
162 lines
4.8 KiB
TypeScript
import { useMutation, useQuery, useQueryClient } from '@tanstack/react-query'
|
|
import { motion } from 'framer-motion'
|
|
import { useEffect, useRef, useState } from 'react'
|
|
import { Helmet } from 'react-helmet-async'
|
|
import { useParams } from 'react-router-dom'
|
|
|
|
import PostEditForm from '@/components/PostEditForm'
|
|
import PostEmbed from '@/components/PostEmbed'
|
|
import PostList from '@/components/PostList'
|
|
import TagDetailSidebar from '@/components/TagDetailSidebar'
|
|
import TabGroup, { Tab } from '@/components/common/TabGroup'
|
|
import MainArea from '@/components/layout/MainArea'
|
|
import { Button } from '@/components/ui/button'
|
|
import { toast } from '@/components/ui/use-toast'
|
|
import { SITE_TITLE } from '@/config'
|
|
import { fetchPost, toggleViewedFlg } from '@/lib/posts'
|
|
import { postsKeys, tagsKeys } from '@/lib/queryKeys'
|
|
import { cn } from '@/lib/utils'
|
|
import NotFound from '@/pages/NotFound'
|
|
import ServiceUnavailable from '@/pages/ServiceUnavailable'
|
|
|
|
import type { FC } from 'react'
|
|
|
|
import type { NiconicoViewerHandle, User } from '@/types'
|
|
|
|
type Props = { user: User | null }
|
|
|
|
|
|
export default (({ user }: Props) => {
|
|
const { id } = useParams ()
|
|
const postId = String (id ?? '')
|
|
const postKey = postsKeys.show (postId)
|
|
|
|
const { data: post, isError: errorFlg, error } = useQuery ({
|
|
enabled: Boolean (id),
|
|
queryKey: postKey,
|
|
queryFn: () => fetchPost (postId) })
|
|
|
|
const qc = useQueryClient ()
|
|
|
|
const embedRef = useRef<NiconicoViewerHandle> (null)
|
|
|
|
const [status, setStatus] = useState (200)
|
|
|
|
const changeViewedFlg = useMutation ({
|
|
mutationFn: async () => {
|
|
const cur = qc.getQueryData<any> (postKey)
|
|
const next = !(cur?.viewed)
|
|
await toggleViewedFlg (postId, next)
|
|
return next
|
|
},
|
|
onMutate: async () => {
|
|
await qc.cancelQueries ({ queryKey: postKey })
|
|
const prev = qc.getQueryData<any> (postKey)
|
|
qc.setQueryData (postKey,
|
|
(cur: any) => cur ? { ...cur, viewed: !(cur.viewed) } : cur)
|
|
return { prev }
|
|
},
|
|
onError: (...[, , ctx]) => {
|
|
if (ctx?.prev)
|
|
qc.setQueryData (postKey, ctx.prev)
|
|
toast ({ title: '失敗……', description: '通信に失敗しました……' })
|
|
},
|
|
onSuccess: () => {
|
|
qc.invalidateQueries ({ queryKey: postsKeys.root })
|
|
} })
|
|
|
|
useEffect (() => {
|
|
if (!(errorFlg))
|
|
return
|
|
|
|
const code = (error as any)?.response.status ?? (error as any)?.status
|
|
if (code)
|
|
setStatus (code)
|
|
}, [errorFlg, error])
|
|
|
|
useEffect (() => {
|
|
scroll (0, 0)
|
|
|
|
setStatus (200)
|
|
}, [id])
|
|
|
|
switch (status)
|
|
{
|
|
case 404:
|
|
return <NotFound/>
|
|
case 503:
|
|
return <ServiceUnavailable/>
|
|
}
|
|
|
|
const viewedClass = (post?.viewed
|
|
? 'bg-blue-600 hover:bg-blue-700'
|
|
: 'bg-gray-500 hover:bg-gray-600')
|
|
|
|
return (
|
|
<div className="md:flex md:flex-1 overflow-y-auto md:overflow-y-hidden">
|
|
<Helmet>
|
|
{(post?.thumbnail || post?.thumbnailBase) && (
|
|
<meta name="thumbnail" content={post.thumbnail! || post.thumbnailBase!}/>)}
|
|
{post && <title>{`${ post.title || post.url } | ${ SITE_TITLE }`}</title>}
|
|
</Helmet>
|
|
|
|
<div className="hidden md:block">
|
|
{post && <TagDetailSidebar post={post}/>}
|
|
</div>
|
|
|
|
<MainArea className="relative">
|
|
{post
|
|
? (
|
|
<>
|
|
{(post.thumbnail || post.thumbnailBase) && (
|
|
<motion.div
|
|
layoutId={`page-${ id }`}
|
|
className="absolute top-4 left-4 w-[min(640px,calc(100vw-2rem))] h-[360px]
|
|
overflow-hidden rounded-xl pointer-events-none z-50"
|
|
initial={{ opacity: 1 }}
|
|
animate={{ opacity: 0 }}
|
|
transition={{ duration: .2, ease: 'easeOut' }}>
|
|
<img src={post.thumbnail || post.thumbnailBase || undefined}
|
|
alt={post.title || post.url}
|
|
title={post.title || post.url || undefined}
|
|
className="object-cover w-full h-full"/>
|
|
</motion.div>)}
|
|
|
|
<PostEmbed
|
|
ref={embedRef}
|
|
post={post}
|
|
onLoadComplete={() => embedRef.current?.play ()}/>
|
|
<Button onClick={() => changeViewedFlg.mutate ()}
|
|
disabled={changeViewedFlg.isPending}
|
|
className={cn ('text-white', viewedClass)}>
|
|
{post.viewed ? '閲覧済' : '未閲覧'}
|
|
</Button>
|
|
<TabGroup>
|
|
<Tab name="関聯">
|
|
{post.related.length > 0
|
|
? <PostList posts={post.related}/>
|
|
: 'まだないよ(笑)'}
|
|
</Tab>
|
|
{['admin', 'member'].some (r => user?.role === r) && (
|
|
<Tab name="編輯">
|
|
<PostEditForm
|
|
post={post}
|
|
onSave={newPost => {
|
|
qc.setQueryData (postsKeys.show (postId),
|
|
(prev: any) => newPost ?? prev)
|
|
qc.invalidateQueries ({ queryKey: postsKeys.root })
|
|
qc.invalidateQueries ({ queryKey: tagsKeys.root })
|
|
toast ({ description: '更新しました.' })
|
|
}}/>
|
|
</Tab>)}
|
|
</TabGroup>
|
|
</>)
|
|
: 'Loading...'}
|
|
</MainArea>
|
|
|
|
<div className="md:hidden">
|
|
{post && <TagDetailSidebar post={post} sp/>}
|
|
</div>
|
|
</div>)
|
|
}) satisfies FC<Props>
|