From fc115ea367dd034c7b989819d4f547c5d7519253 Mon Sep 17 00:00:00 2001
From: Adam <2363879+adamdotdevin@users.noreply.github.com>
Date: Mon, 27 Oct 2025 15:35:47 -0500
Subject: wip: desktop work
---
.../desktop/src/components/assistant-message.tsx | 362 +++++++++++++++++++++
packages/desktop/src/components/diff-changes.tsx | 20 ++
packages/desktop/src/context/local.tsx | 8 -
packages/desktop/src/pages/index.tsx | 180 +++++++---
4 files changed, 515 insertions(+), 55 deletions(-)
create mode 100644 packages/desktop/src/components/assistant-message.tsx
create mode 100644 packages/desktop/src/components/diff-changes.tsx
(limited to 'packages/desktop/src')
diff --git a/packages/desktop/src/components/assistant-message.tsx b/packages/desktop/src/components/assistant-message.tsx
new file mode 100644
index 000000000..2e3d659aa
--- /dev/null
+++ b/packages/desktop/src/components/assistant-message.tsx
@@ -0,0 +1,362 @@
+import type { Part, AssistantMessage, ReasoningPart, TextPart, ToolPart } from "@opencode-ai/sdk"
+import type { Tool } from "opencode/tool/tool"
+import type { ReadTool } from "opencode/tool/read"
+import { children, Component, createMemo, For, Match, Show, Switch, type JSX } from "solid-js"
+import { Dynamic } from "solid-js/web"
+import { Markdown } from "./markdown"
+import { Collapsible, Icon, IconProps } from "@opencode-ai/ui"
+import { getDirectory, getFilename } from "@/utils"
+import { ListTool } from "opencode/tool/ls"
+import { GlobTool } from "opencode/tool/glob"
+import { GrepTool } from "opencode/tool/grep"
+import { WebFetchTool } from "opencode/tool/webfetch"
+import { TaskTool } from "opencode/tool/task"
+import { BashTool } from "opencode/tool/bash"
+import { EditTool } from "opencode/tool/edit"
+import { DiffChanges } from "./diff-changes"
+import { WriteTool } from "opencode/tool/write"
+
+export function AssistantMessage(props: { message: AssistantMessage; parts: Part[] }) {
+ return (
+
+
+ {(part) => {
+ const component = createMemo(() => PART_MAPPING[part.type as keyof typeof PART_MAPPING])
+ return (
+
+
+
+ )
+ }}
+
+
+ )
+}
+
+const PART_MAPPING = {
+ text: TextPart,
+ tool: ToolPart,
+ reasoning: ReasoningPart,
+}
+
+function ReasoningPart(props: { part: ReasoningPart; message: AssistantMessage }) {
+ return null
+ // return (
+ //
+ // {props.part.text}
+ //
+ // )
+}
+
+function TextPart(props: { part: TextPart; message: AssistantMessage }) {
+ return (
+
+
+
+ )
+}
+
+function ToolPart(props: { part: ToolPart; message: AssistantMessage }) {
+ // const sync = useSync()
+
+ 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 permissions = sync.data.permission[props.message.sessionID] ?? []
+ // const permissionIndex = permissions.findIndex((x) => x.callID === props.part.callID)
+ // const permission = permissions[permissionIndex]
+
+ return (
+ <>
+
+ {/* {props.part.state.error.replace("Error: ", "")} */}
+ >
+ )
+ })
+
+ return {component()}
+}
+
+type TriggerTitle = {
+ title: string
+ subtitle?: string
+ args?: 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 }) {
+ const resolved = children(() => props.children)
+
+ return (
+
+
+
+
+
+
+
+
+
+
+ {(props.trigger as TriggerTitle).title}
+
+
+ {(props.trigger as TriggerTitle).subtitle}
+
+
+
+ {(arg) => {arg}}
+
+
+
+
{(props.trigger as TriggerTitle).action}
+
+
+ {props.trigger as JSX.Element}
+
+
+
+
+
+
+
+
+ {props.children}
+
+
+ )
+}
+
+function GenericTool(props: ToolProps) {
+ return
+}
+
+type ToolProps = {
+ input: Partial>
+ metadata: Partial>
+ // permission: Record
+ tool: string
+ output?: string
+}
+
+const ToolRegistry = (() => {
+ const state: Record<
+ string,
+ {
+ name: string
+ render?: Component>
+ }
+ > = {}
+ function register(input: { name: string; render?: Component> }) {
+ state[input.name] = input
+ return input
+ }
+ return {
+ register,
+ render(name: string) {
+ return state[name]?.render
+ },
+ }
+})()
+
+ToolRegistry.register({
+ name: "read",
+ render(props) {
+ return (
+
+ )
+ },
+})
+
+ToolRegistry.register({
+ name: "list",
+ render(props) {
+ return (
+
+
+ {props.output}
+
+
+ )
+ },
+})
+
+ToolRegistry.register({
+ name: "glob",
+ render(props) {
+ return (
+
+
+ {props.output}
+
+
+ )
+ },
+})
+
+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 (
+
+
+ {props.output}
+
+
+ )
+ },
+})
+
+ToolRegistry.register({
+ name: "webfetch",
+ render(props) {
+ return (
+
+
+
+ ),
+ }}
+ >
+
+ {props.output}
+
+
+ )
+ },
+})
+
+ToolRegistry.register({
+ name: "task",
+ render(props) {
+ return (
+
+
+ {props.output}
+
+
+ )
+ },
+})
+
+ToolRegistry.register({
+ name: "bash",
+ render(props) {
+ return (
+
+
+ {props.output}
+
+
+ )
+ },
+})
+
+ToolRegistry.register({
+ name: "edit",
+ render(props) {
+ return (
+
+
+
Edit
+
+
+ {getDirectory(props.input.filePath!)}/
+
+ {getFilename(props.input.filePath ?? "")}
+
+
+ {/* */}
+
+ }
+ >
+
+ {props.output}
+
+
+ )
+ },
+})
+
+ToolRegistry.register({
+ name: "write",
+ render(props) {
+ return (
+
+
+
Write
+
+
+ {getDirectory(props.input.filePath!)}/
+
+ {getFilename(props.input.filePath ?? "")}
+
+
+ {/* */}
+
+ }
+ >
+
+ {props.output}
+
+
+ )
+ },
+})
diff --git a/packages/desktop/src/components/diff-changes.tsx b/packages/desktop/src/components/diff-changes.tsx
new file mode 100644
index 000000000..3b633f70f
--- /dev/null
+++ b/packages/desktop/src/components/diff-changes.tsx
@@ -0,0 +1,20 @@
+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 (
+ 0}>
+
+ {`+${additions()}`}
+ {`-${deletions()}`}
+
+
+ )
+}
diff --git a/packages/desktop/src/context/local.tsx b/packages/desktop/src/context/local.tsx
index 6ed8ec17b..978dbfbc6 100644
--- a/packages/desktop/src/context/local.tsx
+++ b/packages/desktop/src/context/local.tsx
@@ -460,13 +460,6 @@ export const { use: useLocal, provider: LocalProvider } = createSimpleContext({
return sync.data.message[store.active]?.find((m) => m.id === store.activeMessage)
})
- const activeAssistantMessages = createMemo(() => {
- if (!store.active || !activeMessage()) return []
- return sync.data.message[store.active]?.filter(
- (m) => m.role === "assistant" && m.parentID == activeMessage()?.id,
- )
- })
-
const model = createMemo(() => {
if (!last()) return
const model = sync.data.provider.find((x) => x.id === last().providerID)?.models[last().modelID]
@@ -504,7 +497,6 @@ export const { use: useLocal, provider: LocalProvider } = createSimpleContext({
return {
active,
activeMessage,
- activeAssistantMessages,
lastUserMessage,
cost,
last,
diff --git a/packages/desktop/src/pages/index.tsx b/packages/desktop/src/pages/index.tsx
index 6702284b2..15da87bd6 100644
--- a/packages/desktop/src/pages/index.tsx
+++ b/packages/desktop/src/pages/index.tsx
@@ -22,6 +22,10 @@ import { Code } from "@/components/code"
import { useSync } from "@/context/sync"
import { useSDK } from "@/context/sdk"
import { Diff } from "@/components/diff"
+import { ProgressCircle } from "@/components/progress-circle"
+import { AssistantMessage } from "@/components/assistant-message"
+import { type AssistantMessage as AssistantMessageType } from "@opencode-ai/sdk"
+import { DiffChanges } from "@/components/diff-changes"
export default function Page() {
const local = useLocal()
@@ -92,7 +96,7 @@ export default function Page() {
}
}
- if (event.key.length === 1 && event.key !== "Unidentified") {
+ if (event.key.length === 1 && event.key !== "Unidentified" && !(event.ctrlKey || event.metaKey)) {
inputRef?.focus()
}
}
@@ -392,9 +396,6 @@ export default function Page() {
{(session) => {
const diffs = createMemo(() => session.summary?.diffs ?? [])
const filesChanged = createMemo(() => diffs().length)
- const additions = createMemo(() => diffs().reduce((acc, diff) => (acc ?? 0) + (diff.additions ?? 0), 0))
- const deletions = createMemo(() => diffs().reduce((acc, diff) => (acc ?? 0) + (diff.deletions ?? 0), 0))
-
return (
@@ -408,12 +409,7 @@ export default function Page() {
{`${filesChanged() || "No"} file${filesChanged() !== 1 ? "s" : ""} changed`}
-
-
- {`+${additions()}`}
- {`-${deletions()}`}
-
-
+
@@ -434,13 +430,12 @@ export default function Page() {
-
+
Chat
-
-
- {local.session.context()}%
-
-
+
+
+ {local.session.context() ?? 0}%
+
{/* Review */}
file.path)}>
@@ -548,33 +543,114 @@ export default function Page() {
1}>
- {(message) => (
- - local.session.setActiveMessage(message.id)}
- >
-
-
-
- {
+ const countLines = (text: string) => {
+ if (!text) return 0
+ return text.split("\n").length
+ }
+
+ const additions = createMemo(
+ () =>
+ message.summary?.diffs.reduce((acc, diff) => acc + (diff.additions ?? 0), 0) ?? 0,
+ )
+
+ const deletions = createMemo(
+ () =>
+ message.summary?.diffs.reduce((acc, diff) => acc + (diff.deletions ?? 0), 0) ?? 0,
+ )
+
+ const totalBeforeLines = createMemo(
+ () =>
+ message.summary?.diffs.reduce((acc, diff) => acc + countLines(diff.before), 0) ??
+ 0,
+ )
+
+ const blockCounts = createMemo(() => {
+ const TOTAL_BLOCKS = 5
+
+ const adds = additions()
+ const dels = deletions()
+ const unchanged = Math.max(0, totalBeforeLines() - dels)
+
+ const totalActivity = unchanged + adds + dels
+
+ if (totalActivity === 0) {
+ return { added: 0, deleted: 0, neutral: TOTAL_BLOCKS }
+ }
+
+ const percentAdded = adds / totalActivity
+ const percentDeleted = dels / totalActivity
+ const added_raw = percentAdded * TOTAL_BLOCKS
+ const deleted_raw = percentDeleted * TOTAL_BLOCKS
+
+ let added = adds > 0 ? Math.ceil(added_raw) : 0
+ let deleted = dels > 0 ? Math.ceil(deleted_raw) : 0
+
+ let total_allocated = added + deleted
+ if (total_allocated > TOTAL_BLOCKS) {
+ if (added_raw < deleted_raw) {
+ added = Math.floor(added_raw)
+ } else {
+ deleted = Math.floor(deleted_raw)
+ }
+
+ total_allocated = added + deleted
+ if (total_allocated > TOTAL_BLOCKS) {
+ if (added_raw < deleted_raw) {
+ deleted = Math.floor(deleted_raw)
+ } else {
+ added = Math.floor(added_raw)
+ }
+ }
+ }
+
+ const neutral = Math.max(0, TOTAL_BLOCKS - added - deleted)
+
+ return { added, deleted, neutral }
+ })
+
+ const ADD_COLOR = "var(--icon-diff-add-base)"
+ const DELETE_COLOR = "var(--icon-diff-delete-base)"
+ const NEUTRAL_COLOR = "var(--icon-weak-base)"
+
+ const visibleBlocks = createMemo(() => {
+ const counts = blockCounts()
+ const blocks = [
+ ...Array(counts.added).fill(ADD_COLOR),
+ ...Array(counts.deleted).fill(DELETE_COLOR),
+ ...Array(counts.neutral).fill(NEUTRAL_COLOR),
+ ]
+ return blocks.slice(0, 5)
+ })
+
+ return (
+
- local.session.setActiveMessage(message.id)}
>
- {message.summary?.title ?? local.session.getMessageText(message)}
-
-
- )}
+
+
+
+
+ {message.summary?.title ?? local.session.getMessageText(message)}
+
+
+ )
+ }}
@@ -585,6 +661,11 @@ export default function Page() {
const title = createMemo(() => message.summary?.title)
const prompt = createMemo(() => local.session.getMessageText(message))
const summary = createMemo(() => message.summary?.body)
+ const assistantMessages = createMemo(() => {
+ return sync.data.message[activeSession().id]?.filter(
+ (m) => m.role === "assistant" && m.parentID == message.id,
+ ) as AssistantMessageType[]
+ })
return (
-
- {`+${diff.additions}`}
- {`-${diff.deletions}`}
-
+
@@ -661,10 +739,18 @@ export default function Page() {
{/* Response */}
-
+
Response
+
+
+ {(assistantMessage) => {
+ const parts = createMemo(() => sync.data.part[assistantMessage.id])
+ return
+ }}
+
+
)
--
cgit v1.2.3