summaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
authoradamelmore <[email protected]>2026-01-27 15:55:26 -0600
committeradamelmore <[email protected]>2026-01-27 15:59:01 -0600
commit605e5335588adeb96bc9549553e9976b84338a8d (patch)
tree0a588ea29f69b113bd0d91e92c7b25d594ce38ca
parent33d400c567a48fc730b2e95f307b9edd99132cc0 (diff)
downloadopencode-605e5335588adeb96bc9549553e9976b84338a8d.tar.gz
opencode-605e5335588adeb96bc9549553e9976b84338a8d.zip
fix(app): file tree not always loading
-rw-r--r--packages/app/src/components/file-tree.tsx24
-rw-r--r--packages/app/src/pages/session.tsx28
2 files changed, 50 insertions, 2 deletions
diff --git a/packages/app/src/components/file-tree.tsx b/packages/app/src/components/file-tree.tsx
index d43310b19..bd989f755 100644
--- a/packages/app/src/components/file-tree.tsx
+++ b/packages/app/src/components/file-tree.tsx
@@ -8,6 +8,7 @@ import {
createMemo,
For,
Match,
+ onCleanup,
Show,
splitProps,
Switch,
@@ -123,7 +124,28 @@ export default function FileTree(props: {
createEffect(() => {
const path = props.path
- untrack(() => void file.tree.list(path))
+ const state = { cancelled: false, timer: undefined as number | undefined }
+
+ const load = (attempt: number) => {
+ if (state.cancelled) return
+ if (file.tree.state(path)?.loaded) return
+
+ void untrack(() => file.tree.list(path)).finally(() => {
+ if (state.cancelled) return
+ if (file.tree.state(path)?.loaded) return
+ if (attempt >= 2) return
+
+ const wait = Math.min(2000, 250 * 2 ** attempt)
+ state.timer = window.setTimeout(() => load(attempt + 1), wait)
+ })
+ }
+
+ load(0)
+
+ onCleanup(() => {
+ state.cancelled = true
+ if (state.timer !== undefined) clearTimeout(state.timer)
+ })
})
const nodes = createMemo(() => {
diff --git a/packages/app/src/pages/session.tsx b/packages/app/src/pages/session.tsx
index 7257cdc11..a4e6e24b1 100644
--- a/packages/app/src/pages/session.tsx
+++ b/packages/app/src/pages/session.tsx
@@ -1256,7 +1256,33 @@ export default function Page() {
if (!wants) return
if (sync.data.session_diff[id] !== undefined) return
- sync.session.diff(id)
+ const state = {
+ cancelled: false,
+ attempt: 0,
+ timer: undefined as number | undefined,
+ }
+
+ const load = () => {
+ if (state.cancelled) return
+ const pending = sync.session.diff(id)
+ if (!pending) return
+ pending.catch(() => {
+ if (state.cancelled) return
+ const attempt = state.attempt + 1
+ state.attempt = attempt
+ if (attempt > 5) return
+ if (state.timer !== undefined) clearTimeout(state.timer)
+ const wait = Math.min(10000, 250 * 2 ** (attempt - 1))
+ state.timer = window.setTimeout(load, wait)
+ })
+ }
+
+ load()
+
+ onCleanup(() => {
+ state.cancelled = true
+ if (state.timer !== undefined) clearTimeout(state.timer)
+ })
})
const autoScroll = createAutoScroll({