mirror of
https://github.com/MoonTechLab/LunaTV.git
synced 2026-02-28 09:23:14 +08:00
519 lines
21 KiB
TypeScript
519 lines
21 KiB
TypeScript
/* eslint-disable no-console,react-hooks/exhaustive-deps */
|
|
|
|
'use client';
|
|
|
|
import {
|
|
Bug,
|
|
CheckCircle,
|
|
ChevronDown,
|
|
ChevronUp,
|
|
Download,
|
|
Plus,
|
|
RefreshCw,
|
|
X,
|
|
} from 'lucide-react';
|
|
import { useEffect, useState } from 'react';
|
|
import { createPortal } from 'react-dom';
|
|
|
|
import { changelog, ChangelogEntry } from '@/lib/changelog';
|
|
import { compareVersions, CURRENT_VERSION, UpdateStatus } from '@/lib/version';
|
|
|
|
interface VersionPanelProps {
|
|
isOpen: boolean;
|
|
onClose: () => void;
|
|
}
|
|
|
|
interface RemoteChangelogEntry {
|
|
version: string;
|
|
date: string;
|
|
added: string[];
|
|
changed: string[];
|
|
fixed: string[];
|
|
}
|
|
|
|
export const VersionPanel: React.FC<VersionPanelProps> = ({
|
|
isOpen,
|
|
onClose,
|
|
}) => {
|
|
const [mounted, setMounted] = useState(false);
|
|
const [remoteChangelog, setRemoteChangelog] = useState<ChangelogEntry[]>([]);
|
|
const [hasUpdate, setIsHasUpdate] = useState(false);
|
|
const [latestVersion, setLatestVersion] = useState<string>('');
|
|
const [showRemoteContent, setShowRemoteContent] = useState(false);
|
|
|
|
// 确保组件已挂载
|
|
useEffect(() => {
|
|
setMounted(true);
|
|
return () => setMounted(false);
|
|
}, []);
|
|
|
|
// 获取远程变更日志
|
|
useEffect(() => {
|
|
if (isOpen) {
|
|
fetchRemoteChangelog();
|
|
}
|
|
}, [isOpen]);
|
|
|
|
// 获取远程变更日志
|
|
const fetchRemoteChangelog = async () => {
|
|
try {
|
|
const response = await fetch(
|
|
'https://codeberg.org/LunaTechLab/MoonTV/raw/branch/main/CHANGELOG'
|
|
);
|
|
if (response.ok) {
|
|
const content = await response.text();
|
|
const parsed = parseChangelog(content);
|
|
setRemoteChangelog(parsed);
|
|
|
|
// 检查是否有更新
|
|
if (parsed.length > 0) {
|
|
const latest = parsed[0];
|
|
setLatestVersion(latest.version);
|
|
setIsHasUpdate(
|
|
compareVersions(latest.version) === UpdateStatus.HAS_UPDATE
|
|
);
|
|
}
|
|
} else {
|
|
console.error(
|
|
'获取远程变更日志失败:',
|
|
response.status,
|
|
response.statusText
|
|
);
|
|
}
|
|
} catch (error) {
|
|
console.error('获取远程变更日志失败:', error);
|
|
}
|
|
};
|
|
|
|
// 解析变更日志格式
|
|
const parseChangelog = (content: string): RemoteChangelogEntry[] => {
|
|
const lines = content.split('\n');
|
|
const versions: RemoteChangelogEntry[] = [];
|
|
let currentVersion: RemoteChangelogEntry | null = null;
|
|
let currentSection: string | null = null;
|
|
let inVersionContent = false;
|
|
|
|
for (const line of lines) {
|
|
const trimmedLine = line.trim();
|
|
|
|
// 匹配版本行: ## [X.Y.Z] - YYYY-MM-DD
|
|
const versionMatch = trimmedLine.match(
|
|
/^## \[([\d.]+)\] - (\d{4}-\d{2}-\d{2})$/
|
|
);
|
|
if (versionMatch) {
|
|
if (currentVersion) {
|
|
versions.push(currentVersion);
|
|
}
|
|
|
|
currentVersion = {
|
|
version: versionMatch[1],
|
|
date: versionMatch[2],
|
|
added: [],
|
|
changed: [],
|
|
fixed: [],
|
|
};
|
|
currentSection = null;
|
|
inVersionContent = true;
|
|
continue;
|
|
}
|
|
|
|
// 如果遇到下一个版本或到达文件末尾,停止处理当前版本
|
|
if (inVersionContent && currentVersion) {
|
|
// 匹配章节标题
|
|
if (trimmedLine === '### Added') {
|
|
currentSection = 'added';
|
|
continue;
|
|
} else if (trimmedLine === '### Changed') {
|
|
currentSection = 'changed';
|
|
continue;
|
|
} else if (trimmedLine === '### Fixed') {
|
|
currentSection = 'fixed';
|
|
continue;
|
|
}
|
|
|
|
// 匹配条目: - 内容
|
|
if (trimmedLine.startsWith('- ') && currentSection) {
|
|
const entry = trimmedLine.substring(2);
|
|
if (currentSection === 'added') {
|
|
currentVersion.added.push(entry);
|
|
} else if (currentSection === 'changed') {
|
|
currentVersion.changed.push(entry);
|
|
} else if (currentSection === 'fixed') {
|
|
currentVersion.fixed.push(entry);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// 添加最后一个版本
|
|
if (currentVersion) {
|
|
versions.push(currentVersion);
|
|
}
|
|
|
|
return versions;
|
|
};
|
|
|
|
// 渲染变更日志条目
|
|
const renderChangelogEntry = (
|
|
entry: ChangelogEntry | RemoteChangelogEntry,
|
|
isCurrentVersion = false,
|
|
isRemote = false
|
|
) => {
|
|
const isUpdate = isRemote && hasUpdate && entry.version === latestVersion;
|
|
|
|
return (
|
|
<div
|
|
key={entry.version}
|
|
className={`p-4 rounded-lg border ${isCurrentVersion
|
|
? 'bg-blue-50 dark:bg-blue-900/20 border-blue-200 dark:border-blue-800'
|
|
: isUpdate
|
|
? 'bg-yellow-50 dark:bg-yellow-900/20 border-yellow-200 dark:border-yellow-800'
|
|
: 'bg-gray-50 dark:bg-gray-800/60 border-gray-200 dark:border-gray-700'
|
|
}`}
|
|
>
|
|
{/* 版本标题 */}
|
|
<div className='flex flex-col sm:flex-row sm:items-center justify-between gap-2 mb-3'>
|
|
<div className='flex flex-wrap items-center gap-2'>
|
|
<h4 className='text-lg font-semibold text-gray-900 dark:text-gray-100'>
|
|
v{entry.version}
|
|
</h4>
|
|
{isCurrentVersion && (
|
|
<span className='px-2 py-1 text-xs font-medium bg-blue-100 text-blue-800 dark:bg-blue-900/30 dark:text-blue-300 rounded-full'>
|
|
当前版本
|
|
</span>
|
|
)}
|
|
{isUpdate && (
|
|
<span className='px-2 py-1 text-xs font-medium bg-yellow-100 text-yellow-800 dark:bg-yellow-900/30 dark:text-yellow-300 rounded-full flex items-center gap-1'>
|
|
<Download className='w-3 h-3' />
|
|
可更新
|
|
</span>
|
|
)}
|
|
</div>
|
|
<div className='flex items-center gap-2 text-sm text-gray-500 dark:text-gray-400'>
|
|
{entry.date}
|
|
</div>
|
|
</div>
|
|
|
|
{/* 变更内容 */}
|
|
<div className='space-y-3'>
|
|
{entry.added.length > 0 && (
|
|
<div>
|
|
<h5 className='text-sm font-medium text-green-700 dark:text-green-400 mb-2 flex items-center gap-1'>
|
|
<Plus className='w-4 h-4' />
|
|
新增功能
|
|
</h5>
|
|
<ul className='space-y-1'>
|
|
{entry.added.map((item, index) => (
|
|
<li
|
|
key={index}
|
|
className='text-sm text-gray-700 dark:text-gray-300 flex items-start gap-2'
|
|
>
|
|
<span className='w-1.5 h-1.5 bg-green-500 rounded-full mt-2 flex-shrink-0'></span>
|
|
{item}
|
|
</li>
|
|
))}
|
|
</ul>
|
|
</div>
|
|
)}
|
|
|
|
{entry.changed.length > 0 && (
|
|
<div>
|
|
<h5 className='text-sm font-medium text-blue-700 dark:text-blue-400 mb-2 flex items-center gap-1'>
|
|
<RefreshCw className='w-4 h-4' />
|
|
功能改进
|
|
</h5>
|
|
<ul className='space-y-1'>
|
|
{entry.changed.map((item, index) => (
|
|
<li
|
|
key={index}
|
|
className='text-sm text-gray-700 dark:text-gray-300 flex items-start gap-2'
|
|
>
|
|
<span className='w-1.5 h-1.5 bg-blue-500 rounded-full mt-2 flex-shrink-0'></span>
|
|
{item}
|
|
</li>
|
|
))}
|
|
</ul>
|
|
</div>
|
|
)}
|
|
|
|
{entry.fixed.length > 0 && (
|
|
<div>
|
|
<h5 className='text-sm font-medium text-purple-700 dark:text-purple-400 mb-2 flex items-center gap-1'>
|
|
<Bug className='w-4 h-4' />
|
|
问题修复
|
|
</h5>
|
|
<ul className='space-y-1'>
|
|
{entry.fixed.map((item, index) => (
|
|
<li
|
|
key={index}
|
|
className='text-sm text-gray-700 dark:text-gray-300 flex items-start gap-2'
|
|
>
|
|
<span className='w-1.5 h-1.5 bg-purple-500 rounded-full mt-2 flex-shrink-0'></span>
|
|
{item}
|
|
</li>
|
|
))}
|
|
</ul>
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
// 版本面板内容
|
|
const versionPanelContent = (
|
|
<>
|
|
{/* 背景遮罩 */}
|
|
<div
|
|
className='fixed inset-0 bg-black/50 backdrop-blur-sm z-[1000]'
|
|
onClick={onClose}
|
|
/>
|
|
|
|
{/* 版本面板 */}
|
|
<div className='fixed top-1/2 left-1/2 -translate-x-1/2 -translate-y-1/2 w-full max-w-xl max-h-[90vh] bg-white dark:bg-gray-900 rounded-xl shadow-xl z-[1001] overflow-hidden'>
|
|
{/* 标题栏 */}
|
|
<div className='flex items-center justify-between p-3 sm:p-6 border-b border-gray-200 dark:border-gray-700'>
|
|
<div className='flex items-center gap-2 sm:gap-3'>
|
|
<h3 className='text-lg sm:text-xl font-bold text-gray-800 dark:text-gray-200'>
|
|
版本信息
|
|
</h3>
|
|
<div className='flex flex-wrap items-center gap-1 sm:gap-2'>
|
|
<span className='px-2 sm:px-3 py-1 text-xs sm:text-sm font-medium bg-green-100 text-green-800 dark:bg-green-900/30 dark:text-green-300 rounded-full'>
|
|
v{CURRENT_VERSION}
|
|
</span>
|
|
{hasUpdate && (
|
|
<span className='px-2 sm:px-3 py-1 text-xs sm:text-sm font-medium bg-yellow-100 text-yellow-800 dark:bg-yellow-900/30 dark:text-yellow-300 rounded-full flex items-center gap-1'>
|
|
<Download className='w-3 h-3 sm:w-4 sm:h-4' />
|
|
<span className='hidden sm:inline'>有新版本可用</span>
|
|
<span className='sm:hidden'>可更新</span>
|
|
</span>
|
|
)}
|
|
</div>
|
|
</div>
|
|
<button
|
|
onClick={onClose}
|
|
className='w-6 h-6 sm:w-8 sm:h-8 p-1 rounded-full flex items-center justify-center text-gray-500 hover:bg-gray-100 dark:hover:bg-gray-800 transition-colors'
|
|
aria-label='关闭'
|
|
>
|
|
<X className='w-full h-full' />
|
|
</button>
|
|
</div>
|
|
|
|
{/* 内容区域 */}
|
|
<div className='p-3 sm:p-6 overflow-y-auto max-h-[calc(95vh-140px)] sm:max-h-[calc(90vh-120px)]'>
|
|
<div className='space-y-3 sm:space-y-6'>
|
|
{/* 远程更新信息 */}
|
|
{hasUpdate && (
|
|
<div className='bg-gradient-to-r from-yellow-50 to-amber-50 dark:from-yellow-900/20 dark:to-amber-900/20 border border-yellow-200 dark:border-yellow-800 rounded-lg p-3 sm:p-4'>
|
|
<div className='flex flex-col gap-3'>
|
|
<div className='flex items-center gap-2 sm:gap-3'>
|
|
<div className='w-8 h-8 sm:w-10 sm:h-10 bg-yellow-100 dark:bg-yellow-800/40 rounded-full flex items-center justify-center flex-shrink-0'>
|
|
<Download className='w-4 h-4 sm:w-5 sm:h-5 text-yellow-600 dark:text-yellow-400' />
|
|
</div>
|
|
<div className='min-w-0 flex-1'>
|
|
<h4 className='text-sm sm:text-base font-semibold text-yellow-800 dark:text-yellow-200'>
|
|
发现新版本
|
|
</h4>
|
|
<p className='text-xs sm:text-sm text-yellow-700 dark:text-yellow-300 break-all'>
|
|
v{CURRENT_VERSION} → v{latestVersion}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
<a
|
|
href='https://github.com/MoonTechLab/LunaTV'
|
|
target='_blank'
|
|
rel='noopener noreferrer'
|
|
className='inline-flex items-center justify-center gap-2 px-3 py-2 bg-yellow-600 hover:bg-yellow-700 text-white text-xs sm:text-sm rounded-lg transition-colors shadow-sm w-full'
|
|
>
|
|
<Download className='w-3 h-3 sm:w-4 sm:h-4' />
|
|
前往仓库
|
|
</a>
|
|
</div>
|
|
</div>
|
|
)}
|
|
|
|
{/* 当前为最新版本信息 */}
|
|
{!hasUpdate && (
|
|
<div className='bg-gradient-to-r from-green-50 to-emerald-50 dark:from-green-900/20 dark:to-emerald-900/20 border border-green-200 dark:border-green-800 rounded-lg p-3 sm:p-4'>
|
|
<div className='flex flex-col gap-3'>
|
|
<div className='flex items-center gap-2 sm:gap-3'>
|
|
<div className='w-8 h-8 sm:w-10 sm:h-10 bg-green-100 dark:bg-green-800/40 rounded-full flex items-center justify-center flex-shrink-0'>
|
|
<CheckCircle className='w-4 h-4 sm:w-5 sm:h-5 text-green-600 dark:text-green-400' />
|
|
</div>
|
|
<div className='min-w-0 flex-1'>
|
|
<h4 className='text-sm sm:text-base font-semibold text-green-800 dark:text-green-200'>
|
|
当前为最新版本
|
|
</h4>
|
|
<p className='text-xs sm:text-sm text-green-700 dark:text-green-300 break-all'>
|
|
已是最新版本 v{CURRENT_VERSION}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
<a
|
|
href='https://github.com/MoonTechLab/LunaTV'
|
|
target='_blank'
|
|
rel='noopener noreferrer'
|
|
className='inline-flex items-center justify-center gap-2 px-3 py-2 bg-green-600 hover:bg-green-700 text-white text-xs sm:text-sm rounded-lg transition-colors shadow-sm w-full'
|
|
>
|
|
<CheckCircle className='w-3 h-3 sm:w-4 sm:h-4' />
|
|
前往仓库
|
|
</a>
|
|
</div>
|
|
</div>
|
|
)}
|
|
|
|
{/* 远程可更新内容 */}
|
|
{hasUpdate && (
|
|
<div className='space-y-4'>
|
|
<div className='flex flex-col sm:flex-row sm:items-center justify-between gap-3'>
|
|
<h4 className='text-lg font-semibold text-gray-800 dark:text-gray-200 flex items-center gap-2'>
|
|
<Download className='w-5 h-5 text-yellow-500' />
|
|
远程更新内容
|
|
</h4>
|
|
<button
|
|
onClick={() => setShowRemoteContent(!showRemoteContent)}
|
|
className='inline-flex items-center justify-center gap-2 px-3 py-1.5 bg-yellow-100 hover:bg-yellow-200 text-yellow-800 dark:bg-yellow-800/30 dark:hover:bg-yellow-800/50 dark:text-yellow-200 rounded-lg transition-colors text-sm w-full sm:w-auto'
|
|
>
|
|
{showRemoteContent ? (
|
|
<>
|
|
<ChevronUp className='w-4 h-4' />
|
|
收起
|
|
</>
|
|
) : (
|
|
<>
|
|
<ChevronDown className='w-4 h-4' />
|
|
查看更新内容
|
|
</>
|
|
)}
|
|
</button>
|
|
</div>
|
|
|
|
{showRemoteContent && remoteChangelog.length > 0 && (
|
|
<div className='space-y-4'>
|
|
{remoteChangelog
|
|
.filter((entry) => {
|
|
// 找到第一个本地版本,过滤掉本地已有的版本
|
|
const localVersions = changelog.map(
|
|
(local) => local.version
|
|
);
|
|
return !localVersions.includes(entry.version);
|
|
})
|
|
.map((entry, index) => (
|
|
<div
|
|
key={index}
|
|
className={`p-4 rounded-lg border ${entry.version === latestVersion
|
|
? 'bg-yellow-50 dark:bg-yellow-900/20 border-yellow-200 dark:border-yellow-800'
|
|
: 'bg-gray-50 dark:bg-gray-800/60 border-gray-200 dark:border-gray-700'
|
|
}`}
|
|
>
|
|
<div className='flex flex-col sm:flex-row sm:items-center justify-between gap-2 mb-3'>
|
|
<div className='flex flex-wrap items-center gap-2'>
|
|
<h4 className='text-lg font-semibold text-gray-900 dark:text-gray-100'>
|
|
v{entry.version}
|
|
</h4>
|
|
{entry.version === latestVersion && (
|
|
<span className='px-2 py-1 text-xs font-medium bg-yellow-100 text-yellow-800 dark:bg-yellow-900/30 dark:text-yellow-300 rounded-full flex items-center gap-1'>
|
|
远程最新
|
|
</span>
|
|
)}
|
|
</div>
|
|
<div className='flex items-center gap-2 text-sm text-gray-500 dark:text-gray-400'>
|
|
{entry.date}
|
|
</div>
|
|
</div>
|
|
|
|
{entry.added && entry.added.length > 0 && (
|
|
<div className='mb-3'>
|
|
<h5 className='text-sm font-medium text-green-600 dark:text-green-400 mb-2 flex items-center gap-1'>
|
|
<Plus className='w-4 h-4' />
|
|
新增功能
|
|
</h5>
|
|
<ul className='space-y-1'>
|
|
{entry.added.map((item, itemIndex) => (
|
|
<li
|
|
key={itemIndex}
|
|
className='text-sm text-gray-700 dark:text-gray-300 flex items-start gap-2'
|
|
>
|
|
<span className='w-1.5 h-1.5 bg-green-400 rounded-full mt-2 flex-shrink-0'></span>
|
|
{item}
|
|
</li>
|
|
))}
|
|
</ul>
|
|
</div>
|
|
)}
|
|
|
|
{entry.changed && entry.changed.length > 0 && (
|
|
<div className='mb-3'>
|
|
<h5 className='text-sm font-medium text-blue-600 dark:text-blue-400 mb-2 flex items-center gap-1'>
|
|
<RefreshCw className='w-4 h-4' />
|
|
功能改进
|
|
</h5>
|
|
<ul className='space-y-1'>
|
|
{entry.changed.map((item, itemIndex) => (
|
|
<li
|
|
key={itemIndex}
|
|
className='text-sm text-gray-700 dark:text-gray-300 flex items-start gap-2'
|
|
>
|
|
<span className='w-1.5 h-1.5 bg-blue-400 rounded-full mt-2 flex-shrink-0'></span>
|
|
{item}
|
|
</li>
|
|
))}
|
|
</ul>
|
|
</div>
|
|
)}
|
|
|
|
{entry.fixed && entry.fixed.length > 0 && (
|
|
<div>
|
|
<h5 className='text-sm font-medium text-purple-700 dark:text-purple-400 mb-2 flex items-center gap-1'>
|
|
<Bug className='w-4 h-4' />
|
|
问题修复
|
|
</h5>
|
|
<ul className='space-y-1'>
|
|
{entry.fixed.map((item, itemIndex) => (
|
|
<li
|
|
key={itemIndex}
|
|
className='text-sm text-gray-700 dark:text-gray-300 flex items-start gap-2'
|
|
>
|
|
<span className='w-1.5 h-1.5 bg-purple-500 rounded-full mt-2 flex-shrink-0'></span>
|
|
{item}
|
|
</li>
|
|
))}
|
|
</ul>
|
|
</div>
|
|
)}
|
|
</div>
|
|
))}
|
|
</div>
|
|
)}
|
|
</div>
|
|
)}
|
|
|
|
{/* 变更日志标题 */}
|
|
<div className='border-b border-gray-200 dark:border-gray-700 pb-4'>
|
|
<h4 className='text-lg font-semibold text-gray-800 dark:text-gray-200 pb-3 sm:pb-4'>
|
|
变更日志
|
|
</h4>
|
|
|
|
<div className='space-y-4'>
|
|
{/* 本地变更日志 */}
|
|
{changelog.map((entry) =>
|
|
renderChangelogEntry(
|
|
entry,
|
|
entry.version === CURRENT_VERSION,
|
|
false
|
|
)
|
|
)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</>
|
|
);
|
|
|
|
// 使用 Portal 渲染到 document.body
|
|
if (!mounted || !isOpen) return null;
|
|
|
|
return createPortal(versionPanelContent, document.body);
|
|
};
|