|
import { Fragment, useRef, useState } from 'react' |
|
import { Dialog, Transition } from '@headlessui/react' |
|
import { usePlogGlobal } from '..' |
|
import { ArrowPath, ChevronLeft, ChevronRight } from '@/components/HeroIcons' |
|
import Link from 'next/link' |
|
import { siteConfig } from '@/lib/config' |
|
import LazyImage from '@/components/LazyImage' |
|
import { compressImage } from '@/lib/notion/mapImage' |
|
|
|
|
|
|
|
|
|
export default function Modal(props) { |
|
const { showModal, setShowModal, modalContent, setModalContent } = usePlogGlobal() |
|
const { siteInfo, posts } = props |
|
const cancelButtonRef = useRef(null) |
|
const img = compressImage(modalContent?.pageCover || siteInfo?.pageCover, 1200, 85, 'webp') |
|
const imgRef = useRef(null) |
|
|
|
|
|
const [loading, setLoading] = useState(true) |
|
|
|
|
|
function handleImageLoad() { |
|
setLoading(false) |
|
} |
|
|
|
|
|
function handleClose() { |
|
setShowModal(false) |
|
setLoading(true) |
|
} |
|
|
|
|
|
function prev() { |
|
setLoading(true) |
|
const index = posts?.findIndex(post => post.slug === modalContent.slug) |
|
if (index === 0) { |
|
setModalContent(posts[posts.length - 1]) |
|
} else { |
|
setModalContent(posts[index - 1]) |
|
} |
|
} |
|
|
|
const next = () => { |
|
setLoading(true) |
|
const index = posts.findIndex(post => post.slug === modalContent.slug) |
|
if (index === posts.length - 1) { |
|
setModalContent(posts[0]) |
|
} else { |
|
setModalContent(posts[index + 1]) |
|
} |
|
} |
|
|
|
return ( |
|
<Transition.Root show={showModal} as={Fragment}> |
|
<Dialog as="div" className="relative z-20" initialFocus={cancelButtonRef} onClose={handleClose}> |
|
{/* 遮罩 */} |
|
<Transition.Child |
|
as={Fragment} |
|
enter="ease-out duration-300" |
|
enterFrom="opacity-0" |
|
enterTo="opacity-100" |
|
leave="ease-in duration-200" |
|
leaveFrom="opacity-100" |
|
leaveTo="opacity-0" |
|
> |
|
<div style={{ backgroundColor: 'rgba(0, 0, 0, 0.5)' }} className="fixed inset-0 glassmorphism transition-opacity" /> |
|
</Transition.Child> |
|
|
|
<div className="fixed inset-0 z-30 overflow-y-auto"> |
|
<div className="flex min-h-full justify-center p-4 text-center items-center"> |
|
<Transition.Child |
|
as={Fragment} |
|
enter="ease-out duration-300" |
|
enterFrom="opacity-0 translate-y-4 scale-50 w-0" |
|
enterTo={'opacity-100 translate-y-0 max-w-screen'} |
|
leave="ease-in duration-200" |
|
leaveFrom="opacity-100 translate-y-0 scale-100 max-w-screen" |
|
leaveTo="opacity-0 translate-y-4 scale-50 w-0" |
|
> |
|
<Dialog.Panel className="relative transform overflow-hidden rounded-xl text-left shadow-xl transition-all "> |
|
{/* 添加loading状态 */} |
|
<div className={`bg-hexo-black-gray w-32 h-32 flex justify-center items-center ${loading ? '' : 'hidden'}`}> |
|
<ArrowPath className='w-10 h-10 animate-spin text-gray-200' /> |
|
</div> |
|
|
|
{/* 添加onLoad事件处理函数 */} |
|
<LazyImage onLoad={handleImageLoad} src={img} ref={imgRef} style={{ display: loading ? 'none' : 'block' }} className={`w-full max-w-7xl max-h-[90vh] shadow-xl ${!loading ? ' animate__animated animate__fadeIn' : ''}`} /> |
|
|
|
{!loading && (<> |
|
<div className='absolute bottom-0 left-0 m-4 z-20'> |
|
<Link href={`${siteConfig('SUB_PATH', '')}/${modalContent.slug}`}> |
|
<div className='flex'> |
|
<h2 style={{ textShadow: '0.1em 0.1em 0.2em black' }} className='text-2xl md:text-5xl text-white mb-4 px-2 py-1 rounded-lg'>{modalContent?.title}</h2> |
|
</div> |
|
</Link> |
|
<Link href={`${siteConfig('SUB_PATH', '')}/${modalContent.slug}`}> |
|
<div style={{ textShadow: '0.1em 0.1em 0.2em black' }} className={'line-clamp-3 md:line-clamp-none overflow-hidden cursor-pointer text-gray-50 rounded-lg m-2'}> |
|
{modalContent?.summary} |
|
</div> |
|
</Link> |
|
|
|
{modalContent?.category && ( |
|
<div className='flex'> |
|
<Link href={`/category/${modalContent?.category}`} className='text-xs rounded-lg mt-3 px-2 py-1 bg-black bg-opacity-20 text-white hover:bg-blue-700 hover:text-white duration-200'> |
|
{modalContent?.category} |
|
</Link> |
|
</div> |
|
)} |
|
</div> |
|
<div className='z-10 absolute hover:opacity-50 opacity-0 duration-200 transition-opacity w-full top-0 left-0 px-4 h-full items-center flex justify-between'> |
|
<div onClick={prev}><ChevronLeft className='cursor-pointer w-24 h-32 hover:opacity-100 stroke-white stroke-1 scale-y-150' /></div> |
|
<div onClick={next}><ChevronRight className='cursor-pointer w-24 h-32 hover:opacity-100 stroke-white stroke-1 scale-y-150' /></div> |
|
</div> |
|
</>)} |
|
|
|
</Dialog.Panel> |
|
</Transition.Child> |
|
</div> |
|
</div> |
|
</Dialog> |
|
</Transition.Root> |
|
) |
|
} |
|
|