diff --git a/packages/desktop/src/app.tsx b/packages/desktop/src/app.tsx index bc6f082a4..789a8fa2d 100644 --- a/packages/desktop/src/app.tsx +++ b/packages/desktop/src/app.tsx @@ -12,7 +12,7 @@ import { GlobalSDKProvider } from "@/context/global-sdk" import { TerminalProvider } from "@/context/terminal" import { PromptProvider } from "@/context/prompt" import { NotificationProvider } from "@/context/notification" -import { DialogProvider, DialogRoot } from "@opencode-ai/ui/context/dialog" +import { DialogProvider } from "@opencode-ai/ui/context/dialog" import { CommandProvider } from "@/context/command" import Layout from "@/pages/layout" import Home from "@/pages/home" @@ -36,46 +36,46 @@ const url = export function App() { return ( - - - - - - - - - ( - + + + + + + + + + + ( {props.children} - - )} - > - - - } /> - ( - - - - - - - - )} - /> - - - - - - - - - + )} + > + + + } /> + ( + + + + + + + + )} + /> + + + + + + + + + + ) } diff --git a/packages/desktop/src/hooks/use-providers.ts b/packages/desktop/src/hooks/use-providers.ts index 501ff9d0c..4a73fa055 100644 --- a/packages/desktop/src/hooks/use-providers.ts +++ b/packages/desktop/src/hooks/use-providers.ts @@ -6,8 +6,8 @@ import { createMemo } from "solid-js" export const popularProviders = ["opencode", "anthropic", "github-copilot", "openai", "google", "openrouter", "vercel"] export function useProviders() { - const params = useParams() const globalSync = useGlobalSync() + const params = useParams() const currentDirectory = createMemo(() => base64Decode(params.dir ?? "")) const providers = createMemo(() => { if (currentDirectory()) { diff --git a/packages/desktop/src/pages/directory-layout.tsx b/packages/desktop/src/pages/directory-layout.tsx index 0dbb3f6d6..c909a373d 100644 --- a/packages/desktop/src/pages/directory-layout.tsx +++ b/packages/desktop/src/pages/directory-layout.tsx @@ -6,7 +6,6 @@ import { LocalProvider } from "@/context/local" import { base64Decode } from "@opencode-ai/util/encode" import { DataProvider } from "@opencode-ai/ui/context" import { iife } from "@opencode-ai/util/iife" -import { DialogRoot } from "@opencode-ai/ui/context/dialog" export default function Layout(props: ParentProps) { const params = useParams() @@ -21,9 +20,7 @@ export default function Layout(props: ParentProps) { const sync = useSync() return ( - - {props.children} - + {props.children} ) })} diff --git a/packages/ui/src/context/dialog.tsx b/packages/ui/src/context/dialog.tsx index af5da06f9..fae0c57b4 100644 --- a/packages/ui/src/context/dialog.tsx +++ b/packages/ui/src/context/dialog.tsx @@ -1,79 +1,116 @@ -import { For, Show, type JSX } from "solid-js" -import { createStore } from "solid-js/store" -import { createSimpleContext } from "@opencode-ai/ui/context" - -type DialogElement = JSX.Element | (() => JSX.Element) - -export const { use: useDialog, provider: DialogProvider } = createSimpleContext({ - name: "Dialog", - init: () => { - const [store, setStore] = createStore({ - stack: [] as { - element: DialogElement - onClose?: () => void - }[], - }) - - return { - get stack() { - return store.stack - }, - push(element: DialogElement, onClose?: () => void) { - setStore("stack", (s) => [...s, { element, onClose }]) - }, - pop() { - const current = store.stack.at(-1) - current?.onClose?.() - setStore("stack", store.stack.slice(0, -1)) - }, - replace(element: DialogElement, onClose?: () => void) { - for (const item of store.stack) { - item.onClose?.() - } - setStore("stack", [{ element, onClose }]) - }, - clear() { - for (const item of store.stack) { - item.onClose?.() - } - setStore("stack", []) - }, - } - }, -}) - +import { + createContext, + createMemo, + createSignal, + getOwner, + Owner, + ParentProps, + runWithOwner, + Show, + useContext, + type JSX, +} from "solid-js" import { Dialog as Kobalte } from "@kobalte/core/dialog" -export function DialogRoot(props: { children?: JSX.Element }) { - const dialog = useDialog() +type DialogElement = () => JSX.Element + +const Context = createContext>() + +function init() { + const [store, setStore] = createSignal< + { + element: DialogElement + onClose?: () => void + owner: Owner + }[] + >([]) + + return { + get stack() { + return store() + }, + push(element: DialogElement, owner: Owner, onClose?: () => void) { + setStore((s) => [...s, { element, onClose, owner }]) + }, + pop() { + const current = store().at(-1) + current?.onClose?.() + setStore((stack) => stack.slice(0, -1)) + }, + replace(element: DialogElement, owner: Owner, onClose?: () => void) { + for (const item of store()) { + item.onClose?.() + } + setStore([{ element, onClose, owner }]) + }, + clear() { + for (const item of store()) { + item.onClose?.() + } + setStore([]) + }, + } +} + +export function DialogProvider(props: ParentProps) { + const ctx = init() + const last = createMemo(() => ctx.stack.at(-1)) return ( - <> + {props.children} - 0}> -
- - {(item, index) => ( - +
+ + {(item) => + runWithOwner(item().owner, () => { + return ( { if (!open) { - item.onClose?.() - dialog.pop() + item().onClose?.() + ctx.pop() } }} > - {typeof item.element === "function" ? item.element() : item.element} + {item().element()} - - )} - -
-
- + ) + }) + } + +
+
) } + +export function useDialog() { + const ctx = useContext(Context) + const owner = getOwner() + if (!owner) { + throw new Error("useDialog must be used within a DialogProvider") + } + if (!ctx) { + throw new Error("useDialog must be used within a DialogProvider") + } + return { + get stack() { + return ctx.stack + }, + replace(element: DialogElement, onClose?: () => void) { + ctx.replace(element, owner, onClose) + }, + push(element: DialogElement, onClose?: () => void) { + ctx.push(element, owner, onClose) + }, + pop() { + ctx.pop() + }, + clear() { + ctx.clear() + }, + } +}