reigncode-app 1.3.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (300) hide show
  1. package/AGENTS.md +30 -0
  2. package/Dockerfile +21 -0
  3. package/README.md +51 -0
  4. package/bunfig.toml +3 -0
  5. package/create-effect-simplification-spec.md +515 -0
  6. package/e2e/AGENTS.md +226 -0
  7. package/e2e/actions.ts +1018 -0
  8. package/e2e/app/home.spec.ts +24 -0
  9. package/e2e/app/navigation.spec.ts +10 -0
  10. package/e2e/app/palette.spec.ts +20 -0
  11. package/e2e/app/server-default.spec.ts +58 -0
  12. package/e2e/app/session.spec.ts +16 -0
  13. package/e2e/app/titlebar-history.spec.ts +120 -0
  14. package/e2e/commands/input-focus.spec.ts +15 -0
  15. package/e2e/commands/panels.spec.ts +33 -0
  16. package/e2e/commands/tab-close.spec.ts +32 -0
  17. package/e2e/files/file-open.spec.ts +31 -0
  18. package/e2e/files/file-tree.spec.ts +56 -0
  19. package/e2e/files/file-viewer.spec.ts +156 -0
  20. package/e2e/fixtures.ts +154 -0
  21. package/e2e/models/model-picker.spec.ts +48 -0
  22. package/e2e/models/models-visibility.spec.ts +61 -0
  23. package/e2e/projects/project-edit.spec.ts +43 -0
  24. package/e2e/projects/projects-close.spec.ts +54 -0
  25. package/e2e/projects/projects-switch.spec.ts +116 -0
  26. package/e2e/projects/workspace-new-session.spec.ts +94 -0
  27. package/e2e/projects/workspaces.spec.ts +375 -0
  28. package/e2e/prompt/context.spec.ts +95 -0
  29. package/e2e/prompt/prompt-async.spec.ts +76 -0
  30. package/e2e/prompt/prompt-drop-file-uri.spec.ts +22 -0
  31. package/e2e/prompt/prompt-drop-file.spec.ts +30 -0
  32. package/e2e/prompt/prompt-history.spec.ts +184 -0
  33. package/e2e/prompt/prompt-mention.spec.ts +26 -0
  34. package/e2e/prompt/prompt-multiline.spec.ts +24 -0
  35. package/e2e/prompt/prompt-shell.spec.ts +62 -0
  36. package/e2e/prompt/prompt-slash-open.spec.ts +22 -0
  37. package/e2e/prompt/prompt-slash-share.spec.ts +64 -0
  38. package/e2e/prompt/prompt-slash-terminal.spec.ts +18 -0
  39. package/e2e/prompt/prompt.spec.ts +55 -0
  40. package/e2e/selectors.ts +75 -0
  41. package/e2e/session/session-child-navigation.spec.ts +37 -0
  42. package/e2e/session/session-composer-dock.spec.ts +530 -0
  43. package/e2e/session/session-model-persistence.spec.ts +359 -0
  44. package/e2e/session/session-review.spec.ts +426 -0
  45. package/e2e/session/session-undo-redo.spec.ts +233 -0
  46. package/e2e/session/session.spec.ts +174 -0
  47. package/e2e/settings/settings-keybinds.spec.ts +389 -0
  48. package/e2e/settings/settings-models.spec.ts +122 -0
  49. package/e2e/settings/settings-providers.spec.ts +136 -0
  50. package/e2e/settings/settings.spec.ts +519 -0
  51. package/e2e/sidebar/sidebar-popover-actions.spec.ts +118 -0
  52. package/e2e/sidebar/sidebar-session-links.spec.ts +30 -0
  53. package/e2e/sidebar/sidebar.spec.ts +40 -0
  54. package/e2e/status/status-popover.spec.ts +94 -0
  55. package/e2e/terminal/terminal-init.spec.ts +28 -0
  56. package/e2e/terminal/terminal-reconnect.spec.ts +46 -0
  57. package/e2e/terminal/terminal-tabs.spec.ts +168 -0
  58. package/e2e/terminal/terminal.spec.ts +18 -0
  59. package/e2e/thinking-level.spec.ts +25 -0
  60. package/e2e/tsconfig.json +9 -0
  61. package/e2e/utils.ts +63 -0
  62. package/happydom.ts +75 -0
  63. package/index.html +23 -0
  64. package/package.json +77 -0
  65. package/playwright.config.ts +45 -0
  66. package/public/_headers +17 -0
  67. package/public/oc-theme-preload.js +35 -0
  68. package/script/e2e-local.ts +180 -0
  69. package/src/addons/serialize.test.ts +319 -0
  70. package/src/addons/serialize.ts +634 -0
  71. package/src/app.tsx +308 -0
  72. package/src/components/debug-bar.tsx +443 -0
  73. package/src/components/dialog-connect-provider.tsx +617 -0
  74. package/src/components/dialog-custom-provider-form.ts +158 -0
  75. package/src/components/dialog-custom-provider.test.ts +80 -0
  76. package/src/components/dialog-custom-provider.tsx +329 -0
  77. package/src/components/dialog-edit-project.tsx +255 -0
  78. package/src/components/dialog-fork.tsx +108 -0
  79. package/src/components/dialog-manage-models.tsx +101 -0
  80. package/src/components/dialog-release-notes.tsx +144 -0
  81. package/src/components/dialog-select-directory.tsx +392 -0
  82. package/src/components/dialog-select-file.tsx +466 -0
  83. package/src/components/dialog-select-mcp.tsx +107 -0
  84. package/src/components/dialog-select-model-unpaid.tsx +137 -0
  85. package/src/components/dialog-select-model.tsx +220 -0
  86. package/src/components/dialog-select-provider.tsx +86 -0
  87. package/src/components/dialog-select-server.tsx +649 -0
  88. package/src/components/dialog-settings.tsx +73 -0
  89. package/src/components/file-tree.test.ts +78 -0
  90. package/src/components/file-tree.tsx +507 -0
  91. package/src/components/link.tsx +26 -0
  92. package/src/components/model-tooltip.tsx +91 -0
  93. package/src/components/prompt-input/attachments.test.ts +44 -0
  94. package/src/components/prompt-input/attachments.ts +201 -0
  95. package/src/components/prompt-input/build-request-parts.test.ts +312 -0
  96. package/src/components/prompt-input/build-request-parts.ts +175 -0
  97. package/src/components/prompt-input/context-items.tsx +88 -0
  98. package/src/components/prompt-input/drag-overlay.tsx +25 -0
  99. package/src/components/prompt-input/editor-dom.test.ts +99 -0
  100. package/src/components/prompt-input/editor-dom.ts +148 -0
  101. package/src/components/prompt-input/files.ts +66 -0
  102. package/src/components/prompt-input/history.test.ts +153 -0
  103. package/src/components/prompt-input/history.ts +256 -0
  104. package/src/components/prompt-input/image-attachments.tsx +58 -0
  105. package/src/components/prompt-input/paste.ts +24 -0
  106. package/src/components/prompt-input/placeholder.test.ts +48 -0
  107. package/src/components/prompt-input/placeholder.ts +15 -0
  108. package/src/components/prompt-input/slash-popover.tsx +141 -0
  109. package/src/components/prompt-input/submit.test.ts +346 -0
  110. package/src/components/prompt-input/submit.ts +579 -0
  111. package/src/components/prompt-input.tsx +1595 -0
  112. package/src/components/server/server-row.tsx +130 -0
  113. package/src/components/session/index.ts +5 -0
  114. package/src/components/session/session-context-breakdown.test.ts +61 -0
  115. package/src/components/session/session-context-breakdown.ts +132 -0
  116. package/src/components/session/session-context-format.ts +20 -0
  117. package/src/components/session/session-context-metrics.test.ts +101 -0
  118. package/src/components/session/session-context-metrics.ts +82 -0
  119. package/src/components/session/session-context-tab.tsx +339 -0
  120. package/src/components/session/session-header.tsx +486 -0
  121. package/src/components/session/session-new-view.tsx +91 -0
  122. package/src/components/session/session-sortable-tab.tsx +70 -0
  123. package/src/components/session/session-sortable-terminal-tab.tsx +193 -0
  124. package/src/components/session-context-usage.tsx +122 -0
  125. package/src/components/settings-general.tsx +585 -0
  126. package/src/components/settings-keybinds.tsx +453 -0
  127. package/src/components/settings-list.tsx +5 -0
  128. package/src/components/settings-models.tsx +137 -0
  129. package/src/components/settings-providers.tsx +251 -0
  130. package/src/components/status-popover.tsx +419 -0
  131. package/src/components/terminal.tsx +653 -0
  132. package/src/components/titlebar-history.test.ts +63 -0
  133. package/src/components/titlebar-history.ts +57 -0
  134. package/src/components/titlebar.tsx +312 -0
  135. package/src/constants/file-picker.ts +89 -0
  136. package/src/context/command-keybind.test.ts +69 -0
  137. package/src/context/command.test.ts +25 -0
  138. package/src/context/command.tsx +437 -0
  139. package/src/context/comments.test.ts +186 -0
  140. package/src/context/comments.tsx +243 -0
  141. package/src/context/file/content-cache.ts +88 -0
  142. package/src/context/file/path.test.ts +360 -0
  143. package/src/context/file/path.ts +151 -0
  144. package/src/context/file/tree-store.ts +170 -0
  145. package/src/context/file/types.ts +41 -0
  146. package/src/context/file/view-cache.ts +146 -0
  147. package/src/context/file/watcher.test.ts +149 -0
  148. package/src/context/file/watcher.ts +53 -0
  149. package/src/context/file-content-eviction-accounting.test.ts +65 -0
  150. package/src/context/file.tsx +280 -0
  151. package/src/context/global-sdk.tsx +232 -0
  152. package/src/context/global-sync/bootstrap.ts +206 -0
  153. package/src/context/global-sync/child-store.test.ts +38 -0
  154. package/src/context/global-sync/child-store.ts +281 -0
  155. package/src/context/global-sync/event-reducer.test.ts +552 -0
  156. package/src/context/global-sync/event-reducer.ts +359 -0
  157. package/src/context/global-sync/eviction.ts +28 -0
  158. package/src/context/global-sync/queue.ts +83 -0
  159. package/src/context/global-sync/session-cache.test.ts +102 -0
  160. package/src/context/global-sync/session-cache.ts +62 -0
  161. package/src/context/global-sync/session-load.ts +25 -0
  162. package/src/context/global-sync/session-prefetch.test.ts +96 -0
  163. package/src/context/global-sync/session-prefetch.ts +100 -0
  164. package/src/context/global-sync/session-trim.test.ts +59 -0
  165. package/src/context/global-sync/session-trim.ts +56 -0
  166. package/src/context/global-sync/types.ts +133 -0
  167. package/src/context/global-sync/utils.ts +25 -0
  168. package/src/context/global-sync.test.ts +122 -0
  169. package/src/context/global-sync.tsx +408 -0
  170. package/src/context/highlights.tsx +233 -0
  171. package/src/context/language.tsx +248 -0
  172. package/src/context/layout-scroll.test.ts +64 -0
  173. package/src/context/layout-scroll.ts +126 -0
  174. package/src/context/layout.test.ts +69 -0
  175. package/src/context/layout.tsx +937 -0
  176. package/src/context/local.tsx +422 -0
  177. package/src/context/model-variant.test.ts +86 -0
  178. package/src/context/model-variant.ts +52 -0
  179. package/src/context/models.tsx +163 -0
  180. package/src/context/notification.tsx +373 -0
  181. package/src/context/permission-auto-respond.test.ts +102 -0
  182. package/src/context/permission-auto-respond.ts +51 -0
  183. package/src/context/permission.tsx +277 -0
  184. package/src/context/platform.tsx +99 -0
  185. package/src/context/prompt.tsx +297 -0
  186. package/src/context/sdk.tsx +49 -0
  187. package/src/context/server.tsx +295 -0
  188. package/src/context/settings.tsx +241 -0
  189. package/src/context/sync-optimistic.test.ts +123 -0
  190. package/src/context/sync.tsx +618 -0
  191. package/src/context/terminal-title.ts +51 -0
  192. package/src/context/terminal.test.ts +82 -0
  193. package/src/context/terminal.tsx +437 -0
  194. package/src/entry.tsx +144 -0
  195. package/src/env.d.ts +18 -0
  196. package/src/hooks/use-providers.ts +44 -0
  197. package/src/i18n/ar.ts +855 -0
  198. package/src/i18n/br.ts +867 -0
  199. package/src/i18n/bs.ts +943 -0
  200. package/src/i18n/da.ts +937 -0
  201. package/src/i18n/de.ts +879 -0
  202. package/src/i18n/en.ts +948 -0
  203. package/src/i18n/es.ts +950 -0
  204. package/src/i18n/fr.ts +878 -0
  205. package/src/i18n/ja.ts +861 -0
  206. package/src/i18n/ko.ts +860 -0
  207. package/src/i18n/no.ts +944 -0
  208. package/src/i18n/parity.test.ts +32 -0
  209. package/src/i18n/pl.ts +865 -0
  210. package/src/i18n/ru.ts +946 -0
  211. package/src/i18n/th.ts +933 -0
  212. package/src/i18n/tr.ts +952 -0
  213. package/src/i18n/zh.ts +930 -0
  214. package/src/i18n/zht.ts +925 -0
  215. package/src/index.css +29 -0
  216. package/src/index.ts +6 -0
  217. package/src/pages/directory-layout.tsx +88 -0
  218. package/src/pages/error.tsx +327 -0
  219. package/src/pages/home.tsx +131 -0
  220. package/src/pages/layout/deep-links.ts +50 -0
  221. package/src/pages/layout/helpers.test.ts +211 -0
  222. package/src/pages/layout/helpers.ts +98 -0
  223. package/src/pages/layout/inline-editor.tsx +126 -0
  224. package/src/pages/layout/sidebar-items.tsx +437 -0
  225. package/src/pages/layout/sidebar-project.tsx +384 -0
  226. package/src/pages/layout/sidebar-shell.tsx +125 -0
  227. package/src/pages/layout/sidebar-workspace.tsx +504 -0
  228. package/src/pages/layout.tsx +2509 -0
  229. package/src/pages/session/composer/index.ts +2 -0
  230. package/src/pages/session/composer/session-composer-region.tsx +255 -0
  231. package/src/pages/session/composer/session-composer-state.test.ts +128 -0
  232. package/src/pages/session/composer/session-composer-state.ts +249 -0
  233. package/src/pages/session/composer/session-followup-dock.tsx +109 -0
  234. package/src/pages/session/composer/session-permission-dock.tsx +74 -0
  235. package/src/pages/session/composer/session-question-dock.tsx +449 -0
  236. package/src/pages/session/composer/session-request-tree.ts +52 -0
  237. package/src/pages/session/composer/session-revert-dock.tsx +99 -0
  238. package/src/pages/session/composer/session-todo-dock.tsx +330 -0
  239. package/src/pages/session/file-tab-scroll.test.ts +40 -0
  240. package/src/pages/session/file-tab-scroll.ts +67 -0
  241. package/src/pages/session/file-tabs.tsx +456 -0
  242. package/src/pages/session/handoff.ts +36 -0
  243. package/src/pages/session/helpers.test.ts +181 -0
  244. package/src/pages/session/helpers.ts +198 -0
  245. package/src/pages/session/message-gesture.test.ts +62 -0
  246. package/src/pages/session/message-gesture.ts +21 -0
  247. package/src/pages/session/message-id-from-hash.ts +6 -0
  248. package/src/pages/session/message-timeline.tsx +1013 -0
  249. package/src/pages/session/review-tab.tsx +170 -0
  250. package/src/pages/session/session-layout.ts +20 -0
  251. package/src/pages/session/session-model-helpers.test.ts +51 -0
  252. package/src/pages/session/session-model-helpers.ts +16 -0
  253. package/src/pages/session/session-side-panel.tsx +453 -0
  254. package/src/pages/session/terminal-label.ts +16 -0
  255. package/src/pages/session/terminal-panel.test.ts +25 -0
  256. package/src/pages/session/terminal-panel.tsx +326 -0
  257. package/src/pages/session/use-session-commands.tsx +495 -0
  258. package/src/pages/session/use-session-hash-scroll.test.ts +16 -0
  259. package/src/pages/session/use-session-hash-scroll.ts +197 -0
  260. package/src/pages/session.tsx +1841 -0
  261. package/src/sst-env.d.ts +12 -0
  262. package/src/testing/model-selection.ts +80 -0
  263. package/src/testing/prompt.ts +56 -0
  264. package/src/testing/session-composer.ts +84 -0
  265. package/src/testing/terminal.ts +118 -0
  266. package/src/theme-preload.test.ts +46 -0
  267. package/src/utils/agent.ts +23 -0
  268. package/src/utils/aim.ts +138 -0
  269. package/src/utils/base64.ts +10 -0
  270. package/src/utils/comment-note.ts +88 -0
  271. package/src/utils/id.ts +99 -0
  272. package/src/utils/notification-click.test.ts +27 -0
  273. package/src/utils/notification-click.ts +13 -0
  274. package/src/utils/persist.test.ts +115 -0
  275. package/src/utils/persist.ts +476 -0
  276. package/src/utils/prompt.test.ts +44 -0
  277. package/src/utils/prompt.ts +203 -0
  278. package/src/utils/runtime-adapters.test.ts +62 -0
  279. package/src/utils/runtime-adapters.ts +39 -0
  280. package/src/utils/same.ts +6 -0
  281. package/src/utils/scoped-cache.test.ts +69 -0
  282. package/src/utils/scoped-cache.ts +104 -0
  283. package/src/utils/server-errors.test.ts +131 -0
  284. package/src/utils/server-errors.ts +80 -0
  285. package/src/utils/server-health.test.ts +123 -0
  286. package/src/utils/server-health.ts +91 -0
  287. package/src/utils/server.ts +22 -0
  288. package/src/utils/solid-dnd.tsx +49 -0
  289. package/src/utils/sound.ts +117 -0
  290. package/src/utils/terminal-writer.test.ts +64 -0
  291. package/src/utils/terminal-writer.ts +65 -0
  292. package/src/utils/time.ts +22 -0
  293. package/src/utils/uuid.test.ts +78 -0
  294. package/src/utils/uuid.ts +12 -0
  295. package/src/utils/worktree.test.ts +46 -0
  296. package/src/utils/worktree.ts +73 -0
  297. package/sst-env.d.ts +10 -0
  298. package/tsconfig.json +26 -0
  299. package/vite.config.ts +15 -0
  300. package/vite.js +26 -0
package/AGENTS.md ADDED
@@ -0,0 +1,30 @@
1
+ ## Debugging
2
+
3
+ - NEVER try to restart the app, or the server process, EVER.
4
+
5
+ ## Local Dev
6
+
7
+ - `opencode dev web` proxies `https://app.code.reign-labs.com`, so local UI/CSS changes will not show there.
8
+ - For local UI changes, run the backend and app dev servers separately.
9
+ - Backend (from `packages/reigncode`): `bun run --conditions=browser ./src/index.ts serve --port 4096`
10
+ - App (from `packages/app`): `bun dev -- --port 4444`
11
+ - Open `http://localhost:4444` to verify UI changes (it targets the backend at `http://localhost:4096`).
12
+
13
+ ## SolidJS
14
+
15
+ - Always prefer `createStore` over multiple `createSignal` calls
16
+
17
+ ## Tool Calling
18
+
19
+ - ALWAYS USE PARALLEL TOOLS WHEN APPLICABLE.
20
+
21
+ ## Browser Automation
22
+
23
+ Use `agent-browser` for web automation. Run `agent-browser --help` for all commands.
24
+
25
+ Core workflow:
26
+
27
+ 1. `agent-browser open <url>` - Navigate to page
28
+ 2. `agent-browser snapshot -i` - Get interactive elements with refs (@e1, @e2)
29
+ 3. `agent-browser click @e1` / `fill @e2 "text"` - Interact using refs
30
+ 4. Re-snapshot after page changes
package/Dockerfile ADDED
@@ -0,0 +1,21 @@
1
+ FROM oven/bun:1.3.11-alpine AS builder
2
+
3
+ WORKDIR /app
4
+
5
+ RUN apk add --no-cache git
6
+
7
+ COPY . .
8
+ RUN bun install
9
+
10
+ WORKDIR /app/packages/app
11
+ RUN bun run build
12
+
13
+ FROM oven/bun:1.3.11-alpine
14
+
15
+ WORKDIR /app
16
+ COPY --from=builder /app/packages/app/dist ./dist
17
+
18
+ RUN bun add serve
19
+
20
+ EXPOSE 3000
21
+ CMD ["bunx", "serve", "dist", "-l", "3000", "--single"]
package/README.md ADDED
@@ -0,0 +1,51 @@
1
+ ## Usage
2
+
3
+ Dependencies for these templates are managed with [pnpm](https://pnpm.io) using `pnpm up -Lri`.
4
+
5
+ This is the reason you see a `pnpm-lock.yaml`. That said, any package manager will work. This file can safely be removed once you clone a template.
6
+
7
+ ```bash
8
+ $ npm install # or pnpm install or yarn install
9
+ ```
10
+
11
+ ### Learn more on the [Solid Website](https://solidjs.com) and come chat with us on our [Discord](https://discord.com/invite/solidjs)
12
+
13
+ ## Available Scripts
14
+
15
+ In the project directory, you can run:
16
+
17
+ ### `npm run dev` or `npm start`
18
+
19
+ Runs the app in the development mode.<br>
20
+ Open [http://localhost:3000](http://localhost:3000) to view it in the browser.
21
+
22
+ The page will reload if you make edits.<br>
23
+
24
+ ### `npm run build`
25
+
26
+ Builds the app for production to the `dist` folder.<br>
27
+ It correctly bundles Solid in production mode and optimizes the build for the best performance.
28
+
29
+ The build is minified and the filenames include the hashes.<br>
30
+ Your app is ready to be deployed!
31
+
32
+ ## E2E Testing
33
+
34
+ Playwright starts the Vite dev server automatically via `webServer`, and UI tests need a reigncode backend (defaults to `localhost:4096`).
35
+ Use the local runner to create a temp sandbox, seed data, and run the tests.
36
+
37
+ ```bash
38
+ bunx playwright install
39
+ bun run test:e2e:local
40
+ bun run test:e2e:local -- --grep "settings"
41
+ ```
42
+
43
+ Environment options:
44
+
45
+ - `PLAYWRIGHT_SERVER_HOST` / `PLAYWRIGHT_SERVER_PORT` (backend address, default: `localhost:4096`)
46
+ - `PLAYWRIGHT_PORT` (Vite dev server port, default: `3000`)
47
+ - `PLAYWRIGHT_BASE_URL` (override base URL, default: `http://localhost:<PLAYWRIGHT_PORT>`)
48
+
49
+ ## Deployment
50
+
51
+ You can deploy the `dist` folder to any static host provider (netlify, surge, now, etc.)
package/bunfig.toml ADDED
@@ -0,0 +1,3 @@
1
+ [test]
2
+ root = "./src"
3
+ preload = ["./happydom.ts"]
@@ -0,0 +1,515 @@
1
+ # CreateEffect Simplification Implementation Spec
2
+
3
+ Reduce reactive misuse across `packages/app`.
4
+
5
+ ---
6
+
7
+ ## Context
8
+
9
+ This work targets `packages/app/src`, which currently has 101 `createEffect` calls across 37 files.
10
+
11
+ The biggest clusters are `pages/session.tsx` (19), `pages/layout.tsx` (13), `pages/session/file-tabs.tsx` (6), and several context providers that mirror one store into another.
12
+
13
+ Key issues from the audit:
14
+
15
+ - Derived state is being written through effects instead of computed directly
16
+ - Session and file resets are handled by watch-and-clear effects instead of keyed state boundaries
17
+ - User-driven actions are hidden inside reactive effects
18
+ - Context layers mirror and hydrate child stores with multiple sync effects
19
+ - Several areas repeat the same imperative trigger pattern in multiple effects
20
+
21
+ Keep the implementation focused on removing unnecessary effects, not on broad UI redesign.
22
+
23
+ ## Goals
24
+
25
+ - Cut high-churn `createEffect` usage in the hottest files first
26
+ - Replace effect-driven derived state with reactive derivation
27
+ - Replace reset-on-key effects with keyed ownership boundaries
28
+ - Move event-driven work to direct actions and write paths
29
+ - Remove mirrored store hydration where a single source of truth can exist
30
+ - Leave necessary external sync effects in place, but make them narrower and clearer
31
+
32
+ ## Non-Goals
33
+
34
+ - Do not rewrite unrelated component structure just to reduce the count
35
+ - Do not change product behavior, navigation flow, or persisted data shape unless required for a cleaner write boundary
36
+ - Do not remove effects that bridge to DOM, editors, polling, or external APIs unless there is a clearly safer equivalent
37
+ - Do not attempt a repo-wide cleanup outside `packages/app`
38
+
39
+ ## Effect Taxonomy And Replacement Rules
40
+
41
+ Use these rules during implementation.
42
+
43
+ ### Prefer `createMemo`
44
+
45
+ Use `createMemo` when the target value is pure derived state from other signals or stores.
46
+
47
+ Do this when an effect only reads reactive inputs and writes another reactive value that could be computed instead.
48
+
49
+ Apply this to:
50
+
51
+ - `packages/app/src/pages/session.tsx:141`
52
+ - `packages/app/src/pages/layout.tsx:557`
53
+ - `packages/app/src/components/terminal.tsx:261`
54
+ - `packages/app/src/components/session/session-header.tsx:309`
55
+
56
+ Rules:
57
+
58
+ - If no external system is touched, do not use `createEffect`
59
+ - Derive once, then read the memo where needed
60
+ - If normalization is required, prefer normalizing at the write boundary before falling back to a memo
61
+
62
+ ### Prefer Keyed Remounts
63
+
64
+ Use keyed remounts when local UI state should reset because an identity changed.
65
+
66
+ Do this with `sessionKey`, `scope()`, or another stable identity instead of watching the key and manually clearing signals.
67
+
68
+ Apply this to:
69
+
70
+ - `packages/app/src/pages/session.tsx:325`
71
+ - `packages/app/src/pages/session.tsx:336`
72
+ - `packages/app/src/pages/session.tsx:477`
73
+ - `packages/app/src/pages/session.tsx:869`
74
+ - `packages/app/src/pages/session.tsx:963`
75
+ - `packages/app/src/pages/session/message-timeline.tsx:149`
76
+ - `packages/app/src/context/file.tsx:100`
77
+
78
+ Rules:
79
+
80
+ - If the desired behavior is "new identity, fresh local state," key the owner subtree
81
+ - Keep state local to the keyed boundary so teardown and recreation handle the reset naturally
82
+
83
+ ### Prefer Event Handlers And Actions
84
+
85
+ Use direct handlers, store actions, and async command functions when work happens because a user clicked, selected, reloaded, or navigated.
86
+
87
+ Do this when an effect is just watching for a flag change, command token, or event-bus signal to trigger imperative logic.
88
+
89
+ Apply this to:
90
+
91
+ - `packages/app/src/pages/layout.tsx:484`
92
+ - `packages/app/src/pages/layout.tsx:652`
93
+ - `packages/app/src/pages/layout.tsx:776`
94
+ - `packages/app/src/pages/layout.tsx:1489`
95
+ - `packages/app/src/pages/layout.tsx:1519`
96
+ - `packages/app/src/components/file-tree.tsx:328`
97
+ - `packages/app/src/pages/session/terminal-panel.tsx:55`
98
+ - `packages/app/src/context/global-sync.tsx:148`
99
+ - Duplicated trigger sets in:
100
+ - `packages/app/src/pages/session/review-tab.tsx:122`
101
+ - `packages/app/src/pages/session/review-tab.tsx:130`
102
+ - `packages/app/src/pages/session/review-tab.tsx:138`
103
+ - `packages/app/src/pages/session/file-tabs.tsx:367`
104
+ - `packages/app/src/pages/session/file-tabs.tsx:378`
105
+ - `packages/app/src/pages/session/file-tabs.tsx:389`
106
+ - `packages/app/src/pages/session/use-session-hash-scroll.ts:144`
107
+ - `packages/app/src/pages/session/use-session-hash-scroll.ts:149`
108
+ - `packages/app/src/pages/session/use-session-hash-scroll.ts:167`
109
+
110
+ Rules:
111
+
112
+ - If the trigger is user intent, call the action at the source of that intent
113
+ - If the same imperative work is triggered from multiple places, extract one function and call it directly
114
+
115
+ ### Prefer `onMount` And `onCleanup`
116
+
117
+ Use `onMount` and `onCleanup` for lifecycle-only setup and teardown.
118
+
119
+ This is the right fit for subscriptions, one-time wiring, timers, and imperative integration that should not rerun for ordinary reactive changes.
120
+
121
+ Use this when:
122
+
123
+ - Setup should happen once per owner lifecycle
124
+ - Cleanup should always pair with teardown
125
+ - The work is not conceptually derived state
126
+
127
+ ### Keep `createEffect` When It Is A Real Bridge
128
+
129
+ Keep `createEffect` when it synchronizes reactive data to an external imperative sink.
130
+
131
+ Examples that should remain, though they may be narrowed or split:
132
+
133
+ - DOM/editor sync in `packages/app/src/components/prompt-input.tsx:690`
134
+ - Scroll sync in `packages/app/src/pages/session.tsx:685`
135
+ - Scroll/hash sync in `packages/app/src/pages/session/use-session-hash-scroll.ts:149`
136
+ - External sync in:
137
+ - `packages/app/src/context/language.tsx:207`
138
+ - `packages/app/src/context/settings.tsx:110`
139
+ - `packages/app/src/context/sdk.tsx:26`
140
+ - Polling in:
141
+ - `packages/app/src/components/status-popover.tsx:59`
142
+ - `packages/app/src/components/dialog-select-server.tsx:273`
143
+
144
+ Rules:
145
+
146
+ - Keep the effect single-purpose
147
+ - Make dependencies explicit and narrow
148
+ - Avoid writing back into the same reactive graph unless absolutely required
149
+
150
+ ## Implementation Plan
151
+
152
+ ### Phase 0: Classification Pass
153
+
154
+ Before changing code, tag each targeted effect as one of: derive, reset, event, lifecycle, or external bridge.
155
+
156
+ Acceptance criteria:
157
+
158
+ - Every targeted effect in this spec is tagged with a replacement strategy before refactoring starts
159
+ - Shared helpers to be introduced are identified up front to avoid repeating patterns
160
+
161
+ ### Phase 1: Derived-State Cleanup
162
+
163
+ Tackle highest-value, lowest-risk derived-state cleanup first.
164
+
165
+ Priority items:
166
+
167
+ - Normalize tabs at write boundaries and remove `packages/app/src/pages/session.tsx:141`
168
+ - Stop syncing `workspaceOrder` in `packages/app/src/pages/layout.tsx:557`
169
+ - Make prompt slash filtering reactive so `packages/app/src/components/prompt-input.tsx:652` can be removed
170
+ - Replace other obvious derived-state effects in terminal and session header
171
+
172
+ Acceptance criteria:
173
+
174
+ - No behavior change in tab ordering, prompt filtering, terminal display, or header state
175
+ - Targeted derived-state effects are deleted, not just moved
176
+
177
+ ### Phase 2: Keyed Reset Cleanup
178
+
179
+ Replace reset-on-key effects with keyed ownership boundaries.
180
+
181
+ Priority items:
182
+
183
+ - Key session-scoped UI and state by `sessionKey`
184
+ - Key file-scoped state by `scope()`
185
+ - Remove manual clear-and-reseed effects in session and file context
186
+
187
+ Acceptance criteria:
188
+
189
+ - Switching session or file scope recreates the intended local state cleanly
190
+ - No stale state leaks across session or scope changes
191
+ - Target reset effects are deleted
192
+
193
+ ### Phase 3: Event-Driven Work Extraction
194
+
195
+ Move event-driven work out of reactive effects.
196
+
197
+ Priority items:
198
+
199
+ - Replace `globalStore.reload` effect dispatching with direct calls
200
+ - Split mixed-responsibility effect in `packages/app/src/pages/layout.tsx:1489`
201
+ - Collapse duplicated imperative trigger triplets into single functions
202
+ - Move file-tree and terminal-panel imperative work to explicit handlers
203
+
204
+ Acceptance criteria:
205
+
206
+ - User-triggered behavior still fires exactly once per intended action
207
+ - No effect remains whose only job is to notice a command-like state and trigger an imperative function
208
+
209
+ ### Phase 4: Context Ownership Cleanup
210
+
211
+ Remove mirrored child-store hydration patterns.
212
+
213
+ Priority items:
214
+
215
+ - Remove child-store hydration mirrors in `packages/app/src/context/global-sync/child-store.ts:184`, `:190`, `:193`
216
+ - Simplify mirror logic in `packages/app/src/context/global-sync.tsx:130`, `:138`
217
+ - Revisit `packages/app/src/context/layout.tsx:424` if it still mirrors instead of deriving
218
+
219
+ Acceptance criteria:
220
+
221
+ - There is one clear source of truth for each synced value
222
+ - Child stores no longer need effect-based hydration to stay consistent
223
+ - Initialization and updates both work without manual mirror effects
224
+
225
+ ### Phase 5: Cleanup And Keeper Review
226
+
227
+ Clean up remaining targeted hotspots and narrow the effects that should stay.
228
+
229
+ Acceptance criteria:
230
+
231
+ - Remaining `createEffect` calls in touched files are all true bridges or clearly justified lifecycle sync
232
+ - Mixed-responsibility effects are split into smaller units where still needed
233
+
234
+ ## Detailed Work Items By Area
235
+
236
+ ### 1. Normalize Tab State
237
+
238
+ Files:
239
+
240
+ - `packages/app/src/pages/session.tsx:141`
241
+
242
+ Work:
243
+
244
+ - Move tab normalization into the functions that create, load, or update tab state
245
+ - Make readers consume already-normalized tab data
246
+ - Remove the effect that rewrites derived tab state after the fact
247
+
248
+ Rationale:
249
+
250
+ - Tabs should become valid when written, not be repaired later
251
+ - This removes a feedback loop and makes state easier to trust
252
+
253
+ Acceptance criteria:
254
+
255
+ - The effect at `packages/app/src/pages/session.tsx:141` is removed
256
+ - Newly created and restored tabs are normalized before they enter local state
257
+ - Tab rendering still matches current behavior for valid and edge-case inputs
258
+
259
+ ### 2. Key Session-Owned State
260
+
261
+ Files:
262
+
263
+ - `packages/app/src/pages/session.tsx:325`
264
+ - `packages/app/src/pages/session.tsx:336`
265
+ - `packages/app/src/pages/session.tsx:477`
266
+ - `packages/app/src/pages/session.tsx:869`
267
+ - `packages/app/src/pages/session.tsx:963`
268
+ - `packages/app/src/pages/session/message-timeline.tsx:149`
269
+
270
+ Work:
271
+
272
+ - Identify state that should reset when `sessionKey` changes
273
+ - Move that state under a keyed subtree or keyed owner boundary
274
+ - Remove effects that watch `sessionKey` just to clear local state, refs, or temporary UI flags
275
+
276
+ Rationale:
277
+
278
+ - Session identity already defines the lifetime of this UI state
279
+ - Keyed ownership makes reset behavior automatic and easier to reason about
280
+
281
+ Acceptance criteria:
282
+
283
+ - The targeted reset effects are removed
284
+ - Changing sessions resets only the intended session-local state
285
+ - Scroll and editor state that should persist are not accidentally reset
286
+
287
+ ### 3. Derive Workspace Order
288
+
289
+ Files:
290
+
291
+ - `packages/app/src/pages/layout.tsx:557`
292
+
293
+ Work:
294
+
295
+ - Stop writing `workspaceOrder` from live workspace data in an effect
296
+ - Represent user overrides separately from live workspace data
297
+ - Compute effective order from current data plus overrides with a memo or pure helper
298
+
299
+ Rationale:
300
+
301
+ - Persisted user intent and live source data should not mirror each other through an effect
302
+ - A computed effective order avoids drift and racey resync behavior
303
+
304
+ Acceptance criteria:
305
+
306
+ - The effect at `packages/app/src/pages/layout.tsx:557` is removed
307
+ - Workspace order updates correctly when workspaces appear, disappear, or are reordered by the user
308
+ - User overrides persist without requiring a sync-back effect
309
+
310
+ ### 4. Remove Child-Store Mirrors
311
+
312
+ Files:
313
+
314
+ - `packages/app/src/context/global-sync.tsx:130`
315
+ - `packages/app/src/context/global-sync.tsx:138`
316
+ - `packages/app/src/context/global-sync.tsx:148`
317
+ - `packages/app/src/context/global-sync/child-store.ts:184`
318
+ - `packages/app/src/context/global-sync/child-store.ts:190`
319
+ - `packages/app/src/context/global-sync/child-store.ts:193`
320
+ - `packages/app/src/context/layout.tsx:424`
321
+
322
+ Work:
323
+
324
+ - Trace the actual ownership of global and child store values
325
+ - Replace hydration and mirror effects with explicit initialization and direct updates
326
+ - Remove the `globalStore.reload` event-bus pattern and call the needed reload paths directly
327
+
328
+ Rationale:
329
+
330
+ - Mirrors make it hard to tell which state is authoritative
331
+ - Event-bus style state toggles hide control flow and create accidental reruns
332
+
333
+ Acceptance criteria:
334
+
335
+ - Child store hydration no longer depends on effect-based copying
336
+ - Reload work can be followed from the event source to the handler without a reactive relay
337
+ - State remains correct on first load, child creation, and subsequent updates
338
+
339
+ ### 5. Key File-Scoped State
340
+
341
+ Files:
342
+
343
+ - `packages/app/src/context/file.tsx:100`
344
+
345
+ Work:
346
+
347
+ - Move file-scoped local state under a boundary keyed by `scope()`
348
+ - Remove any effect that watches `scope()` only to reset file-local state
349
+
350
+ Rationale:
351
+
352
+ - File scope changes are identity changes
353
+ - Keyed ownership gives a cleaner reset than manual clear logic
354
+
355
+ Acceptance criteria:
356
+
357
+ - The effect at `packages/app/src/context/file.tsx:100` is removed
358
+ - Switching scopes resets only scope-local state
359
+ - No previous-scope data appears after a scope change
360
+
361
+ ### 6. Split Layout Side Effects
362
+
363
+ Files:
364
+
365
+ - `packages/app/src/pages/layout.tsx:1489`
366
+ - Related event-driven effects near `packages/app/src/pages/layout.tsx:484`, `:652`, `:776`, `:1519`
367
+
368
+ Work:
369
+
370
+ - Break the mixed-responsibility effect at `:1489` into direct actions and smaller bridge effects only where required
371
+ - Move user-triggered branches into the actual command or handler that causes them
372
+ - Remove any branch that only exists because one effect is handling unrelated concerns
373
+
374
+ Rationale:
375
+
376
+ - Mixed effects hide cause and make reruns hard to predict
377
+ - Smaller units reduce accidental coupling and make future cleanup safer
378
+
379
+ Acceptance criteria:
380
+
381
+ - The effect at `packages/app/src/pages/layout.tsx:1489` no longer mixes unrelated responsibilities
382
+ - Event-driven branches execute from direct handlers
383
+ - Remaining effects in this area each have one clear external sync purpose
384
+
385
+ ### 7. Remove Duplicate Triggers
386
+
387
+ Files:
388
+
389
+ - `packages/app/src/pages/session/review-tab.tsx:122`
390
+ - `packages/app/src/pages/session/review-tab.tsx:130`
391
+ - `packages/app/src/pages/session/review-tab.tsx:138`
392
+ - `packages/app/src/pages/session/file-tabs.tsx:367`
393
+ - `packages/app/src/pages/session/file-tabs.tsx:378`
394
+ - `packages/app/src/pages/session/file-tabs.tsx:389`
395
+ - `packages/app/src/pages/session/use-session-hash-scroll.ts:144`
396
+ - `packages/app/src/pages/session/use-session-hash-scroll.ts:149`
397
+ - `packages/app/src/pages/session/use-session-hash-scroll.ts:167`
398
+
399
+ Work:
400
+
401
+ - Extract one explicit imperative function per behavior
402
+ - Call that function from each source event instead of replicating the same effect pattern multiple times
403
+ - Preserve the scroll-sync effect that is truly syncing with the DOM, but remove duplicate trigger scaffolding around it
404
+
405
+ Rationale:
406
+
407
+ - Duplicate triggers make it easy to miss a case or fire twice
408
+ - One named action is easier to test and reason about
409
+
410
+ Acceptance criteria:
411
+
412
+ - Repeated imperative effect triplets are collapsed into shared functions
413
+ - Scroll behavior still works, including hash-based navigation
414
+ - No duplicate firing is introduced
415
+
416
+ ### 8. Make Prompt Filtering Reactive
417
+
418
+ Files:
419
+
420
+ - `packages/app/src/components/prompt-input.tsx:652`
421
+ - Keep `packages/app/src/components/prompt-input.tsx:690` as needed
422
+
423
+ Work:
424
+
425
+ - Convert slash filtering into a pure reactive derivation from the current input and candidate command list
426
+ - Keep only the editor or DOM bridge effect if it is still needed for imperative syncing
427
+
428
+ Rationale:
429
+
430
+ - Filtering is classic derived state
431
+ - It should not need an effect if it can be computed from current inputs
432
+
433
+ Acceptance criteria:
434
+
435
+ - The effect at `packages/app/src/components/prompt-input.tsx:652` is removed
436
+ - Filtered slash-command results update correctly as the input changes
437
+ - The editor sync effect at `:690` still behaves correctly
438
+
439
+ ### 9. Clean Up Smaller Derived-State Cases
440
+
441
+ Files:
442
+
443
+ - `packages/app/src/components/terminal.tsx:261`
444
+ - `packages/app/src/components/session/session-header.tsx:309`
445
+
446
+ Work:
447
+
448
+ - Replace effect-written local state with memos or inline derivation
449
+ - Remove intermediate setters when the value can be computed directly
450
+
451
+ Rationale:
452
+
453
+ - These are low-risk wins that reinforce the same pattern
454
+ - They also help keep follow-up cleanup consistent
455
+
456
+ Acceptance criteria:
457
+
458
+ - Targeted effects are removed
459
+ - UI output remains unchanged under the same inputs
460
+
461
+ ## Verification And Regression Checks
462
+
463
+ Run focused checks after each phase, not only at the end.
464
+
465
+ ### Suggested Verification
466
+
467
+ - Switch between sessions rapidly and confirm local session UI resets only where intended
468
+ - Open, close, and reorder tabs and confirm order and normalization remain stable
469
+ - Change workspaces, reload workspace data, and verify effective ordering is correct
470
+ - Change file scope and confirm stale file state does not bleed across scopes
471
+ - Trigger layout actions that previously depended on effects and confirm they still fire once
472
+ - Use slash commands in the prompt and verify filtering updates as you type
473
+ - Test review tab, file tab, and hash-scroll flows for duplicate or missing triggers
474
+ - Verify global sync initialization, reload, and child-store creation paths
475
+
476
+ ### Regression Checks
477
+
478
+ - No accidental infinite reruns
479
+ - No double-firing network or command actions
480
+ - No lost cleanup for listeners, timers, or scroll handlers
481
+ - No preserved stale state after identity changes
482
+ - No removed effect that was actually bridging to DOM or an external API
483
+
484
+ If available, add or update tests around pure helpers introduced during this cleanup.
485
+
486
+ Favor tests for derived ordering, normalization, and action extraction, since those are easiest to lock down.
487
+
488
+ ## Definition Of Done
489
+
490
+ This work is done when all of the following are true:
491
+
492
+ - The highest-leverage targets in this spec are implemented
493
+ - Each removed effect has been replaced by a clearer pattern: memo, keyed boundary, direct action, or lifecycle hook
494
+ - The "should remain" effects still exist only where they serve a real external sync purpose
495
+ - Touched files have fewer mixed-responsibility effects and clearer ownership of state
496
+ - Manual verification covers session switching, file scope changes, workspace ordering, prompt filtering, and reload flows
497
+ - No behavior regressions are found in the targeted areas
498
+
499
+ A reduced raw `createEffect` count is helpful, but it is not the main success metric.
500
+
501
+ The main success metric is clearer ownership and fewer effect-driven state repairs.
502
+
503
+ ## Risks And Rollout Notes
504
+
505
+ Main risks:
506
+
507
+ - Keyed remounts can reset too much if state boundaries are drawn too high
508
+ - Store mirror removal can break initialization order if ownership is not mapped first
509
+ - Moving event work out of effects can accidentally skip triggers that were previously implicit
510
+
511
+ Rollout notes:
512
+
513
+ - Land in small phases, with each phase keeping the app behaviorally stable
514
+ - Prefer isolated PRs by phase or by file cluster, especially for context-store changes
515
+ - Review each remaining effect in touched files and leave it only if it clearly bridges to something external