Refactor status icons to use a shared ProgressCircle SVG base with pie-arc fill. Backlog, Todo, InProgress, InReview, Done, Cancelled, Blocked all use the unified component. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
179 lines
4.8 KiB
TypeScript
179 lines
4.8 KiB
TypeScript
import type { IssueStatus } from "@multica/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",
|
|
}: {
|
|
status: IssueStatus;
|
|
className?: string;
|
|
}) {
|
|
const cfg = STATUS_CONFIG[status];
|
|
const Renderer = STATUS_RENDERERS[status];
|
|
|
|
return (
|
|
<svg
|
|
viewBox="0 0 14 14"
|
|
fill="none"
|
|
className={`${className} ${cfg.iconColor} shrink-0`}
|
|
>
|
|
<Renderer />
|
|
</svg>
|
|
);
|
|
}
|