AirShelf/components/Icon.tsx
iye f420af2069
All checks were successful
Build and Deploy / build-and-deploy (push) Successful in 6s
chore: 全量推送 · 累积页面改动 + Next.js 工程骨架 + v1/ 归档 + 文档
页面 (电商AI平台/)
- account / team / settings / index / products / projects: 累积迭代
- restraint.css: 设计 token 补充
- login.html / register.html: 新增登录注册页
- _ARCHIVE.md: 归档说明

Next.js 工程骨架
- app/ + components/: 新一代 SPA 雏形 (page / layout / sidebar / topbar / GridBg / Icon)
- package.json / package-lock.json / next.config.mjs / tsconfig.json / postcss.config.mjs / next-env.d.ts

历史归档 / 文档
- v1/: 原 V1 静态稿镜像 (含 mockup-A/B/C)
- PRD.md / deployment-guide.md / _check.html
- ui参考/ / screenshots/

杂项
- .gitignore 调整
- 删除根 README.md

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-21 21:16:46 +08:00

178 lines
4.6 KiB
TypeScript

import type { SVGProps } from "react";
type IconName =
| "home"
| "play"
| "folder"
| "tile"
| "bars"
| "bars2"
| "key"
| "cog"
| "chev-down"
| "chev-right"
| "search"
| "bell"
| "help"
| "doc"
| "up"
| "plus"
| "flame"
| "check"
| "x"
| "play-tri"
| "rotate"
| "more"
| "wallet"
| "coin"
| "download"
| "team"
| "lightbulb"
| "sparkles"
| "info"
| "arrow-right";
const PATHS: Record<IconName, React.ReactNode> = {
home: (
<>
<path d="M3 12 12 3l9 9" />
<path d="M5 10v10h14V10" />
</>
),
play: <polygon points="6 4 20 12 6 20" fill="currentColor" stroke="none" />,
folder: (
<>
<rect x="3" y="4" width="18" height="16" rx="2" />
<path d="M3 10h18" />
</>
),
tile: <path d="M3 6h18M3 12h18M3 18h18" />,
bars: (
<>
<rect x="3" y="4" width="6" height="16" />
<rect x="11" y="4" width="4" height="16" />
<rect x="17" y="6" width="4" height="14" />
</>
),
bars2: <path d="M3 21V9M9 21V5M15 21v-8M21 21V11" />,
key: (
<>
<circle cx="9" cy="12" r="6" />
<path d="m15 12 6 0M19 9v6" />
</>
),
cog: (
<>
<circle cx="12" cy="12" r="3" />
<path d="M19.4 15a1.7 1.7 0 0 0 .3 1.8 2 2 0 0 1-2.8 2.8 1.7 1.7 0 0 0-1.8-.3 1.7 1.7 0 0 0-1 1.5 2 2 0 0 1-4 0 1.7 1.7 0 0 0-1.1-1.5 1.7 1.7 0 0 0-1.8.3 2 2 0 0 1-2.8-2.8 1.7 1.7 0 0 0 .3-1.8 1.7 1.7 0 0 0-1.5-1 2 2 0 0 1 0-4 1.7 1.7 0 0 0 1.5-1.1 1.7 1.7 0 0 0-.3-1.8 2 2 0 0 1 2.8-2.8 1.7 1.7 0 0 0 1.8.3 1.7 1.7 0 0 0 1-1.5 2 2 0 0 1 4 0 1.7 1.7 0 0 0 1 1.5 1.7 1.7 0 0 0 1.8-.3 2 2 0 0 1 2.8 2.8 1.7 1.7 0 0 0-.3 1.8 1.7 1.7 0 0 0 1.5 1 2 2 0 0 1 0 4 1.7 1.7 0 0 0-1.5 1.1Z" />
</>
),
"chev-down": <path d="m6 9 6 6 6-6" />,
"chev-right": <path d="m9 6 6 6-6 6" />,
search: (
<>
<circle cx="11" cy="11" r="7" />
<path d="m21 21-4.3-4.3" />
</>
),
bell: <path d="M6 8a6 6 0 0 1 12 0c0 7 3 9 3 9H3s3-2 3-9M10 21a2 2 0 0 0 4 0" />,
help: (
<>
<circle cx="12" cy="12" r="9" />
<path d="M9.5 9a2.5 2.5 0 1 1 4 2.2c-.7.5-1.5 1-1.5 2v.3M12 17h.01" />
</>
),
doc: (
<>
<path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z" />
<path d="M14 2v6h6M9 13h6M9 17h6" />
</>
),
up: <path d="M12 19V5M5 12l7-7 7 7" />,
plus: <path d="M12 5v14M5 12h14" />,
flame: <path fill="currentColor" stroke="none" d="M12 2c1 3 4 5 4 9a4 4 0 0 1-4 4 4 4 0 0 1-4-4 5 5 0 0 1 1.5-3.5C10.5 6 11.5 4 12 2zm-1 13c0 2 1 3 1 5 1-1 3-2 3-5 0-1.5-1-2-2-3-1 1-2 2-2 3z" />,
check: <path d="M4 12l5 5L20 6" />,
x: <path d="M5 5l14 14M19 5L5 19" />,
"play-tri": <path fill="currentColor" stroke="none" d="M5 4l14 8-14 8z" />,
rotate: (
<>
<path d="M4 12a8 8 0 0 1 14-5.5L21 9" />
<path d="M21 4v5h-5" />
<path d="M20 12a8 8 0 0 1-14 5.5L3 15" />
<path d="M3 20v-5h5" />
</>
),
more: (
<>
<circle cx="5" cy="12" r="1.5" fill="currentColor" />
<circle cx="12" cy="12" r="1.5" fill="currentColor" />
<circle cx="19" cy="12" r="1.5" fill="currentColor" />
</>
),
wallet: (
<>
<rect x="3" y="6" width="18" height="13" rx="2" />
<path d="M3 10h18M16 14h2" />
</>
),
coin: (
<>
<circle cx="12" cy="12" r="9" />
<path d="M12 7v10M9 10h4a1.5 1.5 0 0 1 0 3h-3a1.5 1.5 0 0 0 0 3h4" />
</>
),
download: <path d="M12 4v12m0 0l-5-5m5 5l5-5M4 20h16" />,
team: (
<>
<circle cx="9" cy="9" r="3" />
<path d="M3 20c0-3 3-5 6-5s6 2 6 5" />
<circle cx="17" cy="10" r="2.4" />
<path d="M21 19c0-2-1.6-4-4-4-.6 0-1.2.2-1.7.4" />
</>
),
lightbulb: (
<>
<path d="M9 18h6" />
<path d="M10 22h4" />
<path d="M15.09 14c.18-.98.65-1.74 1.41-2.5A4.65 4.65 0 0 0 18 8 6 6 0 0 0 6 8c0 1 .23 2.23 1.5 3.5A4.61 4.61 0 0 1 8.91 14" />
</>
),
sparkles: (
<>
<path d="M12 3l1.7 4.6L18 9l-4.3 1.4L12 15l-1.7-4.6L6 9l4.3-1.4L12 3z" />
<path d="M19 16l.85 2.3L22 19l-2.15.7L19 22l-.85-2.3L16 19l2.15-.7L19 16z" />
</>
),
info: (
<>
<circle cx="12" cy="12" r="9" />
<path d="M12 16v-4M12 8h.01" />
</>
),
"arrow-right": <path d="M5 12h14M13 6l6 6-6 6" />,
};
interface Props extends Omit<SVGProps<SVGSVGElement>, "name"> {
name: IconName;
size?: number;
strokeWidth?: number;
}
export default function Icon({ name, size = 16, strokeWidth = 1.5, ...rest }: Props) {
return (
<svg
viewBox="0 0 24 24"
width={size}
height={size}
fill="none"
stroke="currentColor"
strokeWidth={strokeWidth}
strokeLinecap="round"
strokeLinejoin="round"
{...rest}
>
{PATHS[name]}
</svg>
);
}