| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448 |
- import { Component, createMemo, For, Match, Show, Switch } from "solid-js"
- import { Dynamic } from "solid-js/web"
- import {
- AssistantMessage,
- Message as MessageType,
- Part as PartType,
- TextPart,
- ToolPart,
- UserMessage,
- } from "@opencode-ai/sdk"
- import { BasicTool } from "./basic-tool"
- import { GenericTool } from "./basic-tool"
- import { Card } from "./card"
- import { Icon } from "./icon"
- import { Checkbox } from "./checkbox"
- import { Diff } from "./diff"
- import { DiffChanges } from "./diff-changes"
- import { Markdown } from "./markdown"
- export interface MessageProps {
- message: MessageType
- parts: PartType[]
- }
- export interface MessagePartProps {
- part: PartType
- message: MessageType
- hideDetails?: boolean
- }
- export type PartComponent = Component<MessagePartProps>
- export const PART_MAPPING: Record<string, PartComponent | undefined> = {}
- function getFilename(path: string) {
- if (!path) return ""
- const trimmed = path.replace(/[\/]+$/, "")
- const parts = trimmed.split("/")
- return parts[parts.length - 1] ?? ""
- }
- function getDirectory(path: string) {
- const parts = path.split("/")
- const dir = parts.slice(0, parts.length - 1).join("/")
- return dir ? dir + "/" : ""
- }
- export function registerPartComponent(type: string, component: PartComponent) {
- PART_MAPPING[type] = component
- }
- export function Message(props: MessageProps) {
- return (
- <Switch>
- <Match when={props.message.role === "user" && props.message}>
- {(userMessage) => (
- <UserMessageDisplay message={userMessage() as UserMessage} parts={props.parts} />
- )}
- </Match>
- <Match when={props.message.role === "assistant" && props.message}>
- {(assistantMessage) => (
- <AssistantMessageDisplay
- message={assistantMessage() as AssistantMessage}
- parts={props.parts}
- />
- )}
- </Match>
- </Switch>
- )
- }
- export function AssistantMessageDisplay(props: { message: AssistantMessage; parts: PartType[] }) {
- const filteredParts = createMemo(() => {
- return props.parts?.filter((x) => {
- if (x.type === "reasoning") return false
- return x.type !== "tool" || (x as ToolPart).tool !== "todoread"
- })
- })
- return <For each={filteredParts()}>{(part) => <Part part={part} message={props.message} />}</For>
- }
- export function UserMessageDisplay(props: { message: UserMessage; parts: PartType[] }) {
- const text = createMemo(() =>
- props.parts
- ?.filter((p) => p.type === "text" && !(p as TextPart).synthetic)
- ?.map((p) => (p as TextPart).text)
- ?.join(""),
- )
- return <div data-component="user-message">{text()}</div>
- }
- export function Part(props: MessagePartProps) {
- const component = createMemo(() => PART_MAPPING[props.part.type])
- return (
- <Show when={component()}>
- <Dynamic
- component={component()}
- part={props.part}
- message={props.message}
- hideDetails={props.hideDetails}
- />
- </Show>
- )
- }
- export interface ToolProps {
- input: Record<string, any>
- metadata: Record<string, any>
- tool: string
- output?: string
- hideDetails?: boolean
- }
- export type ToolComponent = Component<ToolProps>
- const state: Record<
- string,
- {
- name: string
- render?: ToolComponent
- }
- > = {}
- export function registerTool(input: { name: string; render?: ToolComponent }) {
- state[input.name] = input
- return input
- }
- export function getTool(name: string) {
- return state[name]?.render
- }
- export const ToolRegistry = {
- register: registerTool,
- render: getTool,
- }
- PART_MAPPING["tool"] = function ToolPartDisplay(props) {
- const part = props.part as ToolPart
- const component = createMemo(() => {
- 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 (
- <Switch>
- <Match when={part.state.status === "error" && part.state.error}>
- {(error) => {
- const cleaned = error().replace("Error: ", "")
- const [title, ...rest] = cleaned.split(": ")
- return (
- <Card variant="error">
- <div data-component="tool-error">
- <Icon name="circle-ban-sign" size="small" data-slot="tool-error-icon" />
- <Switch>
- <Match when={title && title.length < 30}>
- <div data-slot="tool-error-content">
- <div data-slot="tool-error-title">{title}</div>
- <span data-slot="tool-error-message">{rest.join(": ")}</span>
- </div>
- </Match>
- <Match when={true}>
- <span data-slot="tool-error-message">{cleaned}</span>
- </Match>
- </Switch>
- </div>
- </Card>
- )
- }}
- </Match>
- <Match when={true}>
- <Dynamic
- component={render}
- input={input}
- tool={part.tool}
- metadata={metadata}
- output={part.state.status === "completed" ? part.state.output : undefined}
- hideDetails={props.hideDetails}
- />
- </Match>
- </Switch>
- )
- })
- return <Show when={component()}>{component()}</Show>
- }
- PART_MAPPING["text"] = function TextPartDisplay(props) {
- const part = props.part as TextPart
- return (
- <Show when={part.text.trim()}>
- <div data-component="text-part">
- <Markdown text={part.text.trim()} />
- </div>
- </Show>
- )
- }
- PART_MAPPING["reasoning"] = function ReasoningPartDisplay(props) {
- const part = props.part as any
- return (
- <Show when={part.text.trim()}>
- <div data-component="reasoning-part">
- <Markdown text={part.text.trim()} />
- </div>
- </Show>
- )
- }
- ToolRegistry.register({
- name: "read",
- render(props) {
- return (
- <BasicTool
- icon="glasses"
- trigger={{
- title: "Read",
- subtitle: props.input.filePath ? getFilename(props.input.filePath) : "",
- }}
- />
- )
- },
- })
- ToolRegistry.register({
- name: "list",
- render(props) {
- return (
- <BasicTool
- icon="bullet-list"
- trigger={{ title: "List", subtitle: getDirectory(props.input.path || "/") }}
- >
- <Show when={false && props.output}>
- <div data-component="tool-output">{props.output}</div>
- </Show>
- </BasicTool>
- )
- },
- })
- ToolRegistry.register({
- name: "glob",
- render(props) {
- return (
- <BasicTool
- icon="magnifying-glass-menu"
- trigger={{
- title: "Glob",
- subtitle: getDirectory(props.input.path || "/"),
- args: props.input.pattern ? ["pattern=" + props.input.pattern] : [],
- }}
- >
- <Show when={false && props.output}>
- <div data-component="tool-output">{props.output}</div>
- </Show>
- </BasicTool>
- )
- },
- })
- ToolRegistry.register({
- name: "grep",
- render(props) {
- const args = []
- if (props.input.pattern) args.push("pattern=" + props.input.pattern)
- if (props.input.include) args.push("include=" + props.input.include)
- return (
- <BasicTool
- icon="magnifying-glass-menu"
- trigger={{
- title: "Grep",
- subtitle: getDirectory(props.input.path || "/"),
- args,
- }}
- >
- <Show when={false && props.output}>
- <div data-component="tool-output">{props.output}</div>
- </Show>
- </BasicTool>
- )
- },
- })
- ToolRegistry.register({
- name: "webfetch",
- render(props) {
- return (
- <BasicTool
- icon="window-cursor"
- trigger={{
- title: "Webfetch",
- subtitle: props.input.url || "",
- args: props.input.format ? ["format=" + props.input.format] : [],
- action: (
- <div data-component="tool-action">
- <Icon name="square-arrow-top-right" size="small" />
- </div>
- ),
- }}
- >
- <Show when={false && props.output}>
- <div data-component="tool-output">{props.output}</div>
- </Show>
- </BasicTool>
- )
- },
- })
- ToolRegistry.register({
- name: "task",
- render(props) {
- return (
- <BasicTool
- icon="task"
- trigger={{
- title: `${props.input.subagent_type || props.tool} Agent`,
- titleClass: "capitalize",
- subtitle: props.input.description,
- }}
- >
- <Show when={false && props.output}>
- <div data-component="tool-output">{props.output}</div>
- </Show>
- </BasicTool>
- )
- },
- })
- ToolRegistry.register({
- name: "bash",
- render(props) {
- return (
- <BasicTool
- icon="console"
- trigger={{
- title: "Shell",
- subtitle: "Ran " + props.input.command,
- }}
- >
- <Show when={false && props.output}>
- <div data-component="tool-output">{props.output}</div>
- </Show>
- </BasicTool>
- )
- },
- })
- ToolRegistry.register({
- name: "edit",
- render(props) {
- return (
- <BasicTool
- icon="code-lines"
- trigger={
- <div data-component="edit-trigger">
- <div data-slot="title-area">
- <div data-slot="title">Edit</div>
- <div data-slot="path">
- <Show when={props.input.filePath?.includes("/")}>
- <span data-slot="directory">{getDirectory(props.input.filePath!)}</span>
- </Show>
- <span data-slot="filename">{getFilename(props.input.filePath ?? "")}</span>
- </div>
- </div>
- <div data-slot="actions">
- <Show when={props.metadata.filediff}>
- <DiffChanges changes={props.metadata.filediff} />
- </Show>
- </div>
- </div>
- }
- >
- <Show when={props.metadata.filediff}>
- <div data-component="edit-content">
- <Diff
- before={{
- name: getFilename(props.metadata.filediff.path),
- contents: props.metadata.filediff.before,
- }}
- after={{
- name: getFilename(props.metadata.filediff.path),
- contents: props.metadata.filediff.after,
- }}
- />
- </div>
- </Show>
- </BasicTool>
- )
- },
- })
- ToolRegistry.register({
- name: "write",
- render(props) {
- return (
- <BasicTool
- icon="code-lines"
- trigger={
- <div data-component="write-trigger">
- <div data-slot="title-area">
- <div data-slot="title">Write</div>
- <div data-slot="path">
- <Show when={props.input.filePath?.includes("/")}>
- <span data-slot="directory">{getDirectory(props.input.filePath!)}</span>
- </Show>
- <span data-slot="filename">{getFilename(props.input.filePath ?? "")}</span>
- </div>
- </div>
- <div data-slot="actions">{/* <DiffChanges diff={diff} /> */}</div>
- </div>
- }
- >
- <Show when={false && props.output}>
- <div data-component="tool-output">{props.output}</div>
- </Show>
- </BasicTool>
- )
- },
- })
- ToolRegistry.register({
- name: "todowrite",
- render(props) {
- return (
- <BasicTool
- icon="checklist"
- trigger={{
- title: "To-dos",
- subtitle: `${props.input.todos?.filter((t: any) => t.status === "completed").length}/${props.input.todos?.length}`,
- }}
- >
- <Show when={props.input.todos?.length}>
- <div data-component="todos">
- <For each={props.input.todos}>
- {(todo: any) => (
- <Checkbox readOnly checked={todo.status === "completed"}>
- <div data-slot="todo-content" data-completed={todo.status === "completed"}>
- {todo.content}
- </div>
- </Checkbox>
- )}
- </For>
- </div>
- </Show>
- </BasicTool>
- )
- },
- })
|