import { useDialog } from "@opencode-ai/ui/context/dialog" import { Dialog } from "@opencode-ai/ui/dialog" import { FileIcon } from "@opencode-ai/ui/file-icon" import { Icon } from "@opencode-ai/ui/icon" import { Keybind } from "@opencode-ai/ui/keybind" import { List } from "@opencode-ai/ui/list" import { base64Encode } from "@opencode-ai/util/encode" import { getDirectory, getFilename } from "@opencode-ai/util/path" import { useNavigate, useParams } from "@solidjs/router" import { createMemo, createSignal, Match, onCleanup, Show, Switch } from "solid-js" import { formatKeybind, useCommand, type CommandOption } from "@/context/command" import { useGlobalSDK } from "@/context/global-sdk" import { useGlobalSync } from "@/context/global-sync" import { useLayout } from "@/context/layout" import { useFile } from "@/context/file" import { useLanguage } from "@/context/language" import { decode64 } from "@/utils/base64" import { getRelativeTime } from "@/utils/time" type EntryType = "command" | "file" | "session" type Entry = { id: string type: EntryType title: string description?: string keybind?: string category: string option?: CommandOption path?: string directory?: string sessionID?: string archived?: number updated?: number } type DialogSelectFileMode = "all" | "files" export function DialogSelectFile(props: { mode?: DialogSelectFileMode; onOpenFile?: (path: string) => void }) { const command = useCommand() const language = useLanguage() const layout = useLayout() const file = useFile() const dialog = useDialog() const params = useParams() const navigate = useNavigate() const globalSDK = useGlobalSDK() const globalSync = useGlobalSync() const filesOnly = () => props.mode === "files" const sessionKey = createMemo(() => `${params.dir}${params.id ? "/" + params.id : ""}`) const tabs = createMemo(() => layout.tabs(sessionKey)) const view = createMemo(() => layout.view(sessionKey)) const state = { cleanup: undefined as (() => void) | void, committed: false } const [grouped, setGrouped] = createSignal(false) const common = [ "session.new", "workspace.new", "session.previous", "session.next", "terminal.toggle", "review.toggle", ] const limit = 5 const allowed = createMemo(() => { if (filesOnly()) return [] return command.options.filter( (option) => !option.disabled && !option.id.startsWith("suggested.") && option.id !== "file.open", ) }) const commandItem = (option: CommandOption): Entry => ({ id: "command:" + option.id, type: "command", title: option.title, description: option.description, keybind: option.keybind, category: language.t("palette.group.commands"), option, }) const fileItem = (path: string): Entry => ({ id: "file:" + path, type: "file", title: path, category: language.t("palette.group.files"), path, }) const projectDirectory = createMemo(() => decode64(params.dir) ?? "") const project = createMemo(() => { const directory = projectDirectory() if (!directory) return return layout.projects.list().find((p) => p.worktree === directory || p.sandboxes?.includes(directory)) }) const workspaces = createMemo(() => { const directory = projectDirectory() const current = project() if (!current) return directory ? [directory] : [] const dirs = [current.worktree, ...(current.sandboxes ?? [])] if (directory && !dirs.includes(directory)) return [...dirs, directory] return dirs }) const homedir = createMemo(() => globalSync.data.path.home) const label = (directory: string) => { const current = project() const kind = current && directory === current.worktree ? language.t("workspace.type.local") : language.t("workspace.type.sandbox") const [store] = globalSync.child(directory, { bootstrap: false }) const home = homedir() const path = home ? directory.replace(home, "~") : directory const name = store.vcs?.branch ?? getFilename(directory) return `${kind} : ${name || path}` } const sessionItem = (input: { directory: string id: string title: string description: string archived?: number updated?: number }): Entry => ({ id: `session:${input.directory}:${input.id}`, type: "session", title: input.title, description: input.description, category: language.t("command.category.session"), directory: input.directory, sessionID: input.id, archived: input.archived, updated: input.updated, }) const list = createMemo(() => allowed().map(commandItem)) const picks = createMemo(() => { const all = allowed() const order = new Map(common.map((id, index) => [id, index])) const picked = all.filter((option) => order.has(option.id)) const base = picked.length ? picked : all.slice(0, limit) const sorted = picked.length ? [...base].sort((a, b) => (order.get(a.id) ?? 0) - (order.get(b.id) ?? 0)) : base return sorted.map(commandItem) }) const recent = createMemo(() => { const all = tabs().all() const active = tabs().active() const order = active ? [active, ...all.filter((item) => item !== active)] : all const seen = new Set() const items: Entry[] = [] for (const item of order) { const path = file.pathFromTab(item) if (!path) continue if (seen.has(path)) continue seen.add(path) items.push(fileItem(path)) } return items.slice(0, limit) }) const root = createMemo(() => { const nodes = file.tree.children("") const paths = nodes .filter((node) => node.type === "file") .map((node) => node.path) .sort((a, b) => a.localeCompare(b)) return paths.slice(0, limit).map(fileItem) }) const unique = (items: Entry[]) => { const seen = new Set() const out: Entry[] = [] for (const item of items) { if (seen.has(item.id)) continue seen.add(item.id) out.push(item) } return out } const sessionToken = { value: 0 } let sessionInflight: Promise | undefined let sessionAll: Entry[] | undefined const sessions = (text: string) => { const query = text.trim() if (!query) { sessionToken.value += 1 sessionInflight = undefined sessionAll = undefined return [] as Entry[] } if (sessionAll) return sessionAll if (sessionInflight) return sessionInflight const current = sessionToken.value const dirs = workspaces() if (dirs.length === 0) return [] as Entry[] sessionInflight = Promise.all( dirs.map((directory) => { const description = label(directory) return globalSDK.client.session .list({ directory, roots: true }) .then((x) => (x.data ?? []) .filter((s) => !!s?.id) .map((s) => ({ id: s.id, title: s.title ?? language.t("command.session.new"), description, directory, archived: s.time?.archived, updated: s.time?.updated, })), ) .catch(() => [] as { id: string; title: string; description: string; directory: string; archived?: number }[]) }), ) .then((results) => { if (sessionToken.value !== current) return [] as Entry[] const seen = new Set() const next = results .flat() .filter((item) => { const key = `${item.directory}:${item.id}` if (seen.has(key)) return false seen.add(key) return true }) .map(sessionItem) sessionAll = next return next }) .catch(() => [] as Entry[]) .finally(() => { sessionInflight = undefined }) return sessionInflight } const items = async (text: string) => { const query = text.trim() setGrouped(query.length > 0) if (!query && filesOnly()) { const loaded = file.tree.state("")?.loaded const pending = loaded ? Promise.resolve() : file.tree.list("") const next = unique([...recent(), ...root()]) if (loaded || next.length > 0) { void pending return next } await pending return unique([...recent(), ...root()]) } if (!query) return [...picks(), ...recent()] if (filesOnly()) { const files = await file.searchFiles(query) return files.map(fileItem) } const [files, nextSessions] = await Promise.all([file.searchFiles(query), Promise.resolve(sessions(query))]) const entries = files.map(fileItem) return [...list(), ...nextSessions, ...entries] } const handleMove = (item: Entry | undefined) => { state.cleanup?.() if (!item) return if (item.type !== "command") return state.cleanup = item.option?.onHighlight?.() } const open = (path: string) => { const value = file.tab(path) tabs().open(value) file.load(path) if (!view().reviewPanel.opened()) view().reviewPanel.open() layout.fileTree.open() layout.fileTree.setTab("all") props.onOpenFile?.(path) } const handleSelect = (item: Entry | undefined) => { if (!item) return state.committed = true state.cleanup = undefined dialog.close() if (item.type === "command") { item.option?.onSelect?.("palette") return } if (item.type === "session") { if (!item.directory || !item.sessionID) return navigate(`/${base64Encode(item.directory)}/session/${item.sessionID}`) return } if (!item.path) return open(item.path) } onCleanup(() => { if (state.committed) return state.cleanup?.() }) return ( item.id} filterKeys={["title", "description", "category"]} groupBy={grouped() ? (item) => item.category : () => ""} onMove={handleMove} onSelect={handleSelect} > {(item) => (
{getDirectory(item.path ?? "")} {getFilename(item.path ?? "")}
} >
{item.title} {item.description}
{formatKeybind(item.keybind ?? "")}
{item.title} {item.description}
{getRelativeTime(new Date(item.updated!).toISOString())}
)}
) }