diff options
| author | Adam <[email protected]> | 2025-10-30 07:26:06 -0500 |
|---|---|---|
| committer | Adam <[email protected]> | 2025-10-30 12:02:49 -0500 |
| commit | 30f4c2cf4c6c01339434c617fb9d930f6e960883 (patch) | |
| tree | db5da342a227724e11609e05f9e3c1fd6e2e7741 /packages/desktop/src/components | |
| parent | 3541fdcb2019676fb82351e909a8e9b740cb8237 (diff) | |
| download | opencode-30f4c2cf4c6c01339434c617fb9d930f6e960883.tar.gz opencode-30f4c2cf4c6c01339434c617fb9d930f6e960883.zip | |
wip: desktop work
Diffstat (limited to 'packages/desktop/src/components')
| -rw-r--r-- | packages/desktop/src/components/diff-changes.tsx | 20 | ||||
| -rw-r--r-- | packages/desktop/src/components/message.tsx | 253 | ||||
| -rw-r--r-- | packages/desktop/src/components/session-timeline.tsx | 536 |
3 files changed, 36 insertions, 773 deletions
diff --git a/packages/desktop/src/components/diff-changes.tsx b/packages/desktop/src/components/diff-changes.tsx deleted file mode 100644 index 3b633f70f..000000000 --- a/packages/desktop/src/components/diff-changes.tsx +++ /dev/null @@ -1,20 +0,0 @@ -import { FileDiff } from "@opencode-ai/sdk" -import { createMemo, Show } from "solid-js" - -export function DiffChanges(props: { diff: FileDiff | FileDiff[] }) { - const additions = createMemo(() => - Array.isArray(props.diff) ? props.diff.reduce((acc, diff) => acc + (diff.additions ?? 0), 0) : props.diff.additions, - ) - const deletions = createMemo(() => - Array.isArray(props.diff) ? props.diff.reduce((acc, diff) => acc + (diff.deletions ?? 0), 0) : props.diff.deletions, - ) - const total = createMemo(() => additions() + deletions()) - return ( - <Show when={total() > 0}> - <div class="flex gap-2 justify-end items-center"> - <span class="text-12-mono text-right text-text-diff-add-base">{`+${additions()}`}</span> - <span class="text-12-mono text-right text-text-diff-delete-base">{`-${deletions()}`}</span> - </div> - </Show> - ) -} diff --git a/packages/desktop/src/components/message.tsx b/packages/desktop/src/components/message.tsx index 589ca3118..9e9e06d35 100644 --- a/packages/desktop/src/components/message.tsx +++ b/packages/desktop/src/components/message.tsx @@ -1,238 +1,57 @@ -import type { Part, ReasoningPart, TextPart, ToolPart, Message, AssistantMessage, UserMessage } from "@opencode-ai/sdk" -import { children, Component, createMemo, For, Match, Show, Switch, type JSX } from "solid-js" +import type { Part, TextPart, ToolPart, Message } from "@opencode-ai/sdk" +import { createMemo, For, Show } from "solid-js" import { Dynamic } from "solid-js/web" import { Markdown } from "./markdown" -import { Checkbox, Collapsible, Diff, Icon, IconProps } from "@opencode-ai/ui" +import { Checkbox, Diff, Icon } from "@opencode-ai/ui" +import { Message as MessageDisplay, registerPartComponent } from "@opencode-ai/ui" +import { BasicTool, GenericTool, ToolRegistry, DiffChanges } from "@opencode-ai/ui" import { getDirectory, getFilename } from "@/utils" -import type { Tool } from "opencode/tool/tool" -import type { ReadTool } from "opencode/tool/read" -import type { ListTool } from "opencode/tool/ls" -import type { GlobTool } from "opencode/tool/glob" -import type { GrepTool } from "opencode/tool/grep" -import type { WebFetchTool } from "opencode/tool/webfetch" -import type { TaskTool } from "opencode/tool/task" -import type { BashTool } from "opencode/tool/bash" -import type { EditTool } from "opencode/tool/edit" -import type { WriteTool } from "opencode/tool/write" -import type { TodoWriteTool } from "opencode/tool/todo" -import { DiffChanges } from "./diff-changes" export function Message(props: { message: Message; parts: Part[] }) { - return ( - <Switch> - <Match when={props.message.role === "user" && props.message}> - {(userMessage) => <UserMessage message={userMessage()} parts={props.parts} />} - </Match> - <Match when={props.message.role === "assistant" && props.message}> - {(assistantMessage) => <AssistantMessage message={assistantMessage()} parts={props.parts} />} - </Match> - </Switch> - ) + return <MessageDisplay message={props.message} parts={props.parts} /> } -function AssistantMessage(props: { message: AssistantMessage; parts: Part[] }) { - const filteredParts = createMemo(() => { - return props.parts?.filter((x) => { - if (x.type === "reasoning") return false - return x.type !== "tool" || x.tool !== "todoread" - }) - }) +registerPartComponent("text", function TextPartDisplay(props) { + const part = props.part as TextPart return ( - <div class="w-full flex flex-col items-start gap-4"> - <For each={filteredParts()}>{(part) => <Part part={part} message={props.message} />}</For> - </div> - ) -} - -function UserMessage(props: { message: UserMessage; parts: Part[] }) { - const text = createMemo(() => - props.parts - ?.filter((p) => p.type === "text" && !p.synthetic) - ?.map((p) => (p as TextPart).text) - ?.join(""), - ) - return <div class="text-12-regular text-text-base line-clamp-3">{text()}</div> -} - -export function Part(props: { part: Part; message: Message; hideDetails?: boolean }) { - const component = createMemo(() => PART_MAPPING[props.part.type as keyof typeof PART_MAPPING]) - return ( - <Show when={component()}> - <Dynamic - component={component()} - part={props.part as any} - message={props.message} - hideDetails={props.hideDetails} - /> - </Show> - ) -} - -const PART_MAPPING = { - text: TextPart, - tool: ToolPart, - reasoning: ReasoningPart, -} - -function ReasoningPart(props: { part: ReasoningPart; message: Message }) { - return ( - <Show when={props.part.text.trim()}> - <Markdown text={props.part.text.trim()} /> + <Show when={part.text.trim()}> + <Markdown text={part.text.trim()} /> </Show> ) -} +}) -function TextPart(props: { part: TextPart; message: Message }) { +registerPartComponent("reasoning", function ReasoningPartDisplay(props) { + const part = props.part as any return ( - <Show when={props.part.text.trim()}> - <Markdown text={props.part.text.trim()} /> + <Show when={part.text.trim()}> + <Markdown text={part.text.trim()} /> </Show> ) -} +}) -function ToolPart(props: { part: ToolPart; message: Message; hideDetails?: boolean }) { +registerPartComponent("tool", function ToolPartDisplay(props) { + const part = props.part as ToolPart const component = createMemo(() => { - const render = ToolRegistry.render(props.part.tool) ?? GenericTool - const metadata = props.part.state.status === "pending" ? {} : (props.part.state.metadata ?? {}) - const input = props.part.state.status === "completed" ? props.part.state.input : {} + const render = ToolRegistry.render(part.tool) ?? GenericTool + const metadata = part.state.status === "pending" ? {} : (part.state.metadata ?? {}) + const input = part.state.status === "completed" ? part.state.input : {} return ( <Dynamic component={render} input={input} - tool={props.part.tool} + tool={part.tool} metadata={metadata} - output={props.part.state.status === "completed" ? props.part.state.output : undefined} + output={part.state.status === "completed" ? part.state.output : undefined} hideDetails={props.hideDetails} /> ) }) return <Show when={component()}>{component()}</Show> -} - -type TriggerTitle = { - title: string - titleClass?: string - subtitle?: string - subtitleClass?: string - args?: string[] - argsClass?: string - action?: JSX.Element -} - -const isTriggerTitle = (val: any): val is TriggerTitle => { - return typeof val === "object" && val !== null && "title" in val && !(val instanceof Node) -} - -function BasicTool(props: { - icon: IconProps["name"] - trigger: TriggerTitle | JSX.Element - children?: JSX.Element - hideDetails?: boolean -}) { - const resolved = children(() => props.children) - return ( - <Collapsible> - <Collapsible.Trigger> - <div class="w-full flex items-center self-stretch gap-5 justify-between"> - <div class="w-full flex items-center self-stretch gap-5"> - <Icon name={props.icon} size="small" class="shrink-0" /> - <div class="grow min-w-0"> - <Switch> - <Match when={isTriggerTitle(props.trigger) && props.trigger}> - {(trigger) => ( - <div class="w-full flex items-center gap-2 justify-between"> - <div class="flex items-center gap-2 whitespace-nowrap truncate"> - <span - classList={{ - "text-12-medium text-text-base": true, - [trigger().titleClass ?? ""]: !!trigger().titleClass, - }} - > - {trigger().title} - </span> - <Show when={trigger().subtitle}> - <span - classList={{ - "text-12-medium text-text-weak": true, - [trigger().subtitleClass ?? ""]: !!trigger().subtitleClass, - }} - > - {trigger().subtitle} - </span> - </Show> - <Show when={trigger().args?.length}> - <For each={trigger().args}> - {(arg) => ( - <span - classList={{ - "text-12-regular text-text-weak": true, - [trigger().argsClass ?? ""]: !!trigger().argsClass, - }} - > - {arg} - </span> - )} - </For> - </Show> - </div> - <Show when={trigger().action}>{trigger().action}</Show> - </div> - )} - </Match> - <Match when={true}>{props.trigger as JSX.Element}</Match> - </Switch> - </div> - </div> - <Show when={resolved() && !props.hideDetails}> - <Collapsible.Arrow /> - </Show> - </div> - </Collapsible.Trigger> - <Show when={resolved() && !props.hideDetails}> - <Collapsible.Content>{resolved()}</Collapsible.Content> - </Show> - </Collapsible> - // <> - // <Show when={props.part.state.status === "error"}>{props.part.state.error.replace("Error: ", "")}</Show> - // </> - ) -} - -function GenericTool(props: ToolProps<any>) { - return <BasicTool icon="mcp" trigger={{ title: props.tool }} hideDetails={props.hideDetails} /> -} - -type ToolProps<T extends Tool.Info> = { - input: Partial<Tool.InferParameters<T>> - metadata: Partial<Tool.InferMetadata<T>> - tool: string - output?: string - hideDetails?: boolean -} - -const ToolRegistry = (() => { - const state: Record< - string, - { - name: string - render?: Component<ToolProps<any>> - } - > = {} - function register<T extends Tool.Info>(input: { name: string; render?: Component<ToolProps<T>> }) { - state[input.name] = input - return input - } - return { - register, - render(name: string) { - return state[name]?.render - }, - } -})() +}) -ToolRegistry.register<typeof ReadTool>({ +ToolRegistry.register({ name: "read", render(props) { return ( @@ -244,7 +63,7 @@ ToolRegistry.register<typeof ReadTool>({ }, }) -ToolRegistry.register<typeof ListTool>({ +ToolRegistry.register({ name: "list", render(props) { return ( @@ -257,7 +76,7 @@ ToolRegistry.register<typeof ListTool>({ }, }) -ToolRegistry.register<typeof GlobTool>({ +ToolRegistry.register({ name: "glob", render(props) { return ( @@ -277,7 +96,7 @@ ToolRegistry.register<typeof GlobTool>({ }, }) -ToolRegistry.register<typeof GrepTool>({ +ToolRegistry.register({ name: "grep", render(props) { const args = [] @@ -300,7 +119,7 @@ ToolRegistry.register<typeof GrepTool>({ }, }) -ToolRegistry.register<typeof WebFetchTool>({ +ToolRegistry.register({ name: "webfetch", render(props) { return ( @@ -325,7 +144,7 @@ ToolRegistry.register<typeof WebFetchTool>({ }, }) -ToolRegistry.register<typeof TaskTool>({ +ToolRegistry.register({ name: "task", render(props) { return ( @@ -345,7 +164,7 @@ ToolRegistry.register<typeof TaskTool>({ }, }) -ToolRegistry.register<typeof BashTool>({ +ToolRegistry.register({ name: "bash", render(props) { return ( @@ -364,7 +183,7 @@ ToolRegistry.register<typeof BashTool>({ }, }) -ToolRegistry.register<typeof EditTool>({ +ToolRegistry.register({ name: "edit", render(props) { return ( @@ -402,7 +221,7 @@ ToolRegistry.register<typeof EditTool>({ }, }) -ToolRegistry.register<typeof WriteTool>({ +ToolRegistry.register({ name: "write", render(props) { return ( @@ -431,7 +250,7 @@ ToolRegistry.register<typeof WriteTool>({ }, }) -ToolRegistry.register<typeof TodoWriteTool>({ +ToolRegistry.register({ name: "todowrite", render(props) { return ( @@ -439,13 +258,13 @@ ToolRegistry.register<typeof TodoWriteTool>({ icon="checklist" trigger={{ title: "To-dos", - subtitle: `${props.input.todos?.filter((t) => t.status === "completed").length}/${props.input.todos?.length}`, + subtitle: `${props.input.todos?.filter((t: any) => t.status === "completed").length}/${props.input.todos?.length}`, }} > <Show when={props.input.todos?.length}> <div class="px-12 pt-2.5 pb-6 flex flex-col gap-2"> <For each={props.input.todos}> - {(todo) => ( + {(todo: any) => ( <Checkbox readOnly checked={todo.status === "completed"}> <div classList={{ "line-through text-text-weaker": todo.status === "completed" }}>{todo.content}</div> </Checkbox> diff --git a/packages/desktop/src/components/session-timeline.tsx b/packages/desktop/src/components/session-timeline.tsx deleted file mode 100644 index e1f3beae4..000000000 --- a/packages/desktop/src/components/session-timeline.tsx +++ /dev/null @@ -1,536 +0,0 @@ -import { Icon, Tooltip } from "@opencode-ai/ui" -import { Collapsible } from "@/ui" -import type { AssistantMessage, Message, Part, ToolPart } from "@opencode-ai/sdk" -import { DateTime } from "luxon" -import { - createSignal, - For, - Match, - splitProps, - Switch, - type ComponentProps, - type ParentProps, - createEffect, - createMemo, - Show, -} from "solid-js" -import { getFilename } from "@/utils" -import { Markdown } from "./markdown" -import { Code } from "./code" -import { createElementSize } from "@solid-primitives/resize-observer" -import { createScrollPosition } from "@solid-primitives/scroll" -import { ProgressCircle } from "./progress-circle" -import { pipe, sumBy } from "remeda" -import { useSync } from "@/context/sync" -import { useLocal } from "@/context/local" - -function Part(props: ParentProps & ComponentProps<"div">) { - const [local, others] = splitProps(props, ["class", "classList", "children"]) - return ( - <div - classList={{ - ...(local.classList ?? {}), - "h-6 flex items-center": true, - [local.class ?? ""]: !!local.class, - }} - {...others} - > - <p class="text-12-medium text-left">{local.children}</p> - </div> - ) -} - -function CollapsiblePart(props: { title: ParentProps["children"] } & ParentProps & ComponentProps<typeof Collapsible>) { - return ( - <Collapsible {...props}> - <Collapsible.Trigger class="peer/collapsible"> - <Part>{props.title}</Part> - </Collapsible.Trigger> - <Collapsible.Content> - <p class="flex-auto min-w-0 text-pretty"> - <span class="text-12-medium text-text-weak break-words">{props.children}</span> - </p> - </Collapsible.Content> - </Collapsible> - ) -} - -function ReadToolPart(props: { part: ToolPart }) { - const sync = useSync() - const local = useLocal() - return ( - <Switch> - <Match when={props.part.state.status === "pending"}> - <Part>Reading file...</Part> - </Match> - <Match when={props.part.state.status === "completed" && props.part.state}> - {(state) => { - const path = state().input["filePath"] as string - return ( - <Part onClick={() => local.file.open(path)}> - <span class="">Read</span> {getFilename(path)} - </Part> - ) - }} - </Match> - <Match when={props.part.state.status === "error" && props.part.state}> - {(state) => ( - <div> - <Part> - <span class="">Read</span> {getFilename(state().input["filePath"] as string)} - </Part> - <div class="text-icon-critical-active">{sync.sanitize(state().error)}</div> - </div> - )} - </Match> - </Switch> - ) -} - -function EditToolPart(props: { part: ToolPart }) { - const sync = useSync() - return ( - <Switch> - <Match when={props.part.state.status === "pending"}> - <Part>Preparing edit...</Part> - </Match> - <Match when={props.part.state.status === "completed" && props.part.state}> - {(state) => ( - <CollapsiblePart - title={ - <> - <span class="">Edit</span> {getFilename(state().input["filePath"] as string)} - </> - } - > - <Code path={state().input["filePath"] as string} code={state().metadata["diff"] as string} /> - </CollapsiblePart> - )} - </Match> - <Match when={props.part.state.status === "error" && props.part.state}> - {(state) => ( - <CollapsiblePart - title={ - <> - <span class="">Edit</span> {getFilename(state().input["filePath"] as string)} - </> - } - > - <div class="text-icon-critical-active">{sync.sanitize(state().error)}</div> - </CollapsiblePart> - )} - </Match> - </Switch> - ) -} - -function WriteToolPart(props: { part: ToolPart }) { - const sync = useSync() - return ( - <Switch> - <Match when={props.part.state.status === "pending"}> - <Part>Preparing write...</Part> - </Match> - <Match when={props.part.state.status === "completed" && props.part.state}> - {(state) => ( - <CollapsiblePart - title={ - <> - <span class="">Write</span> {getFilename(state().input["filePath"] as string)} - </> - } - > - <div class="p-2 bg-background-panel rounded-md border border-border-subtle"></div> - </CollapsiblePart> - )} - </Match> - <Match when={props.part.state.status === "error" && props.part.state}> - {(state) => ( - <div> - <Part> - <span class="">Write</span> {getFilename(state().input["filePath"] as string)} - </Part> - <div class="text-icon-critical-active">{sync.sanitize(state().error)}</div> - </div> - )} - </Match> - </Switch> - ) -} - -function BashToolPart(props: { part: ToolPart }) { - const sync = useSync() - return ( - <Switch> - <Match when={props.part.state.status === "pending"}> - <Part>Writing shell command...</Part> - </Match> - <Match when={props.part.state.status === "completed" && props.part.state}> - {(state) => ( - <CollapsiblePart - defaultOpen - title={ - <> - <span class="">Run command:</span> {state().input["command"]} - </> - } - > - <Markdown text={`\`\`\`command\n${state().input["command"]}\n${state().output}\`\`\``} /> - </CollapsiblePart> - )} - </Match> - <Match when={props.part.state.status === "error" && props.part.state}> - {(state) => ( - <CollapsiblePart - title={ - <> - <span class="">Shell</span> {state().input["command"]} - </> - } - > - <div class="text-icon-critical-active">{sync.sanitize(state().error)}</div> - </CollapsiblePart> - )} - </Match> - </Switch> - ) -} - -function ToolPart(props: { part: ToolPart }) { - // read - // edit - // write - // bash - // ls - // glob - // grep - // todowrite - // todoread - // webfetch - // websearch - // patch - // task - return ( - <div class="min-w-0 flex-auto text-12-medium"> - <Switch - fallback={ - <span> - {props.part.type}:{props.part.tool} - </span> - } - > - <Match when={props.part.tool === "read"}> - <ReadToolPart part={props.part} /> - </Match> - <Match when={props.part.tool === "edit"}> - <EditToolPart part={props.part} /> - </Match> - <Match when={props.part.tool === "write"}> - <WriteToolPart part={props.part} /> - </Match> - <Match when={props.part.tool === "bash"}> - <BashToolPart part={props.part} /> - </Match> - </Switch> - </div> - ) -} - -export default function SessionTimeline(props: { session: string; class?: string }) { - const sync = useSync() - const [scrollElement, setScrollElement] = createSignal<HTMLElement | undefined>(undefined) - const [root, setRoot] = createSignal<HTMLDivElement | undefined>(undefined) - const [tail, setTail] = createSignal(true) - const size = createElementSize(root) - const scroll = createScrollPosition(scrollElement) - - const valid = (part: Part) => { - if (!part) return false - switch (part.type) { - case "step-start": - case "step-finish": - case "file": - case "patch": - return false - case "text": - return !part.synthetic && part.text.trim() - case "reasoning": - return part.text.trim() - case "tool": - switch (part.tool) { - case "todoread": - case "todowrite": - case "list": - case "grep": - return false - } - return true - default: - return true - } - } - - const hasValidParts = (message: Message) => { - return sync.data.part[message.id]?.filter(valid).length > 0 - } - - const hasTextPart = (message: Message) => { - return !!sync.data.part[message.id]?.filter(valid).find((p) => p.type === "text") - } - - const session = createMemo(() => sync.session.get(props.session)) - const messages = createMemo(() => sync.data.message[props.session] ?? []) - const messagesWithValidParts = createMemo(() => sync.data.message[props.session]?.filter(hasValidParts) ?? []) - const working = createMemo(() => { - const last = messages()[messages().length - 1] - if (!last) return false - if (last.role === "user") return true - return !last.time.completed - }) - - const cost = createMemo(() => { - const total = pipe( - messages(), - sumBy((x) => (x.role === "assistant" ? x.cost : 0)), - ) - return new Intl.NumberFormat("en-US", { - style: "currency", - currency: "USD", - }).format(total) - }) - - const last = createMemo(() => { - return messages().findLast((x) => x.role === "assistant") as AssistantMessage - }) - - const model = createMemo(() => { - if (!last()) return - const model = sync.data.provider.find((x) => x.id === last().providerID)?.models[last().modelID] - return model - }) - - const tokens = createMemo(() => { - if (!last()) return - const tokens = last().tokens - const total = tokens.input + tokens.output + tokens.reasoning + tokens.cache.read + tokens.cache.write - return new Intl.NumberFormat("en-US", { - notation: "compact", - compactDisplay: "short", - }).format(total) - }) - - const context = createMemo(() => { - if (!last()) return - if (!model()?.limit.context) return 0 - const tokens = last().tokens - const total = tokens.input + tokens.output + tokens.reasoning + tokens.cache.read + tokens.cache.write - return Math.round((total / model()!.limit.context) * 100) - }) - - const getScrollParent = (el: HTMLElement | null): HTMLElement | undefined => { - let p = el?.parentElement - while (p && p !== document.body) { - const s = getComputedStyle(p) - if (s.overflowY === "auto" || s.overflowY === "scroll") return p - p = p.parentElement - } - return undefined - } - - createEffect(() => { - if (!root()) return - setScrollElement(getScrollParent(root()!)) - }) - - const scrollToBottom = () => { - const element = scrollElement() - if (!element) return - element.scrollTop = element.scrollHeight - } - - createEffect(() => { - size.height - if (tail()) scrollToBottom() - }) - - createEffect(() => { - if (working()) { - setTail(true) - scrollToBottom() - } - }) - - let lastScrollY = 0 - createEffect(() => { - if (scroll.y < lastScrollY) { - setTail(false) - } - lastScrollY = scroll.y - }) - - const duration = (part: Part) => { - switch (part.type) { - default: - if ( - "time" in part && - part.time && - "start" in part.time && - part.time.start && - "end" in part.time && - part.time.end - ) { - const start = DateTime.fromMillis(part.time.start) - const end = DateTime.fromMillis(part.time.end) - return end.diff(start).toFormat("s") - } - return "" - } - } - - createEffect(() => { - console.log("WHAT") - console.log(JSON.stringify(messagesWithValidParts())) - }) - - return ( - <div - ref={setRoot} - classList={{ - "select-text flex flex-col text-text-weak": true, - [props.class ?? ""]: !!props.class, - }} - > - <div class="flex justify-end items-center self-stretch"> - <div class="flex items-center gap-6"> - <Tooltip value={`${tokens()} Tokens`} class="flex items-center gap-1.5"> - <Show when={context()}> - <ProgressCircle percentage={context()!} /> - </Show> - <div class="text-14-regular text-text-weak text-right">{context()}%</div> - </Tooltip> - <div class="text-14-regular text-text-strong text-right">{cost()}</div> - </div> - </div> - <ul role="list" class="flex flex-col items-start self-stretch"> - <For each={messagesWithValidParts()}> - {(message) => ( - <div - classList={{ - "flex flex-col gap-1 justify-center items-start self-stretch": true, - "mt-6": hasTextPart(message), - }} - > - <For each={sync.data.part[message.id]?.filter(valid) ?? []}> - {(part) => ( - <li class="group/li"> - <Switch fallback={<div class="">{part.type}</div>}> - <Match when={part.type === "text" && part}> - {(part) => ( - <Switch> - <Match when={message.role === "user"}> - <div class="w-fit flex items-center px-3 py-1 rounded-md bg-surface-weak"> - <span class="text-14-regular text-text-strong whitespace-pre-wrap break-words"> - {part().text} - </span> - </div> - </Match> - <Match when={message.role === "assistant"}> - <Markdown text={sync.sanitize(part().text)} /> - </Match> - </Switch> - )} - </Match> - <Match when={part.type === "reasoning" && part}> - {(part) => ( - <CollapsiblePart - title={ - <Switch fallback={<span class="text-text-weak">Thinking</span>}> - <Match when={part().time.end}> - <span class="text-12-medium text-text-weak">Thought</span> for {duration(part())}s - </Match> - </Switch> - } - > - <Markdown text={part().text} /> - </CollapsiblePart> - )} - </Match> - <Match when={part.type === "tool" && part}>{(part) => <ToolPart part={part()} />}</Match> - </Switch> - </li> - )} - </For> - </div> - )} - </For> - </ul> - <Show when={false}> - <Collapsible defaultOpen={false}> - <Collapsible.Trigger> - <div class="mt-12 ml-1 flex items-center gap-x-2 text-xs text-text-muted"> - <Icon name="file-code" /> - <span>Raw Session Data</span> - <Collapsible.Arrow class="text-text-muted" /> - </div> - </Collapsible.Trigger> - <Collapsible.Content class="mt-5"> - <ul role="list" class="space-y-2"> - <li> - <Collapsible> - <Collapsible.Trigger> - <div class="flex items-center gap-x-2 text-xs text-text-muted ml-1"> - <Icon name="file-code" /> - <span>session</span> - <Collapsible.Arrow class="text-text-muted" /> - </div> - </Collapsible.Trigger> - <Collapsible.Content> - <Code path="session.json" code={JSON.stringify(session(), null, 2)} /> - </Collapsible.Content> - </Collapsible> - </li> - <For each={messages()}> - {(message) => ( - <> - <li> - <Collapsible> - <Collapsible.Trigger> - <div class="flex items-center gap-x-2 text-xs text-text-muted ml-1"> - <Icon name="file-code" /> - <span>{message.role === "user" ? "user" : "assistant"}</span> - <Collapsible.Arrow class="text-text-muted" /> - </div> - </Collapsible.Trigger> - <Collapsible.Content> - <Code path={message.id + ".json"} code={JSON.stringify(message, null, 2)} /> - </Collapsible.Content> - </Collapsible> - </li> - <For each={sync.data.part[message.id]}> - {(part) => ( - <li> - <Collapsible> - <Collapsible.Trigger> - <div class="flex items-center gap-x-2 text-xs text-text-muted ml-1"> - <Icon name="file-code" /> - <span>{part.type}</span> - <Collapsible.Arrow class="text-text-muted" /> - </div> - </Collapsible.Trigger> - <Collapsible.Content> - <Code path={message.id + "." + part.id + ".json"} code={JSON.stringify(part, null, 2)} /> - </Collapsible.Content> - </Collapsible> - </li> - )} - </For> - </> - )} - </For> - </ul> - </Collapsible.Content> - </Collapsible> - </Show> - </div> - ) -} |
