diff options
| author | Adam <[email protected]> | 2025-12-29 09:47:53 -0600 |
|---|---|---|
| committer | Adam <[email protected]> | 2025-12-29 09:47:57 -0600 |
| commit | 831e9bce51c035ec22ce9562bf0257d6b59b2fe4 (patch) | |
| tree | b6e86b44cbc980d737a63cb571cb61a9003d237b /packages/ui/src/components | |
| parent | 5de73abd820d9425e82f85f95f50568332ca7cdc (diff) | |
| download | opencode-831e9bce51c035ec22ce9562bf0257d6b59b2fe4.tar.gz opencode-831e9bce51c035ec22ce9562bf0257d6b59b2fe4.zip | |
fix(desktop): jankiness
Diffstat (limited to 'packages/ui/src/components')
| -rw-r--r-- | packages/ui/src/components/list.tsx | 55 | ||||
| -rw-r--r-- | packages/ui/src/components/message-part.tsx | 11 | ||||
| -rw-r--r-- | packages/ui/src/components/session-turn.tsx | 23 |
3 files changed, 43 insertions, 46 deletions
diff --git a/packages/ui/src/components/list.tsx b/packages/ui/src/components/list.tsx index 808c9b032..a549c19e9 100644 --- a/packages/ui/src/components/list.tsx +++ b/packages/ui/src/components/list.tsx @@ -1,5 +1,5 @@ import { type FilteredListProps, useFilteredList } from "@opencode-ai/ui/hooks" -import { createEffect, createSignal, For, type JSX, on, Show } from "solid-js" +import { createEffect, createSignal, For, onCleanup, type JSX, on, Show } from "solid-js" import { createStore } from "solid-js/store" import { Icon, type IconProps } from "./icon" import { IconButton } from "./icon-button" @@ -116,6 +116,33 @@ export function List<T>(props: ListProps<T> & { ref?: (ref: ListRef) => void }) setScrollRef, }) + function GroupHeader(props: { category: string }): JSX.Element { + const [stuck, setStuck] = createSignal(false) + const [header, setHeader] = createSignal<HTMLDivElement | undefined>(undefined) + + createEffect(() => { + const scroll = scrollRef() + const node = header() + if (!scroll || !node) return + + const handler = () => { + const rect = node.getBoundingClientRect() + const scrollRect = scroll.getBoundingClientRect() + setStuck(rect.top <= scrollRect.top + 1 && scroll.scrollTop > 0) + } + + scroll.addEventListener("scroll", handler, { passive: true }) + handler() + onCleanup(() => scroll.removeEventListener("scroll", handler)) + }) + + return ( + <div data-slot="list-header" data-stuck={stuck()} ref={setHeader}> + {props.category} + </div> + ) + } + return ( <div data-component="list" classList={{ [props.class ?? ""]: !!props.class }}> <Show when={!!props.search}> @@ -157,31 +184,7 @@ export function List<T>(props: ListProps<T> & { ref?: (ref: ListRef) => void }) {(group) => ( <div data-slot="list-group"> <Show when={group.category}> - {(() => { - const [stuck, setStuck] = createSignal(false) - return ( - <div - data-slot="list-header" - data-stuck={stuck()} - ref={(el) => { - createEffect(() => { - const scroll = scrollRef() - if (!scroll) return - const handler = () => { - const rect = el.getBoundingClientRect() - const scrollRect = scroll.getBoundingClientRect() - setStuck(rect.top <= scrollRect.top + 1 && scroll.scrollTop > 0) - } - scroll.addEventListener("scroll", handler, { passive: true }) - handler() - return () => scroll.removeEventListener("scroll", handler) - }) - }} - > - {group.category} - </div> - ) - })()} + <GroupHeader category={group.category} /> </Show> <div data-slot="list-items"> <For each={group.items}> diff --git a/packages/ui/src/components/message-part.tsx b/packages/ui/src/components/message-part.tsx index e43ffc322..ea865d7ce 100644 --- a/packages/ui/src/components/message-part.tsx +++ b/packages/ui/src/components/message-part.tsx @@ -364,16 +364,7 @@ PART_MAPPING["tool"] = function ToolPartDisplay(props) { const permission = createMemo(() => { const sessionID = props.message.sessionID const permissions = data.store.permission?.[sessionID] ?? [] - const next = permissions.reduce( - (result, perm) => { - if (!result) return perm - if (perm.id < result.id) return perm - return result - }, - undefined as (typeof permissions)[number] | undefined, - ) - if (!next) return undefined - return next.callID === part.callID ? next : undefined + return permissions.find((perm) => perm.callID === part.callID) }) const [forceOpen, setForceOpen] = createSignal(false) diff --git a/packages/ui/src/components/session-turn.tsx b/packages/ui/src/components/session-turn.tsx index a3ac5b0b4..0a1ee5c4f 100644 --- a/packages/ui/src/components/session-turn.tsx +++ b/packages/ui/src/components/session-turn.tsx @@ -111,13 +111,23 @@ export function SessionTurn( const allMessages = createMemo(() => data.store.message[props.sessionID] ?? []) - const message = createMemo(() => { + const messageIndex = createMemo(() => { const messages = allMessages() const result = Binary.search(messages, props.messageID, (m) => m.id) - if (!result.found) return undefined + if (!result.found) return -1 const msg = messages[result.index] - if (msg.role !== "user") return undefined + if (msg.role !== "user") return -1 + + return result.index + }) + + const message = createMemo(() => { + const index = messageIndex() + if (index < 0) return undefined + + const msg = allMessages()[index] + if (!msg || msg.role !== "user") return undefined return msg }) @@ -141,13 +151,6 @@ export function SessionTurn( return data.store.part[msg.id] ?? [] }) - const messageIndex = createMemo(() => { - const messages = allMessages() - const result = Binary.search(messages, props.messageID, (m) => m.id) - if (!result.found) return -1 - return result.index - }) - const assistantMessages = createMemo(() => { const msg = message() if (!msg) return [] as AssistantMessage[] |
