session-header.tsx 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. import { createMemo, createResource, Show } from "solid-js"
  2. import { A, useNavigate, useParams } from "@solidjs/router"
  3. import { useLayout } from "@/context/layout"
  4. import { useCommand } from "@/context/command"
  5. import { useServer } from "@/context/server"
  6. import { useDialog } from "@opencode-ai/ui/context/dialog"
  7. import { useSync } from "@/context/sync"
  8. import { useGlobalSDK } from "@/context/global-sdk"
  9. import { getFilename } from "@opencode-ai/util/path"
  10. import { base64Decode, base64Encode } from "@opencode-ai/util/encode"
  11. import { iife } from "@opencode-ai/util/iife"
  12. import { Icon } from "@opencode-ai/ui/icon"
  13. import { IconButton } from "@opencode-ai/ui/icon-button"
  14. import { Button } from "@opencode-ai/ui/button"
  15. import { Tooltip, TooltipKeybind } from "@opencode-ai/ui/tooltip"
  16. import { Select } from "@opencode-ai/ui/select"
  17. import { Popover } from "@opencode-ai/ui/popover"
  18. import { TextField } from "@opencode-ai/ui/text-field"
  19. import { DialogSelectServer } from "@/components/dialog-select-server"
  20. import { SessionLspIndicator } from "@/components/session-lsp-indicator"
  21. import { SessionMcpIndicator } from "@/components/session-mcp-indicator"
  22. import type { Session } from "@opencode-ai/sdk/v2/client"
  23. import { same } from "@/utils/same"
  24. export function SessionHeader() {
  25. const globalSDK = useGlobalSDK()
  26. const layout = useLayout()
  27. const params = useParams()
  28. const navigate = useNavigate()
  29. const command = useCommand()
  30. const server = useServer()
  31. const dialog = useDialog()
  32. const sync = useSync()
  33. const projectDirectory = createMemo(() => base64Decode(params.dir ?? ""))
  34. const sessions = createMemo(() => (sync.data.session ?? []).filter((s) => !s.parentID))
  35. const currentSession = createMemo(() => sync.data.session.find((s) => s.id === params.id))
  36. const parentSession = createMemo(() => {
  37. const current = currentSession()
  38. if (!current?.parentID) return undefined
  39. return sync.data.session.find((s) => s.id === current.parentID)
  40. })
  41. const shareEnabled = createMemo(() => sync.data.config.share !== "disabled")
  42. const worktrees = createMemo(() => layout.projects.list().map((p) => p.worktree), [], { equals: same })
  43. const sessionKey = createMemo(() => `${params.dir}${params.id ? "/" + params.id : ""}`)
  44. const view = createMemo(() => layout.view(sessionKey()))
  45. function navigateToProject(directory: string) {
  46. navigate(`/${base64Encode(directory)}`)
  47. }
  48. function navigateToSession(session: Session | undefined) {
  49. if (!session) return
  50. // Only navigate if we're actually changing to a different session
  51. if (session.id === params.id) return
  52. navigate(`/${params.dir}/session/${session.id}`)
  53. }
  54. return (
  55. <header class="h-12 shrink-0 bg-background-base border-b border-border-weak-base flex">
  56. <button
  57. type="button"
  58. class="xl:hidden w-12 shrink-0 flex items-center justify-center border-r border-border-weak-base hover:bg-surface-raised-base-hover active:bg-surface-raised-base-active transition-colors"
  59. onClick={layout.mobileSidebar.toggle}
  60. >
  61. <Icon name="menu" size="small" />
  62. </button>
  63. <div class="px-4 flex items-center justify-between gap-4 w-full">
  64. <div class="flex items-center gap-3 min-w-0">
  65. <div class="flex items-center gap-2 min-w-0">
  66. <div class="hidden xl:flex items-center gap-2">
  67. <Select
  68. options={worktrees()}
  69. current={sync.project?.worktree ?? projectDirectory()}
  70. label={(x) => getFilename(x)}
  71. onSelect={(x) => (x ? navigateToProject(x) : undefined)}
  72. class="text-14-regular text-text-base"
  73. variant="ghost"
  74. >
  75. {/* @ts-ignore */}
  76. {(i) => (
  77. <div class="flex items-center gap-2">
  78. <Icon name="folder" size="small" />
  79. <div class="text-text-strong">{getFilename(i)}</div>
  80. </div>
  81. )}
  82. </Select>
  83. <div class="text-text-weaker">/</div>
  84. </div>
  85. <Show
  86. when={parentSession()}
  87. fallback={
  88. <>
  89. <Select
  90. options={sessions()}
  91. current={currentSession()}
  92. placeholder="New session"
  93. label={(x) => x.title}
  94. value={(x) => x.id}
  95. onSelect={navigateToSession}
  96. class="text-14-regular text-text-base max-w-[calc(100vw-180px)] md:max-w-md"
  97. variant="ghost"
  98. />
  99. </>
  100. }
  101. >
  102. <div class="flex items-center gap-2 min-w-0">
  103. <Select
  104. options={sessions()}
  105. current={parentSession()}
  106. placeholder="Back to parent session"
  107. label={(x) => x.title}
  108. value={(x) => x.id}
  109. onSelect={(session) => {
  110. // Only navigate if selecting a different session than current parent
  111. const currentParent = parentSession()
  112. if (session && currentParent && session.id !== currentParent.id) {
  113. navigateToSession(session)
  114. }
  115. }}
  116. class="text-14-regular text-text-base max-w-[calc(100vw-180px)] md:max-w-md"
  117. variant="ghost"
  118. />
  119. <div class="text-text-weaker">/</div>
  120. <div class="flex items-center gap-1.5 min-w-0">
  121. <Tooltip value="Back to parent session">
  122. <button
  123. type="button"
  124. class="flex items-center justify-center gap-1 p-1 rounded hover:bg-surface-raised-base-hover active:bg-surface-raised-base-active transition-colors flex-shrink-0"
  125. onClick={() => navigateToSession(parentSession())}
  126. >
  127. <Icon name="arrow-left" size="small" class="text-icon-base" />
  128. </button>
  129. </Tooltip>
  130. </div>
  131. </div>
  132. </Show>
  133. </div>
  134. <Show when={currentSession() && !parentSession()}>
  135. <TooltipKeybind class="hidden xl:block" title="New session" keybind={command.keybind("session.new")}>
  136. <IconButton as={A} href={`/${params.dir}/session`} icon="edit-small-2" variant="ghost" />
  137. </TooltipKeybind>
  138. </Show>
  139. </div>
  140. <div class="flex items-center gap-3">
  141. <div class="hidden md:flex items-center gap-1">
  142. <Button
  143. size="small"
  144. variant="ghost"
  145. onClick={() => {
  146. dialog.show(() => <DialogSelectServer />)
  147. }}
  148. >
  149. <div
  150. classList={{
  151. "size-1.5 rounded-full": true,
  152. "bg-icon-success-base": server.healthy() === true,
  153. "bg-icon-critical-base": server.healthy() === false,
  154. "bg-border-weak-base": server.healthy() === undefined,
  155. }}
  156. />
  157. <Icon name="server" size="small" class="text-icon-weak" />
  158. <span class="text-12-regular text-text-weak truncate max-w-[200px]">{server.name}</span>
  159. </Button>
  160. <SessionLspIndicator />
  161. <SessionMcpIndicator />
  162. </div>
  163. <div class="flex items-center gap-1">
  164. <Show when={currentSession()?.summary?.files}>
  165. <TooltipKeybind
  166. class="hidden md:block shrink-0"
  167. title="Toggle review"
  168. keybind={command.keybind("review.toggle")}
  169. >
  170. <Button
  171. variant="ghost"
  172. class="group/review-toggle size-6 p-0"
  173. onClick={() => view().reviewPanel.toggle()}
  174. >
  175. <div class="relative flex items-center justify-center size-4 [&>*]:absolute [&>*]:inset-0">
  176. <Icon
  177. name={view().reviewPanel.opened() ? "layout-right" : "layout-left"}
  178. size="small"
  179. class="group-hover/review-toggle:hidden"
  180. />
  181. <Icon
  182. name={view().reviewPanel.opened() ? "layout-right-partial" : "layout-left-partial"}
  183. size="small"
  184. class="hidden group-hover/review-toggle:inline-block"
  185. />
  186. <Icon
  187. name={view().reviewPanel.opened() ? "layout-right-full" : "layout-left-full"}
  188. size="small"
  189. class="hidden group-active/review-toggle:inline-block"
  190. />
  191. </div>
  192. </Button>
  193. </TooltipKeybind>
  194. </Show>
  195. <TooltipKeybind
  196. class="hidden md:block shrink-0"
  197. title="Toggle terminal"
  198. keybind={command.keybind("terminal.toggle")}
  199. >
  200. <Button variant="ghost" class="group/terminal-toggle size-6 p-0" onClick={() => view().terminal.toggle()}>
  201. <div class="relative flex items-center justify-center size-4 [&>*]:absolute [&>*]:inset-0">
  202. <Icon
  203. size="small"
  204. name={view().terminal.opened() ? "layout-bottom-full" : "layout-bottom"}
  205. class="group-hover/terminal-toggle:hidden"
  206. />
  207. <Icon
  208. size="small"
  209. name="layout-bottom-partial"
  210. class="hidden group-hover/terminal-toggle:inline-block"
  211. />
  212. <Icon
  213. size="small"
  214. name={view().terminal.opened() ? "layout-bottom" : "layout-bottom-full"}
  215. class="hidden group-active/terminal-toggle:inline-block"
  216. />
  217. </div>
  218. </Button>
  219. </TooltipKeybind>
  220. </div>
  221. <Show when={shareEnabled() && currentSession()}>
  222. <Popover
  223. title="Share session"
  224. trigger={
  225. <Tooltip class="shrink-0" value="Share session">
  226. <IconButton icon="share" variant="ghost" class="" />
  227. </Tooltip>
  228. }
  229. >
  230. {iife(() => {
  231. const [url] = createResource(
  232. () => currentSession(),
  233. async (session) => {
  234. if (!session) return
  235. let shareURL = session.share?.url
  236. if (!shareURL) {
  237. shareURL = await globalSDK.client.session
  238. .share({ sessionID: session.id, directory: projectDirectory() })
  239. .then((r) => r.data?.share?.url)
  240. .catch((e) => {
  241. console.error("Failed to share session", e)
  242. return undefined
  243. })
  244. }
  245. return shareURL
  246. },
  247. { initialValue: "" },
  248. )
  249. return (
  250. <Show when={url.latest}>
  251. {(shareUrl) => <TextField value={shareUrl()} readOnly copyable class="w-72" />}
  252. </Show>
  253. )
  254. })}
  255. </Popover>
  256. </Show>
  257. </div>
  258. </div>
  259. </header>
  260. )
  261. }