import { AppIcon } from "@opencode-ai/ui/app-icon" import { Button } from "@opencode-ai/ui/button" import { DropdownMenu } from "@opencode-ai/ui/dropdown-menu" import { Icon } from "@opencode-ai/ui/icon" import { IconButton } from "@opencode-ai/ui/icon-button" import { Keybind } from "@opencode-ai/ui/keybind" import { Spinner } from "@opencode-ai/ui/spinner" import { showToast } from "@opencode-ai/ui/toast" import { Tooltip, TooltipKeybind } from "@opencode-ai/ui/tooltip" import { getFilename } from "@opencode-ai/util/path" import { createEffect, createMemo, For, onCleanup, Show } from "solid-js" import { createStore } from "solid-js/store" import { Portal } from "solid-js/web" import { useCommand } from "@/context/command" import { useLanguage } from "@/context/language" import { useLayout } from "@/context/layout" import { usePlatform } from "@/context/platform" import { useServer } from "@/context/server" import { useSettings } from "@/context/settings" import { useSync } from "@/context/sync" import { useTerminal } from "@/context/terminal" import { focusTerminalById } from "@/pages/session/helpers" import { useSessionLayout } from "@/pages/session/session-layout" import { messageAgentColor } from "@/utils/agent" import { decode64 } from "@/utils/base64" import { Persist, persisted } from "@/utils/persist" import { StatusPopover } from "../status-popover" const OPEN_APPS = [ "vscode", "cursor", "zed", "textmate", "antigravity", "finder", "terminal", "iterm2", "ghostty", "warp", "xcode", "android-studio", "powershell", "sublime-text", ] as const type OpenApp = (typeof OPEN_APPS)[number] type OS = "macos" | "windows" | "linux" | "unknown" const MAC_APPS = [ { id: "vscode", label: "session.header.open.app.vscode", icon: "vscode", openWith: "Visual Studio Code", }, { id: "cursor", label: "session.header.open.app.cursor", icon: "cursor", openWith: "Cursor" }, { id: "zed", label: "session.header.open.app.zed", icon: "zed", openWith: "Zed" }, { id: "textmate", label: "session.header.open.app.textmate", icon: "textmate", openWith: "TextMate" }, { id: "antigravity", label: "session.header.open.app.antigravity", icon: "antigravity", openWith: "Antigravity", }, { id: "terminal", label: "session.header.open.app.terminal", icon: "terminal", openWith: "Terminal" }, { id: "iterm2", label: "session.header.open.app.iterm2", icon: "iterm2", openWith: "iTerm" }, { id: "ghostty", label: "session.header.open.app.ghostty", icon: "ghostty", openWith: "Ghostty" }, { id: "warp", label: "session.header.open.app.warp", icon: "warp", openWith: "Warp" }, { id: "xcode", label: "session.header.open.app.xcode", icon: "xcode", openWith: "Xcode" }, { id: "android-studio", label: "session.header.open.app.androidStudio", icon: "android-studio", openWith: "Android Studio", }, { id: "sublime-text", label: "session.header.open.app.sublimeText", icon: "sublime-text", openWith: "Sublime Text", }, ] as const const WINDOWS_APPS = [ { id: "vscode", label: "session.header.open.app.vscode", icon: "vscode", openWith: "code" }, { id: "cursor", label: "session.header.open.app.cursor", icon: "cursor", openWith: "cursor" }, { id: "zed", label: "session.header.open.app.zed", icon: "zed", openWith: "zed" }, { id: "powershell", label: "session.header.open.app.powershell", icon: "powershell", openWith: "powershell", }, { id: "sublime-text", label: "session.header.open.app.sublimeText", icon: "sublime-text", openWith: "Sublime Text", }, ] as const const LINUX_APPS = [ { id: "vscode", label: "session.header.open.app.vscode", icon: "vscode", openWith: "code" }, { id: "cursor", label: "session.header.open.app.cursor", icon: "cursor", openWith: "cursor" }, { id: "zed", label: "session.header.open.app.zed", icon: "zed", openWith: "zed" }, { id: "sublime-text", label: "session.header.open.app.sublimeText", icon: "sublime-text", openWith: "Sublime Text", }, ] as const const detectOS = (platform: ReturnType): OS => { if (platform.platform === "desktop" && platform.os) return platform.os if (typeof navigator !== "object") return "unknown" const value = navigator.platform || navigator.userAgent if (/Mac/i.test(value)) return "macos" if (/Win/i.test(value)) return "windows" if (/Linux/i.test(value)) return "linux" return "unknown" } const showRequestError = (language: ReturnType, err: unknown) => { showToast({ variant: "error", title: language.t("common.requestFailed"), description: err instanceof Error ? err.message : String(err), }) } export function SessionHeader() { const layout = useLayout() const command = useCommand() const server = useServer() const platform = usePlatform() const language = useLanguage() const settings = useSettings() const sync = useSync() const terminal = useTerminal() const { params, view } = useSessionLayout() 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 name = createMemo(() => { const current = project() if (current) return current.name || getFilename(current.worktree) return getFilename(projectDirectory()) }) const hotkey = createMemo(() => command.keybind("file.open")) const os = createMemo(() => detectOS(platform)) const search = createMemo(() => platform.platform !== "desktop" || settings.general.showSearch()) const tree = createMemo(() => platform.platform !== "desktop" || settings.general.showFileTree()) const term = createMemo(() => platform.platform !== "desktop" || settings.general.showTerminal()) const status = createMemo(() => platform.platform !== "desktop" || settings.general.showStatus()) const [exists, setExists] = createStore>>({ finder: true, }) const apps = createMemo(() => { if (os() === "macos") return MAC_APPS if (os() === "windows") return WINDOWS_APPS return LINUX_APPS }) const fileManager = createMemo(() => { if (os() === "macos") return { label: "session.header.open.finder", icon: "finder" as const } if (os() === "windows") return { label: "session.header.open.fileExplorer", icon: "file-explorer" as const } return { label: "session.header.open.fileManager", icon: "finder" as const } }) createEffect(() => { if (platform.platform !== "desktop") return if (!platform.checkAppExists) return const list = apps() setExists(Object.fromEntries(list.map((app) => [app.id, undefined])) as Partial>) void Promise.all( list.map((app) => Promise.resolve(platform.checkAppExists?.(app.openWith)) .then((value) => Boolean(value)) .catch(() => false) .then((ok) => [app.id, ok] as const), ), ).then((entries) => { setExists(Object.fromEntries(entries) as Partial>) }) }) const options = createMemo(() => { return [ { id: "finder", label: language.t(fileManager().label), icon: fileManager().icon }, ...apps() .filter((app) => exists[app.id]) .map((app) => ({ ...app, label: language.t(app.label) })), ] as const }) const toggleTerminal = () => { const next = !view().terminal.opened() view().terminal.toggle() if (!next) return const id = terminal.active() if (!id) return focusTerminalById(id) } const [prefs, setPrefs] = persisted(Persist.global("open.app"), createStore({ app: "finder" as OpenApp })) const [menu, setMenu] = createStore({ open: false }) const [openRequest, setOpenRequest] = createStore({ app: undefined as OpenApp | undefined, }) const canOpen = createMemo(() => platform.platform === "desktop" && !!platform.openPath && server.isLocal()) const current = createMemo( () => options().find((o) => o.id === prefs.app) ?? options()[0] ?? ({ id: "finder", label: fileManager().label, icon: fileManager().icon } as const), ) const opening = createMemo(() => openRequest.app !== undefined) const tint = createMemo(() => messageAgentColor(params.id ? sync.data.message[params.id] : undefined, sync.data.agent), ) const selectApp = (app: OpenApp) => { if (!options().some((item) => item.id === app)) return setPrefs("app", app) } const openDir = (app: OpenApp) => { if (opening() || !canOpen() || !platform.openPath) return const directory = projectDirectory() if (!directory) return const item = options().find((o) => o.id === app) const openWith = item && "openWith" in item ? item.openWith : undefined setOpenRequest("app", app) platform .openPath(directory, openWith) .catch((err: unknown) => showRequestError(language, err)) .finally(() => { setOpenRequest("app", undefined) }) } const copyPath = () => { const directory = projectDirectory() if (!directory) return navigator.clipboard .writeText(directory) .then(() => { showToast({ variant: "success", icon: "circle-check", title: language.t("session.share.copy.copied"), description: directory, }) }) .catch((err: unknown) => showRequestError(language, err)) } const centerMount = createMemo(() => document.getElementById("opencode-titlebar-center")) const rightMount = createMemo(() => document.getElementById("opencode-titlebar-right")) return ( <> {(mount) => ( )} {(mount) => (
} >
setMenu("open", open)} > {language.t("session.header.openIn")} { if (!OPEN_APPS.includes(value as OpenApp)) return selectApp(value as OpenApp) }} > {(o) => ( { setMenu("open", false) openDir(o.id) }} >
{o.label}
)}
{ setMenu("open", false) copyPath() }} >
{language.t("session.header.open.copyPath")}
)} ) }