summaryrefslogtreecommitdiffhomepage
path: root/packages/app/src
diff options
context:
space:
mode:
authorShoubhit Dash <[email protected]>2026-03-19 19:29:29 +0530
committerGitHub <[email protected]>2026-03-19 19:29:29 +0530
commite6f521477959b1009153d8310fb90ac41d766b8b (patch)
treea995a931aa546702cb87076ccac7e895e6a17927 /packages/app/src
parent84f60d97a0c37c9136dfd965a66a0c8685a19e71 (diff)
downloadopencode-e6f521477959b1009153d8310fb90ac41d766b8b.tar.gz
opencode-e6f521477959b1009153d8310fb90ac41d766b8b.zip
feat: add git-backed session review modes (#17961)
Diffstat (limited to 'packages/app/src')
-rw-r--r--packages/app/src/context/global-sync/bootstrap.ts2
-rw-r--r--packages/app/src/context/global-sync/event-reducer.ts4
-rw-r--r--packages/app/src/i18n/en.ts2
-rw-r--r--packages/app/src/pages/session.tsx192
-rw-r--r--packages/app/src/pages/session/session-side-panel.tsx54
-rw-r--r--packages/app/src/pages/session/use-session-commands.tsx6
6 files changed, 194 insertions, 66 deletions
diff --git a/packages/app/src/context/global-sync/bootstrap.ts b/packages/app/src/context/global-sync/bootstrap.ts
index 13494b7ad..c71e00f6f 100644
--- a/packages/app/src/context/global-sync/bootstrap.ts
+++ b/packages/app/src/context/global-sync/bootstrap.ts
@@ -158,7 +158,7 @@ export async function bootstrapDirectory(input: {
input.sdk.vcs.get().then((x) => {
const next = x.data ?? input.store.vcs
input.setStore("vcs", next)
- if (next?.branch) input.vcsCache.setStore("value", next)
+ if (next) input.vcsCache.setStore("value", next)
}),
input.sdk.permission.list().then((x) => {
const grouped = groupBySession(
diff --git a/packages/app/src/context/global-sync/event-reducer.ts b/packages/app/src/context/global-sync/event-reducer.ts
index b8eda0573..0e243e95a 100644
--- a/packages/app/src/context/global-sync/event-reducer.ts
+++ b/packages/app/src/context/global-sync/event-reducer.ts
@@ -268,9 +268,9 @@ export function applyDirectoryEvent(input: {
break
}
case "vcs.branch.updated": {
- const props = event.properties as { branch: string }
+ const props = event.properties as { branch?: string }
if (input.store.vcs?.branch === props.branch) break
- const next = { branch: props.branch }
+ const next = { ...input.store.vcs, branch: props.branch }
input.setStore("vcs", next)
if (input.vcsCache) input.vcsCache.setStore("value", next)
break
diff --git a/packages/app/src/i18n/en.ts b/packages/app/src/i18n/en.ts
index 72caed40a..c7e0c95ea 100644
--- a/packages/app/src/i18n/en.ts
+++ b/packages/app/src/i18n/en.ts
@@ -533,6 +533,8 @@ export const dict = {
"session.review.noVcs.createGit.action": "Create Git repository",
"session.review.noSnapshot": "Snapshot tracking is disabled in config, so session changes are unavailable",
"session.review.noChanges": "No changes",
+ "session.review.noUncommittedChanges": "No uncommitted changes yet",
+ "session.review.noBranchChanges": "No branch changes yet",
"session.files.selectToOpen": "Select a file to open",
"session.files.all": "All files",
diff --git a/packages/app/src/pages/session.tsx b/packages/app/src/pages/session.tsx
index 6d2917008..6fe2e56b8 100644
--- a/packages/app/src/pages/session.tsx
+++ b/packages/app/src/pages/session.tsx
@@ -1,4 +1,4 @@
-import type { Project, UserMessage } from "@opencode-ai/sdk/v2"
+import type { FileDiff, Project, UserMessage } from "@opencode-ai/sdk/v2"
import { useDialog } from "@opencode-ai/ui/context/dialog"
import {
batch,
@@ -57,6 +57,9 @@ import { formatServerError } from "@/utils/server-errors"
const emptyUserMessages: UserMessage[] = []
const emptyFollowups: (FollowupDraft & { id: string })[] = []
+type ChangeMode = "git" | "branch" | "session" | "turn"
+type VcsMode = "git" | "branch"
+
type SessionHistoryWindowInput = {
sessionID: () => string | undefined
messagesReady: () => boolean
@@ -415,15 +418,16 @@ export default function Page() {
const info = createMemo(() => (params.id ? sync.session.get(params.id) : undefined))
const diffs = createMemo(() => (params.id ? (sync.data.session_diff[params.id] ?? []) : []))
- const reviewCount = createMemo(() => Math.max(info()?.summary?.files ?? 0, diffs().length))
- const hasReview = createMemo(() => reviewCount() > 0)
+ const sessionCount = createMemo(() => Math.max(info()?.summary?.files ?? 0, diffs().length))
+ const hasSessionReview = createMemo(() => sessionCount() > 0)
+ const canReview = createMemo(() => !!params.id)
const reviewTab = createMemo(() => isDesktop())
const tabState = createSessionTabs({
tabs,
pathFromTab: file.pathFromTab,
normalizeTab,
review: reviewTab,
- hasReview,
+ hasReview: canReview,
})
const contextOpen = tabState.contextOpen
const openedTabs = tabState.openedTabs
@@ -499,11 +503,22 @@ export default function Page() {
const [store, setStore] = createStore({
messageId: undefined as string | undefined,
mobileTab: "session" as "session" | "changes",
- changes: "session" as "session" | "turn",
+ changes: "git" as ChangeMode,
newSessionWorktree: "main",
deferRender: false,
})
+ const [vcs, setVcs] = createStore({
+ diff: {
+ git: [] as FileDiff[],
+ branch: [] as FileDiff[],
+ },
+ ready: {
+ git: false,
+ branch: false,
+ },
+ })
+
const [followup, setFollowup] = createStore({
items: {} as Record<string, (FollowupDraft & { id: string })[] | undefined>,
sending: {} as Record<string, string | undefined>,
@@ -531,6 +546,40 @@ export default function Page() {
let refreshTimer: number | undefined
let diffFrame: number | undefined
let diffTimer: number | undefined
+ const vcsTask = new Map<VcsMode, Promise<void>>()
+
+ const resetVcs = () => {
+ vcsTask.clear()
+ setVcs({
+ diff: { git: [], branch: [] },
+ ready: { git: false, branch: false },
+ })
+ }
+
+ const loadVcs = (mode: VcsMode, force = false) => {
+ if (sync.project?.vcs !== "git") return Promise.resolve()
+ if (vcs.ready[mode] && !force) return Promise.resolve()
+ const current = vcsTask.get(mode)
+ if (current) return current
+
+ const task = sdk.client.vcs
+ .diff({ mode })
+ .then((result) => {
+ setVcs("diff", mode, result.data ?? [])
+ setVcs("ready", mode, true)
+ })
+ .catch((error) => {
+ console.debug("[session-review] failed to load vcs diff", { mode, error })
+ setVcs("diff", mode, [])
+ setVcs("ready", mode, true)
+ })
+ .finally(() => {
+ vcsTask.delete(mode)
+ })
+
+ vcsTask.set(mode, task)
+ return task
+ }
createComputed((prev) => {
const open = desktopReviewOpen()
@@ -546,7 +595,38 @@ export default function Page() {
}, desktopReviewOpen())
const turnDiffs = createMemo(() => lastUserMessage()?.summary?.diffs ?? [])
- const reviewDiffs = createMemo(() => (store.changes === "session" ? diffs() : turnDiffs()))
+ const changesOptions = createMemo<ChangeMode[]>(() => {
+ const list: ChangeMode[] = []
+ const git = sync.project?.vcs === "git"
+ if (git) list.push("git")
+ if (git && sync.data.vcs?.branch && sync.data.vcs?.default_branch && sync.data.vcs.branch !== sync.data.vcs.default_branch) {
+ list.push("branch")
+ }
+ list.push("session", "turn")
+ return list
+ })
+ const vcsMode = createMemo<VcsMode | undefined>(() => {
+ if (store.changes === "git" || store.changes === "branch") return store.changes
+ })
+ const reviewDiffs = createMemo(() => {
+ if (store.changes === "git") return vcs.diff.git
+ if (store.changes === "branch") return vcs.diff.branch
+ if (store.changes === "session") return diffs()
+ return turnDiffs()
+ })
+ const reviewCount = createMemo(() => {
+ if (store.changes === "git") return vcs.diff.git.length
+ if (store.changes === "branch") return vcs.diff.branch.length
+ if (store.changes === "session") return sessionCount()
+ return turnDiffs().length
+ })
+ const hasReview = createMemo(() => reviewCount() > 0)
+ const reviewReady = createMemo(() => {
+ if (store.changes === "git") return vcs.ready.git
+ if (store.changes === "branch") return vcs.ready.branch
+ if (store.changes === "session") return !hasSessionReview() || diffsReady()
+ return true
+ })
const newSessionWorktree = createMemo(() => {
if (store.newSessionWorktree === "create") return "create"
@@ -615,10 +695,10 @@ export default function Page() {
const diffsReady = createMemo(() => {
const id = params.id
if (!id) return true
- if (!hasReview()) return true
+ if (!hasSessionReview()) return true
return sync.data.session_diff[id] !== undefined
})
- const reviewEmptyKey = createMemo(() => {
+ const sessionEmptyKey = createMemo(() => {
const project = sync.project
if (project && !project.vcs) return "session.review.noVcs"
if (sync.data.config.snapshot === false) return "session.review.noSnapshot"
@@ -741,7 +821,7 @@ export default function Page() {
sessionKey,
() => {
setStore("messageId", undefined)
- setStore("changes", "session")
+ setStore("changes", "git")
setUi("pendingMessage", undefined)
},
{ defer: true },
@@ -750,6 +830,16 @@ export default function Page() {
createEffect(
on(
+ () => sdk.directory,
+ () => {
+ resetVcs()
+ },
+ { defer: true },
+ ),
+ )
+
+ createEffect(
+ on(
() => params.dir,
(dir) => {
if (!dir) return
@@ -870,6 +960,38 @@ export default function Page() {
}
const mobileChanges = createMemo(() => !isDesktop() && store.mobileTab === "changes")
+ const wantsReview = createMemo(() =>
+ isDesktop() ? desktopFileTreeOpen() || (desktopReviewOpen() && activeTab() === "review") : store.mobileTab === "changes",
+ )
+
+ createEffect(() => {
+ const list = changesOptions()
+ if (list.includes(store.changes)) return
+ const next = list[0]
+ if (!next) return
+ setStore("changes", next)
+ })
+
+ createEffect(() => {
+ const mode = vcsMode()
+ if (!mode) return
+ if (!wantsReview()) return
+ void loadVcs(mode)
+ })
+
+ createEffect(
+ on(
+ () => sync.data.session_status[params.id ?? ""]?.type,
+ (next, prev) => {
+ const mode = vcsMode()
+ if (!mode) return
+ if (!wantsReview()) return
+ if (next !== "idle" || prev === undefined || prev === "idle") return
+ void loadVcs(mode, true)
+ },
+ { defer: true },
+ ),
+ )
const fileTreeTab = () => layout.fileTree.tab()
const setFileTreeTab = (value: "changes" | "all") => layout.fileTree.setTab(value)
@@ -916,21 +1038,23 @@ export default function Page() {
loadFile: file.load,
})
- const changesOptions = ["session", "turn"] as const
- const changesOptionsList = [...changesOptions]
-
const changesTitle = () => {
- if (!hasReview()) {
+ if (!canReview()) {
return null
}
+ const label = (option: ChangeMode) => {
+ if (option === "git") return language.t("ui.sessionReview.title.git")
+ if (option === "branch") return language.t("ui.sessionReview.title.branch")
+ if (option === "session") return language.t("ui.sessionReview.title")
+ return language.t("ui.sessionReview.title.lastTurn")
+ }
+
return (
<Select
- options={changesOptionsList}
+ options={changesOptions()}
current={store.changes}
- label={(option) =>
- option === "session" ? language.t("ui.sessionReview.title") : language.t("ui.sessionReview.title.lastTurn")
- }
+ label={label}
onSelect={(option) => option && setStore("changes", option)}
variant="ghost"
size="small"
@@ -939,20 +1063,34 @@ export default function Page() {
)
}
- const emptyTurn = () => (
+ const empty = (text: string) => (
<div class="h-full pb-64 -mt-4 flex flex-col items-center justify-center text-center gap-6">
- <div class="text-14-regular text-text-weak max-w-56">{language.t("session.review.noChanges")}</div>
+ <div class="text-14-regular text-text-weak max-w-56">{text}</div>
</div>
)
+ const reviewEmptyText = createMemo(() => {
+ if (store.changes === "git") return language.t("session.review.noUncommittedChanges")
+ if (store.changes === "branch") return language.t("session.review.noBranchChanges")
+ if (store.changes === "turn") return language.t("session.review.noChanges")
+ return language.t(sessionEmptyKey())
+ })
+
const reviewEmpty = (input: { loadingClass: string; emptyClass: string }) => {
- if (store.changes === "turn") return emptyTurn()
+ if (store.changes === "git" || store.changes === "branch") {
+ if (!reviewReady()) return <div class={input.loadingClass}>{language.t("session.review.loadingChanges")}</div>
+ return empty(reviewEmptyText())
+ }
+
+ if (store.changes === "turn") {
+ return empty(reviewEmptyText())
+ }
- if (hasReview() && !diffsReady()) {
+ if (hasSessionReview() && !diffsReady()) {
return <div class={input.loadingClass}>{language.t("session.review.loadingChanges")}</div>
}
- if (reviewEmptyKey() === "session.review.noVcs") {
+ if (sessionEmptyKey() === "session.review.noVcs") {
return (
<div class={input.emptyClass}>
<div class="flex flex-col gap-3">
@@ -972,7 +1110,7 @@ export default function Page() {
return (
<div class={input.emptyClass}>
- <div class="text-14-regular text-text-weak max-w-56">{language.t(reviewEmptyKey())}</div>
+ <div class="text-14-regular text-text-weak max-w-56">{reviewEmptyText()}</div>
</div>
)
}
@@ -1076,7 +1214,7 @@ export default function Page() {
const pending = tree.pendingDiff
if (!pending) return
if (!tree.reviewScroll) return
- if (!diffsReady()) return
+ if (!reviewReady()) return
const attempt = (count: number) => {
if (tree.pendingDiff !== pending) return
@@ -1808,6 +1946,12 @@ export default function Page() {
</div>
<SessionSidePanel
+ canReview={canReview}
+ diffs={reviewDiffs}
+ diffsReady={reviewReady}
+ empty={reviewEmptyText}
+ hasReview={hasReview}
+ reviewCount={reviewCount}
reviewPanel={reviewPanel}
activeDiff={tree.activeDiff}
focusReviewDiff={focusReviewDiff}
diff --git a/packages/app/src/pages/session/session-side-panel.tsx b/packages/app/src/pages/session/session-side-panel.tsx
index 3b8b0c96b..ed90c97dc 100644
--- a/packages/app/src/pages/session/session-side-panel.tsx
+++ b/packages/app/src/pages/session/session-side-panel.tsx
@@ -8,6 +8,7 @@ import { ResizeHandle } from "@opencode-ai/ui/resize-handle"
import { Mark } from "@opencode-ai/ui/logo"
import { DragDropProvider, DragDropSensors, DragOverlay, SortableProvider, closestCenter } from "@thisbeyond/solid-dnd"
import type { DragEvent } from "@thisbeyond/solid-dnd"
+import type { FileDiff } from "@opencode-ai/sdk/v2"
import { ConstrainDragYAxis, getDraggableId } from "@/utils/solid-dnd"
import { useDialog } from "@opencode-ai/ui/context/dialog"
@@ -19,7 +20,6 @@ import { useCommand } from "@/context/command"
import { useFile, type SelectedLineRange } from "@/context/file"
import { useLanguage } from "@/context/language"
import { useLayout } from "@/context/layout"
-import { useSync } from "@/context/sync"
import { createFileTabListSync } from "@/pages/session/file-tab-scroll"
import { FileTabContent } from "@/pages/session/file-tabs"
import { createOpenSessionFileTab, createSessionTabs, getTabReorderIndex, type Sizing } from "@/pages/session/helpers"
@@ -27,6 +27,12 @@ import { setSessionHandoff } from "@/pages/session/handoff"
import { useSessionLayout } from "@/pages/session/session-layout"
export function SessionSidePanel(props: {
+ canReview: () => boolean
+ diffs: () => FileDiff[]
+ diffsReady: () => boolean
+ empty: () => string
+ hasReview: () => boolean
+ reviewCount: () => number
reviewPanel: () => JSX.Element
activeDiff?: string
focusReviewDiff: (path: string) => void
@@ -34,12 +40,11 @@ export function SessionSidePanel(props: {
size: Sizing
}) {
const layout = useLayout()
- const sync = useSync()
const file = useFile()
const language = useLanguage()
const command = useCommand()
const dialog = useDialog()
- const { params, sessionKey, tabs, view } = useSessionLayout()
+ const { sessionKey, tabs, view } = useSessionLayout()
const isDesktop = createMediaQuery("(min-width: 768px)")
@@ -54,24 +59,7 @@ export function SessionSidePanel(props: {
})
const treeWidth = createMemo(() => (fileOpen() ? `${layout.fileTree.width()}px` : "0px"))
- const info = createMemo(() => (params.id ? sync.session.get(params.id) : undefined))
- const diffs = createMemo(() => (params.id ? (sync.data.session_diff[params.id] ?? []) : []))
- const reviewCount = createMemo(() => Math.max(info()?.summary?.files ?? 0, diffs().length))
- const hasReview = createMemo(() => reviewCount() > 0)
- const diffsReady = createMemo(() => {
- const id = params.id
- if (!id) return true
- if (!hasReview()) return true
- return sync.data.session_diff[id] !== undefined
- })
-
- const reviewEmptyKey = createMemo(() => {
- if (sync.project && !sync.project.vcs) return "session.review.noVcs"
- if (sync.data.config.snapshot === false) return "session.review.noSnapshot"
- return "session.review.noChanges"
- })
-
- const diffFiles = createMemo(() => diffs().map((d) => d.file))
+ const diffFiles = createMemo(() => props.diffs().map((d) => d.file))
const kinds = createMemo(() => {
const merge = (a: "add" | "del" | "mix" | undefined, b: "add" | "del" | "mix") => {
if (!a) return b
@@ -82,7 +70,7 @@ export function SessionSidePanel(props: {
const normalize = (p: string) => p.replaceAll("\\\\", "/").replace(/\/+$/, "")
const out = new Map<string, "add" | "del" | "mix">()
- for (const diff of diffs()) {
+ for (const diff of props.diffs()) {
const file = normalize(diff.file)
const kind = diff.status === "added" ? "add" : diff.status === "deleted" ? "del" : "mix"
@@ -136,7 +124,7 @@ export function SessionSidePanel(props: {
pathFromTab: file.pathFromTab,
normalizeTab,
review: reviewTab,
- hasReview,
+ hasReview: props.canReview,
})
const contextOpen = tabState.contextOpen
const openedTabs = tabState.openedTabs
@@ -241,12 +229,12 @@ export function SessionSidePanel(props: {
onCleanup(stop)
}}
>
- <Show when={reviewTab()}>
+ <Show when={reviewTab() && props.canReview()}>
<Tabs.Trigger value="review">
<div class="flex items-center gap-1.5">
<div>{language.t("session.tab.review")}</div>
- <Show when={hasReview()}>
- <div>{reviewCount()}</div>
+ <Show when={props.hasReview()}>
+ <div>{props.reviewCount()}</div>
</Show>
</div>
</Tabs.Trigger>
@@ -303,7 +291,7 @@ export function SessionSidePanel(props: {
</Tabs.List>
</div>
- <Show when={reviewTab()}>
+ <Show when={reviewTab() && props.canReview()}>
<Tabs.Content value="review" class="flex flex-col h-full overflow-hidden contain-strict">
<Show when={activeTab() === "review"}>{props.reviewPanel()}</Show>
</Tabs.Content>
@@ -377,8 +365,8 @@ export function SessionSidePanel(props: {
>
<Tabs.List>
<Tabs.Trigger value="changes" class="flex-1" classes={{ button: "w-full" }}>
- {reviewCount()}{" "}
- {language.t(reviewCount() === 1 ? "session.review.change.one" : "session.review.change.other")}
+ {props.reviewCount()}{" "}
+ {language.t(props.reviewCount() === 1 ? "session.review.change.one" : "session.review.change.other")}
</Tabs.Trigger>
<Tabs.Trigger value="all" class="flex-1" classes={{ button: "w-full" }}>
{language.t("session.files.all")}
@@ -386,9 +374,9 @@ export function SessionSidePanel(props: {
</Tabs.List>
<Tabs.Content value="changes" class="bg-background-stronger px-3 py-0">
<Switch>
- <Match when={hasReview()}>
+ <Match when={props.hasReview() || !props.diffsReady()}>
<Show
- when={diffsReady()}
+ when={props.diffsReady()}
fallback={
<div class="px-2 py-2 text-12-regular text-text-weak">
{language.t("common.loading")}
@@ -408,9 +396,7 @@ export function SessionSidePanel(props: {
</Show>
</Match>
<Match when={true}>
- {empty(
- language.t(sync.project && !sync.project.vcs ? "session.review.noChanges" : reviewEmptyKey()),
- )}
+ {empty(props.empty())}
</Match>
</Switch>
</Tabs.Content>
diff --git a/packages/app/src/pages/session/use-session-commands.tsx b/packages/app/src/pages/session/use-session-commands.tsx
index 1a2e777f5..f45374359 100644
--- a/packages/app/src/pages/session/use-session-commands.tsx
+++ b/packages/app/src/pages/session/use-session-commands.tsx
@@ -56,11 +56,7 @@ export const useSessionCommands = (actions: SessionCommandContext) => {
if (!id) return
return sync.session.get(id)
}
- const hasReview = () => {
- const id = params.id
- if (!id) return false
- return Math.max(info()?.summary?.files ?? 0, (sync.data.session_diff[id] ?? []).length) > 0
- }
+ const hasReview = () => !!params.id
const normalizeTab = (tab: string) => {
if (!tab.startsWith("file://")) return tab
return file.tab(tab)