| gio | 5f2f100 | 2025-03-20 18:38:48 +0400 | [diff] [blame] | 1 | import { Button } from "./components/ui/button"; |
| 2 | import { AppNode, AppState, Message, nodeLabel, useMessages, useStateStore } from "./lib/state"; |
| 3 | import { useCallback, useEffect, useState } from "react"; |
| 4 | import { useNodes } from "@xyflow/react"; |
| 5 | import { Accordion, AccordionContent, AccordionItem, AccordionTrigger } from "./components/ui/accordion"; |
| 6 | import { Badge } from "./components/ui/badge"; |
| 7 | |
| 8 | export function Messages() { |
| 9 | const store = useStateStore(); |
| 10 | const nodes = useNodes<AppNode>(); |
| gio | 6cf8c27 | 2025-05-08 09:01:38 +0000 | [diff] [blame] | 11 | const [nodeMap, setNodeMap] = useState<Map<string, AppNode>>(new Map()); |
| gio | 5f2f100 | 2025-03-20 18:38:48 +0400 | [diff] [blame] | 12 | useEffect(() => { |
| 13 | setNodeMap(new Map(nodes.map((n) => [n.id, n]))); |
| 14 | }, [nodes, setNodeMap]); |
| 15 | const onClick = useCallback((fn?: (state: AppState) => void) => { |
| 16 | return () => { |
| 17 | if (fn) { |
| 18 | fn(store); |
| 19 | } |
| 20 | }; |
| 21 | }, [store]); |
| 22 | const messages = useMessages(); |
| 23 | const [grouped, setGrouped] = useState<Map<string, Message[]>>(new Map()); |
| 24 | useEffect(() => { |
| 25 | const g = new Map<string, Message[]>(); |
| 26 | messages.forEach((m) => { |
| 27 | const id = m.nodeId || "global"; |
| 28 | const existing: Message[] = g.get(id) || []; |
| 29 | existing.push(m); |
| 30 | g.set(id, existing); |
| 31 | }); |
| 32 | setGrouped(g); |
| 33 | }, [messages, setGrouped]); |
| 34 | const [open, setOpen] = useState<string[]>([...grouped.keys()]); |
| 35 | useEffect(() => { |
| 36 | // TODO(gio): do not reopen closed ones |
| 37 | setOpen([...grouped.keys()]); |
| 38 | }, [grouped, setOpen]); |
| 39 | return ( |
| gio | 6cf8c27 | 2025-05-08 09:01:38 +0000 | [diff] [blame] | 40 | <Accordion type="multiple" value={open} onValueChange={(v) => setOpen(v)}> |
| 41 | {[...grouped.entries()].map(([id, messages]) => ( |
| 42 | <AccordionItem key={id} value={id}> |
| 43 | <AccordionTrigger className="flex flex-row-reverse !space-x-4 !justify-end"> |
| 44 | <Badge>{messages.length}</Badge> |
| 45 | <div>{id === "global" ? "Global" : nodeLabel(nodeMap.get(id)!)}</div> |
| 46 | </AccordionTrigger> |
| 47 | <AccordionContent> |
| 48 | <div className="flex flex-col space-y-1"> |
| 49 | {messages.map((m) => ( |
| 50 | <Button key={m.id} variant="ghost" style={{ justifyContent: "flex-start" }} onMouseOver={onClick(m.onHighlight)} onMouseLeave={onClick(m.onLooseHighlight)} onClick={onClick(m.onClick)}>{m.message}</Button> |
| 51 | ))} |
| 52 | </div> |
| 53 | </AccordionContent> |
| 54 | </AccordionItem> |
| 55 | ))} |
| 56 | </Accordion> |
| gio | 5f2f100 | 2025-03-20 18:38:48 +0400 | [diff] [blame] | 57 | ) |
| 58 | } |