blob: 290da438b163bdb7782cf434d4ce83e7eb8cb57c [file] [log] [blame]
gio5f2f1002025-03-20 18:38:48 +04001import { useNodes } from "@xyflow/react";
2import { AppNode, nodeLabel } from "@/lib/state";
3import { NodeDetails } from "@/components/node-details";
4import { Accordion, AccordionContent, AccordionTrigger } from "./ui/accordion";
5import { AccordionItem } from "@radix-ui/react-accordion";
6import { useMemo, useState } from "react";
7import { Icon } from "./icon";
8
9function unique<T>(v: T, i: number, a: T[]) {
10 return a.indexOf(v) === i;
11}
12
13export function Details() {
14 const nodes = useNodes<AppNode>();
15 const [open, setOpen] = useState<string[]>([]);
16 const selected = useMemo(() => nodes.filter((n) => n.selected).map((n) => n.id), [nodes]);
17 const all = useMemo(() => open.concat(selected).filter(unique), [open, selected]);
18 return (
19 <Accordion type="multiple" value={all} onValueChange={(v) => setOpen(v)}>
20 {nodes.map((n) => (
21 <AccordionItem key={n.id} value={n.id} className="px-3">
22 <AccordionTrigger>
23 <div className="flex flex-row space-x-2">
24 {Icon(n.type)}
25 <span>{nodeLabel(n)}</span>
26 </div>
27 </AccordionTrigger>
28 <AccordionContent>
29 <NodeDetails {...n} />
30 </AccordionContent>
31 </AccordionItem>
32 ))}
33 </Accordion>
34 );
35}