Files
ittoview/src/components/ChangelogModal.tsx
ittoview b2ec80c199 fix(整合): 再次提升更新日志模态框层级至最高优先级
- 将背景遮罩 z-index 从 z-[100] 提升到 z-[9999]
- 将模态框内容 z-index 从 z-[101] 提升到 z-[10000]
- 确保模态框在所有可能的元素之上显示,包括 Header 搜索下拉等

via [HAPI](https://hapi.run)

Co-Authored-By: HAPI <noreply@hapi.run>
2026-03-08 03:36:50 +00:00

172 lines
7.6 KiB
TypeScript

import { motion, AnimatePresence } from 'framer-motion'
import { X, History, CalendarDays, Tag } from 'lucide-react'
import { changelogEntries } from '@/data'
import type { ChangelogType } from '@/types/itto'
interface ChangelogModalProps {
isOpen: boolean
onClose: () => void
}
const containerVariants = {
hidden: { opacity: 0 },
visible: {
opacity: 1,
transition: {
staggerChildren: 0.06,
},
},
}
const itemVariants = {
hidden: { opacity: 0, y: 12 },
visible: {
opacity: 1,
y: 0,
transition: {
duration: 0.25,
},
},
}
const typeMeta: Record<ChangelogType, { label: string; className: string }> = {
feat: { label: '新功能', className: 'bg-emerald-100 text-emerald-700 dark:bg-emerald-900/30 dark:text-emerald-300' },
fix: { label: '修复', className: 'bg-rose-100 text-rose-700 dark:bg-rose-900/30 dark:text-rose-300' },
style: { label: '样式', className: 'bg-pink-100 text-pink-700 dark:bg-pink-900/30 dark:text-pink-300' },
refactor: { label: '重构', className: 'bg-violet-100 text-violet-700 dark:bg-violet-900/30 dark:text-violet-300' },
docs: { label: '文档', className: 'bg-blue-100 text-blue-700 dark:bg-blue-900/30 dark:text-blue-300' },
perf: { label: '性能', className: 'bg-amber-100 text-amber-700 dark:bg-amber-900/30 dark:text-amber-300' },
test: { label: '测试', className: 'bg-cyan-100 text-cyan-700 dark:bg-cyan-900/30 dark:text-cyan-300' },
chore: { label: '工程', className: 'bg-gray-100 text-gray-700 dark:bg-gray-700 dark:text-gray-300' },
}
function formatDate(date: string) {
const [year, month, day] = date.split('-').map(Number)
if (!year || !month || !day) return date
return `${year}${month}${day}`
}
export function ChangelogModal({ isOpen, onClose }: ChangelogModalProps) {
return (
<AnimatePresence>
{isOpen && (
<>
{/* 背景遮罩 */}
<motion.div
initial={{ opacity: 0 }}
animate={{ opacity: 1 }}
exit={{ opacity: 0 }}
onClick={onClose}
className="fixed inset-0 bg-black/50 z-[9999]"
/>
{/* 模态框 */}
<motion.div
initial={{ opacity: 0, scale: 0.95, y: 20 }}
animate={{ opacity: 1, scale: 1, y: 0 }}
exit={{ opacity: 0, scale: 0.95, y: 20 }}
className="fixed inset-4 md:inset-8 lg:inset-16 z-[10000] flex items-center justify-center"
>
<div className="bg-white dark:bg-gray-800 rounded-2xl shadow-2xl w-full h-full flex flex-col overflow-hidden">
{/* 头部 */}
<div className="flex items-center justify-between px-6 py-4 border-b border-gray-200 dark:border-gray-700 flex-shrink-0">
<div className="flex items-center gap-3">
<div className="flex h-10 w-10 items-center justify-center rounded-lg bg-indigo-50 dark:bg-indigo-900/50">
<History size={20} className="text-indigo-600 dark:text-indigo-400" />
</div>
<div>
<h2 className="text-xl font-bold text-gray-900 dark:text-white"></h2>
<p className="text-sm text-gray-500 dark:text-gray-400">
{changelogEntries.length}
</p>
</div>
</div>
<button
onClick={onClose}
className="flex h-10 w-10 items-center justify-center rounded-lg hover:bg-gray-100 dark:hover:bg-gray-700 text-gray-500 dark:text-gray-400 transition-colors"
aria-label="关闭"
>
<X size={20} />
</button>
</div>
{/* 内容区域 */}
<div className="flex-1 overflow-y-auto p-6">
{changelogEntries.length > 0 ? (
<motion.div
variants={containerVariants}
initial="hidden"
animate="visible"
className="relative max-w-4xl mx-auto"
>
{/* 时间轴竖线 */}
<div className="absolute left-[15px] top-8 bottom-8 w-[2px] bg-gray-200 dark:bg-gray-700" />
{/* 更新记录列表 */}
<div className="space-y-6">
{changelogEntries.map((entry, index) => {
const meta = typeMeta[entry.type]
return (
<motion.div
key={entry.id || `${entry.date}-${index}`}
variants={itemVariants}
className="relative flex gap-6"
>
{/* 时间轴节点 */}
<div className="relative flex-shrink-0">
<div className="w-8 h-8 rounded-full bg-white dark:bg-gray-800 border-2 border-indigo-500 dark:border-indigo-400 flex items-center justify-center z-10">
<div className="w-3 h-3 rounded-full bg-indigo-500 dark:bg-indigo-400" />
</div>
</div>
{/* 更新卡片 */}
<div className="flex-1 pb-2">
<div className="bg-gray-50 dark:bg-gray-900 rounded-xl border border-gray-200 dark:border-gray-700 p-4 hover:border-indigo-300 dark:hover:border-indigo-600 transition-colors">
{/* 标签行 */}
<div className="flex flex-wrap items-center gap-2 mb-2">
{/* 类型标签 */}
<span className={`inline-flex items-center gap-1 px-2.5 py-1 rounded-md text-xs font-medium ${meta.className}`}>
<Tag size={12} />
{meta.label}
</span>
{/* 范围标签 */}
{entry.scope && (
<span className="inline-flex items-center px-2.5 py-1 rounded-md text-xs font-medium bg-gray-200 text-gray-700 dark:bg-gray-700 dark:text-gray-300">
{entry.scope}
</span>
)}
{/* 日期(桌面端靠右,移动端换行) */}
<span className="inline-flex items-center gap-1.5 text-xs text-gray-500 dark:text-gray-400 sm:ml-auto">
<CalendarDays size={12} />
{formatDate(entry.date)}
</span>
</div>
{/* 标题 */}
<h3 className="text-sm font-medium text-gray-900 dark:text-white">
{entry.title}
</h3>
</div>
</div>
</motion.div>
)
})}
</div>
</motion.div>
) : (
<div className="flex flex-col items-center justify-center py-16 text-gray-400 dark:text-gray-500">
<History size={48} className="mb-4" />
<p className="text-sm"></p>
</div>
)}
</div>
</div>
</motion.div>
</>
)}
</AnimatePresence>
)
}