multica/apps/web/features/issues/components/status-icon.tsx
Jiayuan 8a61c94b98 feat(ui): restyle issue status and priority with colored badges
- Status labels use colored pill badges (solid bg for active, muted for inactive)
- Board columns have tinted backgrounds matching their status color
- Priority badges use orange (--priority) design token for clear distinction from status
- Issue cards restructured: identifier, title, then assignee/priority/date row
- Agent avatar default color changed from blue to gray
- New Issue button in header changed to solid/primary style
- Reduced hover shadow on board cards
- Added inheritColor prop to StatusIcon and PriorityIcon for badge use
2026-03-31 03:26:43 +08:00

181 lines
4.9 KiB
TypeScript

import type { IssueStatus } from "@/shared/types";
import { STATUS_CONFIG } from "@/features/issues/config";
// ---------------------------------------------------------------------------
// Geometry constants (viewBox 0 0 14 14, center 7,7)
// ---------------------------------------------------------------------------
const CX = 7;
const CY = 7;
const OUTER_R = 6;
const FILL_R = 3.5;
// ---------------------------------------------------------------------------
// Helpers
// ---------------------------------------------------------------------------
/** Build a pie-wedge SVG path from 12 o'clock, clockwise */
function piePath(cx: number, cy: number, r: number, progress: number): string {
const angle = 2 * Math.PI * progress;
const endX = cx + r * Math.sin(angle);
const endY = cy - r * Math.cos(angle);
const largeArc = progress > 0.5 ? 1 : 0;
return `M${cx},${cy} L${cx},${cy - r} A${r},${r} 0 ${largeArc},1 ${endX},${endY} Z`;
}
// ---------------------------------------------------------------------------
// Base component — dashed outer ring + pie fill + optional center icon
// ---------------------------------------------------------------------------
function ProgressCircle({
progress,
children,
}: {
progress: number;
children?: React.ReactNode;
}) {
return (
<>
{/* Outer dashed ring */}
<circle
cx={CX}
cy={CY}
r={OUTER_R}
fill="none"
stroke="currentColor"
strokeWidth={1.5}
strokeDasharray="3.14 0"
strokeDashoffset={-0.7}
/>
{/* Progress fill */}
{progress === 1 ? (
<circle cx={CX} cy={CY} r={OUTER_R} fill="currentColor" />
) : progress > 0 ? (
<path d={piePath(CX, CY, FILL_R, progress)} fill="currentColor" />
) : null}
{children}
</>
);
}
// ---------------------------------------------------------------------------
// Per-status renderers
// ---------------------------------------------------------------------------
/** 16 small dots arranged in a ring */
function BacklogIcon() {
const count = 16;
const dotR = 0.55;
return (
<g>
{Array.from({ length: count }, (_, i) => {
const angle = (i / count) * Math.PI * 2 - Math.PI / 2;
return (
<circle
key={i}
cx={CX + OUTER_R * Math.cos(angle)}
cy={CY + OUTER_R * Math.sin(angle)}
r={dotR}
fill="currentColor"
/>
);
})}
</g>
);
}
function TodoIcon() {
return <ProgressCircle progress={0} />;
}
function InProgressIcon() {
return <ProgressCircle progress={0.5} />;
}
function InReviewIcon() {
return <ProgressCircle progress={0.75} />;
}
function DoneIcon() {
return (
<ProgressCircle progress={1}>
<path
d="M10.951 4.24896C11.283 4.58091 11.283 5.11909 10.951 5.45104L5.95104 10.451C5.61909 10.783 5.0809 10.783 4.74896 10.451L2.74896 8.45104C2.41701 8.11909 2.41701 7.5809 2.74896 7.24896C3.0809 6.91701 3.61909 6.91701 3.95104 7.24896L5.35 8.64792L9.74896 4.24896C10.0809 3.91701 10.6191 3.91701 10.951 4.24896Z"
fill="white"
stroke="none"
/>
</ProgressCircle>
);
}
/** Outer ring + prohibition slash (🚫 style) */
function BlockedIcon() {
return (
<ProgressCircle progress={0}>
<line
x1={CX + FILL_R * Math.cos(Math.PI * 0.75)}
y1={CY - FILL_R * Math.sin(Math.PI * 0.75)}
x2={CX + FILL_R * Math.cos(-Math.PI * 0.25)}
y2={CY - FILL_R * Math.sin(-Math.PI * 0.25)}
stroke="currentColor"
strokeWidth={1.5}
strokeLinecap="round"
/>
</ProgressCircle>
);
}
function CancelledIcon() {
return (
<ProgressCircle progress={0}>
<path
d="M5 5 L9 9 M9 5 L5 9"
fill="none"
stroke="currentColor"
strokeWidth={1.5}
strokeLinecap="round"
/>
</ProgressCircle>
);
}
// ---------------------------------------------------------------------------
// Renderer map
// ---------------------------------------------------------------------------
const STATUS_RENDERERS: Record<IssueStatus, () => React.ReactNode> = {
backlog: BacklogIcon,
todo: TodoIcon,
in_progress: InProgressIcon,
in_review: InReviewIcon,
done: DoneIcon,
blocked: BlockedIcon,
cancelled: CancelledIcon,
};
// ---------------------------------------------------------------------------
// Public component
// ---------------------------------------------------------------------------
export function StatusIcon({
status,
className = "h-4 w-4",
inheritColor = false,
}: {
status: IssueStatus;
className?: string;
inheritColor?: boolean;
}) {
const cfg = STATUS_CONFIG[status];
const Renderer = STATUS_RENDERERS[status];
return (
<svg
viewBox="0 0 14 14"
fill="none"
className={`${className} ${inheritColor ? "" : cfg.iconColor} shrink-0`}
>
<Renderer />
</svg>
);
}