Upgrade /my-issues from a simple accordion to a full-featured view matching /issues — kanban board, list view, status/priority filtering, sorting, and display settings, scoped to the user's own issues. Key changes: - Extract view store factory (createIssueViewStore) using zustand v5 vanilla createStore + React Context for shared component reuse - Create ViewStoreProvider + useViewStore/useViewStoreApi hooks - Decouple BoardView, BoardColumn, BoardCard, ListView from global useIssueViewStore — they now read from context - New independent persisted store for /my-issues (multica_my_issues_view) - Simplified MyIssuesHeader (no assignee/creator filters) - Pre-filter logic: assigned to me ∪ my agents ∪ created by me - Generalize workspace sync to clear filters on all registered stores - Fix existing debt: text-[10px] → text-xs, w-44 → w-auto, reduce unnecessary selector subscriptions in both headers Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
205 lines
6.7 KiB
TypeScript
205 lines
6.7 KiB
TypeScript
"use client";
|
|
|
|
import { useCallback, memo } from "react";
|
|
import Link from "next/link";
|
|
import { useSortable } from "@dnd-kit/sortable";
|
|
import { CSS } from "@dnd-kit/utilities";
|
|
import { toast } from "sonner";
|
|
import type { Issue, UpdateIssueRequest } from "@/shared/types";
|
|
import { CalendarDays } from "lucide-react";
|
|
import { ActorAvatar } from "@/components/common/actor-avatar";
|
|
import { api } from "@/shared/api";
|
|
import { useIssueStore } from "@/features/issues/store";
|
|
import { PriorityIcon } from "./priority-icon";
|
|
import { PriorityPicker, AssigneePicker, DueDatePicker } from "./pickers";
|
|
import { PRIORITY_CONFIG } from "@/features/issues/config";
|
|
import type { CardProperties } from "@/features/issues/stores/view-store";
|
|
import { useViewStore } from "@/features/issues/stores/view-store-context";
|
|
|
|
function formatDate(date: string): string {
|
|
return new Date(date).toLocaleDateString("en-US", {
|
|
month: "short",
|
|
day: "numeric",
|
|
});
|
|
}
|
|
|
|
/** Stops event from bubbling to Link/drag handlers */
|
|
function PickerWrapper({ children }: { children: React.ReactNode }) {
|
|
const stop = (e: React.SyntheticEvent) => {
|
|
e.stopPropagation();
|
|
e.preventDefault();
|
|
};
|
|
return (
|
|
<div onClick={stop} onMouseDown={stop} onPointerDown={stop}>
|
|
{children}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export const BoardCardContent = memo(function BoardCardContent({
|
|
issue,
|
|
editable = false,
|
|
}: {
|
|
issue: Issue;
|
|
editable?: boolean;
|
|
}) {
|
|
const storeProperties = useViewStore((s) => s.cardProperties);
|
|
const priorityCfg = PRIORITY_CONFIG[issue.priority];
|
|
|
|
const handleUpdate = useCallback(
|
|
(updates: Partial<UpdateIssueRequest>) => {
|
|
const prev = { ...issue };
|
|
useIssueStore.getState().updateIssue(issue.id, updates);
|
|
api.updateIssue(issue.id, updates).catch(() => {
|
|
useIssueStore.getState().updateIssue(issue.id, prev);
|
|
toast.error("Failed to update issue");
|
|
});
|
|
},
|
|
[issue],
|
|
);
|
|
|
|
const showPriority = storeProperties.priority;
|
|
const showDescription = storeProperties.description && issue.description;
|
|
const showAssignee = storeProperties.assignee && issue.assignee_type && issue.assignee_id;
|
|
const showDueDate = storeProperties.dueDate && issue.due_date;
|
|
const showBottom = showAssignee || showDueDate;
|
|
|
|
return (
|
|
<div className="rounded-lg border bg-card p-3.5 shadow-[0_1px_2px_0_rgba(0,0,0,0.03)] transition-shadow group-hover:shadow-sm">
|
|
{/* Row 1: Identifier */}
|
|
<p className="text-xs text-muted-foreground">{issue.identifier}</p>
|
|
|
|
{/* Row 2: Title */}
|
|
<p className="mt-1 text-sm font-medium leading-snug line-clamp-2">
|
|
{issue.title}
|
|
</p>
|
|
|
|
{/* Description */}
|
|
{showDescription && (
|
|
<p className="mt-1 text-xs text-muted-foreground line-clamp-1">
|
|
{issue.description}
|
|
</p>
|
|
)}
|
|
|
|
{/* Row 3: Assignee, priority badge, due date */}
|
|
{(showAssignee || showPriority || showDueDate) && (
|
|
<div className="mt-3 flex items-center gap-2">
|
|
{showAssignee &&
|
|
(editable ? (
|
|
<PickerWrapper>
|
|
<AssigneePicker
|
|
assigneeType={issue.assignee_type}
|
|
assigneeId={issue.assignee_id}
|
|
onUpdate={handleUpdate}
|
|
trigger={
|
|
<ActorAvatar
|
|
actorType={issue.assignee_type!}
|
|
actorId={issue.assignee_id!}
|
|
size={22}
|
|
/>
|
|
}
|
|
/>
|
|
</PickerWrapper>
|
|
) : (
|
|
<ActorAvatar
|
|
actorType={issue.assignee_type!}
|
|
actorId={issue.assignee_id!}
|
|
size={22}
|
|
/>
|
|
))}
|
|
{showPriority &&
|
|
(editable ? (
|
|
<PickerWrapper>
|
|
<PriorityPicker
|
|
priority={issue.priority}
|
|
onUpdate={handleUpdate}
|
|
trigger={
|
|
<span className={`inline-flex items-center gap-1 rounded px-1.5 py-0.5 text-xs font-medium ${priorityCfg.badgeBg} ${priorityCfg.badgeText}`}>
|
|
<PriorityIcon priority={issue.priority} className="h-3 w-3" inheritColor />
|
|
{priorityCfg.label}
|
|
</span>
|
|
}
|
|
/>
|
|
</PickerWrapper>
|
|
) : (
|
|
<span className={`inline-flex items-center gap-1 rounded px-1.5 py-0.5 text-xs font-medium ${priorityCfg.badgeBg} ${priorityCfg.badgeText}`}>
|
|
<PriorityIcon priority={issue.priority} className="h-3 w-3" inheritColor />
|
|
{priorityCfg.label}
|
|
</span>
|
|
))}
|
|
{showDueDate && (
|
|
<div className="ml-auto">
|
|
{editable ? (
|
|
<PickerWrapper>
|
|
<DueDatePicker
|
|
dueDate={issue.due_date}
|
|
onUpdate={handleUpdate}
|
|
trigger={
|
|
<span
|
|
className={`flex items-center gap-1 text-xs ${
|
|
new Date(issue.due_date!) < new Date()
|
|
? "text-destructive"
|
|
: "text-muted-foreground"
|
|
}`}
|
|
>
|
|
<CalendarDays className="size-3" />
|
|
{formatDate(issue.due_date!)}
|
|
</span>
|
|
}
|
|
/>
|
|
</PickerWrapper>
|
|
) : (
|
|
<span
|
|
className={`flex items-center gap-1 text-xs ${
|
|
new Date(issue.due_date!) < new Date()
|
|
? "text-destructive"
|
|
: "text-muted-foreground"
|
|
}`}
|
|
>
|
|
<CalendarDays className="size-3" />
|
|
{formatDate(issue.due_date!)}
|
|
</span>
|
|
)}
|
|
</div>
|
|
)}
|
|
</div>
|
|
)}
|
|
</div>
|
|
);
|
|
});
|
|
|
|
export const DraggableBoardCard = memo(function DraggableBoardCard({ issue }: { issue: Issue }) {
|
|
const {
|
|
attributes,
|
|
listeners,
|
|
setNodeRef,
|
|
transform,
|
|
transition,
|
|
isDragging,
|
|
} = useSortable({
|
|
id: issue.id,
|
|
data: { status: issue.status },
|
|
});
|
|
|
|
const style = {
|
|
transform: CSS.Transform.toString(transform),
|
|
transition,
|
|
};
|
|
|
|
return (
|
|
<div
|
|
ref={setNodeRef}
|
|
style={style}
|
|
{...attributes}
|
|
{...listeners}
|
|
className={isDragging ? "opacity-30" : ""}
|
|
>
|
|
<Link
|
|
href={`/issues/${issue.id}`}
|
|
className={`group block transition-colors ${isDragging ? "pointer-events-none" : ""}`}
|
|
>
|
|
<BoardCardContent issue={issue} editable />
|
|
</Link>
|
|
</div>
|
|
);
|
|
});
|