mirror of
https://github.com/perfect-panel/ppanel-web.git
synced 2026-02-06 03:30:25 -05:00
142 lines
4.8 KiB
TypeScript
142 lines
4.8 KiB
TypeScript
'use client';
|
|
|
|
import { queryDocumentDetail } from '@/services/user/document';
|
|
import { useQuery } from '@tanstack/react-query';
|
|
import { Avatar, AvatarFallback } from '@workspace/ui/components/avatar';
|
|
import { buttonVariants } from '@workspace/ui/components/button';
|
|
import { Markdown } from '@workspace/ui/custom-components/markdown';
|
|
import { useOutsideClick } from '@workspace/ui/hooks/use-outside-click';
|
|
import { cn } from '@workspace/ui/lib/utils';
|
|
import { formatDate } from '@workspace/ui/utils';
|
|
import { AnimatePresence, motion } from 'framer-motion';
|
|
import { useTranslations } from 'next-intl';
|
|
import { RefObject, useEffect, useId, useRef, useState } from 'react';
|
|
import { CloseIcon } from './close-icon';
|
|
|
|
export function DocumentButton({ items }: { items: API.Document[] }) {
|
|
const t = useTranslations('document');
|
|
const [active, setActive] = useState<API.Document | boolean | null>(null);
|
|
const id = useId();
|
|
const ref = useRef<HTMLDivElement>(null);
|
|
|
|
const { data } = useQuery({
|
|
enabled: !!(active as API.Document)?.id,
|
|
queryKey: ['queryDocumentDetail', (active as API.Document)?.id],
|
|
queryFn: async () => {
|
|
const { data } = await queryDocumentDetail({
|
|
id: (active as API.Document)?.id,
|
|
});
|
|
return data.data?.content;
|
|
},
|
|
});
|
|
|
|
useEffect(() => {
|
|
function onKeyDown(event: KeyboardEvent) {
|
|
if (event.key === 'Escape') {
|
|
setActive(false);
|
|
}
|
|
}
|
|
|
|
if (active && typeof active === 'object') {
|
|
document.body.style.overflow = 'hidden';
|
|
} else {
|
|
document.body.style.overflow = 'auto';
|
|
}
|
|
|
|
window.addEventListener('keydown', onKeyDown);
|
|
return () => window.removeEventListener('keydown', onKeyDown);
|
|
}, [active]);
|
|
|
|
useOutsideClick(ref as RefObject<HTMLDivElement>, () => setActive(null));
|
|
|
|
return (
|
|
<>
|
|
<AnimatePresence>
|
|
{active && typeof active === 'object' && (
|
|
<motion.div
|
|
initial={{ opacity: 0 }}
|
|
animate={{ opacity: 1 }}
|
|
exit={{ opacity: 0 }}
|
|
className='fixed inset-0 z-10 h-full w-full bg-black/20'
|
|
/>
|
|
)}
|
|
</AnimatePresence>
|
|
<AnimatePresence>
|
|
{active && typeof active === 'object' ? (
|
|
<div className='fixed inset-0 z-[100] grid place-items-center'>
|
|
<motion.button
|
|
key={`button-${active.title}-${id}`}
|
|
layout
|
|
initial={{
|
|
opacity: 0,
|
|
}}
|
|
animate={{
|
|
opacity: 1,
|
|
}}
|
|
exit={{
|
|
opacity: 0,
|
|
transition: {
|
|
duration: 0.05,
|
|
},
|
|
}}
|
|
className='bg-foreground absolute right-2 top-2 flex h-6 w-6 items-center justify-center rounded-full text-white dark:text-black'
|
|
onClick={() => setActive(null)}
|
|
>
|
|
<CloseIcon />
|
|
</motion.button>
|
|
<motion.div
|
|
layoutId={`card-${active.id}-${id}`}
|
|
ref={ref}
|
|
className='bg-muted flex size-full flex-col overflow-auto p-6 sm:rounded'
|
|
>
|
|
<Markdown>{data || ''}</Markdown>
|
|
</motion.div>
|
|
</div>
|
|
) : null}
|
|
</AnimatePresence>
|
|
<ul className='flex w-full flex-col gap-4'>
|
|
{items.map((item, index) => (
|
|
<motion.div
|
|
layoutId={`card-${item.id}-${id}`}
|
|
key={`card-${item.id}-${id}`}
|
|
onClick={() => setActive(item)}
|
|
className='bg-background hover:bg-accent flex cursor-pointer items-center justify-between rounded-xl border p-4'
|
|
>
|
|
<div className='flex flex-row items-center gap-4'>
|
|
<motion.div layoutId={`image-${item.id}-${id}`}>
|
|
<Avatar className='size-12'>
|
|
<AvatarFallback className='bg-primary/80 text-white'>
|
|
{item.title.split('')[0]}
|
|
</AvatarFallback>
|
|
</Avatar>
|
|
</motion.div>
|
|
<div className=''>
|
|
<motion.h3 layoutId={`title-${item.id}-${id}`} className='font-medium'>
|
|
{item.title}
|
|
</motion.h3>
|
|
<motion.p
|
|
layoutId={`description-${item.id}-${id}`}
|
|
className='text-sm text-neutral-600 dark:text-neutral-400'
|
|
>
|
|
{formatDate(item.updated_at)}
|
|
</motion.p>
|
|
</div>
|
|
</div>
|
|
<motion.button
|
|
layoutId={`button-${item.id}-${id}`}
|
|
className={cn(
|
|
buttonVariants({
|
|
variant: 'secondary',
|
|
}),
|
|
'rounded-full',
|
|
)}
|
|
>
|
|
{t('read')}
|
|
</motion.button>
|
|
</motion.div>
|
|
))}
|
|
</ul>
|
|
</>
|
|
);
|
|
}
|