- 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
181 lines
4.9 KiB
TypeScript
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>
|
|
);
|
|
}
|