mirror of
https://github.com/LHY0125/PathEditor.git
synced 2026-06-30 10:35:54 +08:00
8c0e80d862
新增配置文件: - .editorconfig — 跨编辑器代码风格统一 - .gitattributes — 行尾符 CRLF 规范化 - .prettierrc + .prettierignore — 前端代码格式化 - .markdownlint.json — Markdown 格式规范 - commitlint.config.js — Conventional Commits 强制校验 新增 GitHub 社区文件: - .github/dependabot.yml — 依赖自动更新 (npm + Cargo + Actions) - .github/CODEOWNERS — 自动 PR 审查分配 - .github/FUNDING.yml — 开源赞助入口 新增文档: - ROADMAP.md — v5.1/v5.2/v6.0 路线图 - SUPPORT.md — 帮助与支持指南 - docs/screenshots/ — 截图目录就位 新增 Git Hooks: - .husky/pre-commit — lint-staged 自动格式化+修复 - .husky/commit-msg — commitlint 校验提交消息 CI 强化 (.github/workflows/ci.yml): - 新增 Prettier 格式检查步骤 - 新增 cargo fmt --check 步骤 - 新增 Vitest 覆盖率生成 + Codecov 上报 修复: - index.html 标题 v4.0 → v5.0 - PathEditDialog set-state-in-effect 改用 useRef prevOpen 守卫 - use-app-actions.test.tsx 缺失 @vitest-environment jsdom - 所有 TS/TSX 文件 Prettier 格式化统一 配置更新: - vitest.config.ts — v8 覆盖率 + 阈值门禁 (60%/70%) - package.json — format/format:check/test:coverage/prepare 脚本 + lint-staged - .gitignore — 新增 coverage/sync-conflict/playwright-report - README.md — 新增 coverage + platform 徽章 + 截图区域
79 lines
2.7 KiB
TypeScript
79 lines
2.7 KiB
TypeScript
import { useMemo } from 'react';
|
|
import { useAppStore } from '@/store/app-store';
|
|
import { useTranslation } from 'react-i18next';
|
|
import type { PathEntry } from '@/core/path-entry';
|
|
|
|
export function MergePreview() {
|
|
const sysPaths = useAppStore((s) => s.sysPaths);
|
|
const userPaths = useAppStore((s) => s.userPaths);
|
|
const searchQuery = useAppStore((s) => s.searchQuery);
|
|
const { t } = useTranslation();
|
|
|
|
const allPaths = useMemo(() => {
|
|
const seen = new Set<string>();
|
|
const merged: (PathEntry & { source: string; displayIndex: number })[] = [];
|
|
|
|
for (const entry of sysPaths) {
|
|
const lower = entry.path.toLowerCase();
|
|
if (!seen.has(lower)) {
|
|
seen.add(lower);
|
|
merged.push({ ...entry, source: t('merge.system'), displayIndex: merged.length });
|
|
}
|
|
}
|
|
for (const entry of userPaths) {
|
|
const lower = entry.path.toLowerCase();
|
|
if (!seen.has(lower)) {
|
|
seen.add(lower);
|
|
merged.push({ ...entry, source: t('merge.user'), displayIndex: merged.length });
|
|
}
|
|
}
|
|
|
|
if (!searchQuery) return merged;
|
|
const q = searchQuery.toLowerCase();
|
|
return merged.filter((r) => r.path.toLowerCase().includes(q));
|
|
}, [sysPaths, userPaths, searchQuery, t]);
|
|
|
|
return (
|
|
<div className="flex-1 overflow-auto">
|
|
<table className="w-full border-collapse">
|
|
<thead>
|
|
<tr
|
|
className="sticky top-0 z-10 text-left text-xs uppercase"
|
|
style={{ backgroundColor: 'var(--app-list-alt)', color: 'var(--app-fg)' }}
|
|
>
|
|
<th className="w-10 px-2 py-1">#</th>
|
|
<th className="px-2 py-1">{t('dialog.pathLabel')}</th>
|
|
<th className="w-16 px-2 py-1">{t('merge.source')}</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
{allPaths.map(({ path, enabled, source, displayIndex }, rowIdx) => {
|
|
const textColor = enabled ? 'var(--app-fg)' : '#6b7280';
|
|
const textDecoration = enabled ? 'none' : 'line-through';
|
|
const opacity = enabled ? 1 : 0.6;
|
|
|
|
return (
|
|
<tr
|
|
key={`${source}-${displayIndex}`}
|
|
style={{
|
|
backgroundColor: rowIdx % 2 === 0 ? 'var(--app-list-bg)' : 'var(--app-list-alt)',
|
|
color: 'var(--app-fg)',
|
|
}}
|
|
>
|
|
<td className="px-2 py-0.5 text-xs opacity-50">{rowIdx + 1}</td>
|
|
<td
|
|
className="px-2 py-0.5 text-sm"
|
|
style={{ color: textColor, textDecoration, opacity }}
|
|
>
|
|
{path}
|
|
</td>
|
|
<td className="px-2 py-0.5 text-xs opacity-60">{source}</td>
|
|
</tr>
|
|
);
|
|
})}
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
);
|
|
}
|