multica/apps/web/features/issues/components/status-icon.tsx
Naiyuan Qing 2cf088ddf6 feat: resizable sidebar, issue detail rewrite, package consolidation
- Add drag-to-resize sidebar with localStorage persistence
- Rewrite issue detail page with Tiptap rich text editor, due date picker, acceptance criteria
- Redesign create-issue modal with pill-based property toolbar and expand/collapse
- Consolidate @multica/sdk and @multica/types into apps/web/shared/
- Simplify auth: remove verification codes, PATs, email service (dev-only login)
- Add 401 unauthorized handler to redirect expired sessions to login
- Fix due date format to send full RFC3339 timestamps
- Increase description editor debounce to 1500ms
- Remove arbitrary Tailwind values in create-issue modal
- Renumber migrations (inbox_actor 012→009), remove unused migrations
- UI polish across agents, settings, inbox, knowledge-base pages

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-26 16:47:04 +08:00

179 lines
4.8 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",
}: {
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>
);
}