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
@@ -0,0 +1,339 @@
1
+ import { createMemo, createEffect, on, onCleanup, For, Show } from "solid-js"
2
+ import type { JSX } from "solid-js"
3
+ import { useSync } from "@/context/sync"
4
+ import { checksum } from "@reign-labs/util/encode"
5
+ import { findLast } from "@reign-labs/util/array"
6
+ import { same } from "@/utils/same"
7
+ import { Icon } from "@reign-labs/ui/icon"
8
+ import { Accordion } from "@reign-labs/ui/accordion"
9
+ import { StickyAccordionHeader } from "@reign-labs/ui/sticky-accordion-header"
10
+ import { File } from "@reign-labs/ui/file"
11
+ import { Markdown } from "@reign-labs/ui/markdown"
12
+ import { ScrollView } from "@reign-labs/ui/scroll-view"
13
+ import type { Message, Part, UserMessage } from "@reign-labs/sdk/v2/client"
14
+ import { useLanguage } from "@/context/language"
15
+ import { useSessionLayout } from "@/pages/session/session-layout"
16
+ import { getSessionContextMetrics } from "./session-context-metrics"
17
+ import { estimateSessionContextBreakdown, type SessionContextBreakdownKey } from "./session-context-breakdown"
18
+ import { createSessionContextFormatter } from "./session-context-format"
19
+
20
+ const BREAKDOWN_COLOR: Record<SessionContextBreakdownKey, string> = {
21
+ system: "var(--syntax-info)",
22
+ user: "var(--syntax-success)",
23
+ assistant: "var(--syntax-property)",
24
+ tool: "var(--syntax-warning)",
25
+ other: "var(--syntax-comment)",
26
+ }
27
+
28
+ function Stat(props: { label: string; value: JSX.Element }) {
29
+ return (
30
+ <div class="flex flex-col gap-1">
31
+ <div class="text-12-regular text-text-weak">{props.label}</div>
32
+ <div class="text-12-medium text-text-strong">{props.value}</div>
33
+ </div>
34
+ )
35
+ }
36
+
37
+ function RawMessageContent(props: { message: Message; getParts: (id: string) => Part[]; onRendered: () => void }) {
38
+ const file = createMemo(() => {
39
+ const parts = props.getParts(props.message.id)
40
+ const contents = JSON.stringify({ message: props.message, parts }, null, 2)
41
+ return {
42
+ name: `${props.message.role}-${props.message.id}.json`,
43
+ contents,
44
+ cacheKey: checksum(contents),
45
+ }
46
+ })
47
+
48
+ return (
49
+ <File
50
+ mode="text"
51
+ file={file()}
52
+ overflow="wrap"
53
+ class="select-text"
54
+ onRendered={() => requestAnimationFrame(props.onRendered)}
55
+ />
56
+ )
57
+ }
58
+
59
+ function RawMessage(props: {
60
+ message: Message
61
+ getParts: (id: string) => Part[]
62
+ onRendered: () => void
63
+ time: (value: number | undefined) => string
64
+ }) {
65
+ return (
66
+ <Accordion.Item value={props.message.id}>
67
+ <StickyAccordionHeader>
68
+ <Accordion.Trigger>
69
+ <div class="flex items-center justify-between gap-2 w-full">
70
+ <div class="min-w-0 truncate">
71
+ {props.message.role} <span class="text-text-base">• {props.message.id}</span>
72
+ </div>
73
+ <div class="flex items-center gap-3">
74
+ <div class="shrink-0 text-12-regular text-text-weak">{props.time(props.message.time.created)}</div>
75
+ <Icon name="chevron-grabber-vertical" size="small" class="shrink-0 text-text-weak" />
76
+ </div>
77
+ </div>
78
+ </Accordion.Trigger>
79
+ </StickyAccordionHeader>
80
+ <Accordion.Content class="bg-background-base">
81
+ <div class="p-3">
82
+ <RawMessageContent message={props.message} getParts={props.getParts} onRendered={props.onRendered} />
83
+ </div>
84
+ </Accordion.Content>
85
+ </Accordion.Item>
86
+ )
87
+ }
88
+
89
+ const emptyMessages: Message[] = []
90
+ const emptyUserMessages: UserMessage[] = []
91
+
92
+ export function SessionContextTab() {
93
+ const sync = useSync()
94
+ const language = useLanguage()
95
+ const { params, view } = useSessionLayout()
96
+
97
+ const info = createMemo(() => (params.id ? sync.session.get(params.id) : undefined))
98
+
99
+ const messages = createMemo(
100
+ () => {
101
+ const id = params.id
102
+ if (!id) return emptyMessages
103
+ return (sync.data.message[id] ?? []) as Message[]
104
+ },
105
+ emptyMessages,
106
+ { equals: same },
107
+ )
108
+
109
+ const userMessages = createMemo(
110
+ () => messages().filter((m) => m.role === "user") as UserMessage[],
111
+ emptyUserMessages,
112
+ { equals: same },
113
+ )
114
+
115
+ const visibleUserMessages = createMemo(
116
+ () => {
117
+ const revert = info()?.revert?.messageID
118
+ if (!revert) return userMessages()
119
+ return userMessages().filter((m) => m.id < revert)
120
+ },
121
+ emptyUserMessages,
122
+ { equals: same },
123
+ )
124
+
125
+ const usd = createMemo(
126
+ () =>
127
+ new Intl.NumberFormat(language.intl(), {
128
+ style: "currency",
129
+ currency: "USD",
130
+ }),
131
+ )
132
+
133
+ const metrics = createMemo(() => getSessionContextMetrics(messages(), sync.data.provider.all))
134
+ const ctx = createMemo(() => metrics().context)
135
+ const formatter = createMemo(() => createSessionContextFormatter(language.intl()))
136
+
137
+ const cost = createMemo(() => {
138
+ return usd().format(metrics().totalCost)
139
+ })
140
+
141
+ const counts = createMemo(() => {
142
+ const all = messages()
143
+ const user = all.reduce((count, x) => count + (x.role === "user" ? 1 : 0), 0)
144
+ const assistant = all.reduce((count, x) => count + (x.role === "assistant" ? 1 : 0), 0)
145
+ return {
146
+ all: all.length,
147
+ user,
148
+ assistant,
149
+ }
150
+ })
151
+
152
+ const systemPrompt = createMemo(() => {
153
+ const msg = findLast(visibleUserMessages(), (m) => !!m.system)
154
+ const system = msg?.system
155
+ if (!system) return
156
+ const trimmed = system.trim()
157
+ if (!trimmed) return
158
+ return trimmed
159
+ })
160
+
161
+ const providerLabel = createMemo(() => {
162
+ const c = ctx()
163
+ if (!c) return "—"
164
+ return c.providerLabel
165
+ })
166
+
167
+ const modelLabel = createMemo(() => {
168
+ const c = ctx()
169
+ if (!c) return "—"
170
+ return c.modelLabel
171
+ })
172
+
173
+ const breakdown = createMemo(
174
+ on(
175
+ () => [ctx()?.message.id, ctx()?.input, messages().length, systemPrompt()],
176
+ () => {
177
+ const c = ctx()
178
+ if (!c?.input) return []
179
+ return estimateSessionContextBreakdown({
180
+ messages: messages(),
181
+ parts: sync.data.part as Record<string, Part[] | undefined>,
182
+ input: c.input,
183
+ systemPrompt: systemPrompt(),
184
+ })
185
+ },
186
+ ),
187
+ )
188
+
189
+ const breakdownLabel = (key: SessionContextBreakdownKey) => {
190
+ if (key === "system") return language.t("context.breakdown.system")
191
+ if (key === "user") return language.t("context.breakdown.user")
192
+ if (key === "assistant") return language.t("context.breakdown.assistant")
193
+ if (key === "tool") return language.t("context.breakdown.tool")
194
+ return language.t("context.breakdown.other")
195
+ }
196
+
197
+ const stats = [
198
+ { label: "context.stats.session", value: () => info()?.title ?? params.id ?? "—" },
199
+ { label: "context.stats.messages", value: () => counts().all.toLocaleString(language.intl()) },
200
+ { label: "context.stats.provider", value: providerLabel },
201
+ { label: "context.stats.model", value: modelLabel },
202
+ { label: "context.stats.limit", value: () => formatter().number(ctx()?.limit) },
203
+ { label: "context.stats.totalTokens", value: () => formatter().number(ctx()?.total) },
204
+ { label: "context.stats.usage", value: () => formatter().percent(ctx()?.usage) },
205
+ { label: "context.stats.inputTokens", value: () => formatter().number(ctx()?.input) },
206
+ { label: "context.stats.outputTokens", value: () => formatter().number(ctx()?.output) },
207
+ { label: "context.stats.reasoningTokens", value: () => formatter().number(ctx()?.reasoning) },
208
+ {
209
+ label: "context.stats.cacheTokens",
210
+ value: () => `${formatter().number(ctx()?.cacheRead)} / ${formatter().number(ctx()?.cacheWrite)}`,
211
+ },
212
+ { label: "context.stats.userMessages", value: () => counts().user.toLocaleString(language.intl()) },
213
+ { label: "context.stats.assistantMessages", value: () => counts().assistant.toLocaleString(language.intl()) },
214
+ { label: "context.stats.totalCost", value: cost },
215
+ { label: "context.stats.sessionCreated", value: () => formatter().time(info()?.time.created) },
216
+ { label: "context.stats.lastActivity", value: () => formatter().time(ctx()?.message.time.created) },
217
+ ] satisfies { label: string; value: () => JSX.Element }[]
218
+
219
+ let scroll: HTMLDivElement | undefined
220
+ let frame: number | undefined
221
+ let pending: { x: number; y: number } | undefined
222
+ const getParts = (id: string) => (sync.data.part[id] ?? []) as Part[]
223
+
224
+ const restoreScroll = () => {
225
+ const el = scroll
226
+ if (!el) return
227
+
228
+ const s = view().scroll("context")
229
+ if (!s) return
230
+
231
+ if (el.scrollTop !== s.y) el.scrollTop = s.y
232
+ if (el.scrollLeft !== s.x) el.scrollLeft = s.x
233
+ }
234
+
235
+ const handleScroll = (event: Event & { currentTarget: HTMLDivElement }) => {
236
+ pending = {
237
+ x: event.currentTarget.scrollLeft,
238
+ y: event.currentTarget.scrollTop,
239
+ }
240
+ if (frame !== undefined) return
241
+
242
+ frame = requestAnimationFrame(() => {
243
+ frame = undefined
244
+
245
+ const next = pending
246
+ pending = undefined
247
+ if (!next) return
248
+
249
+ view().setScroll("context", next)
250
+ })
251
+ }
252
+
253
+ createEffect(
254
+ on(
255
+ () => messages().length,
256
+ () => {
257
+ requestAnimationFrame(restoreScroll)
258
+ },
259
+ { defer: true },
260
+ ),
261
+ )
262
+
263
+ onCleanup(() => {
264
+ if (frame === undefined) return
265
+ cancelAnimationFrame(frame)
266
+ })
267
+
268
+ return (
269
+ <ScrollView
270
+ class="@container h-full"
271
+ viewportRef={(el) => {
272
+ scroll = el
273
+ restoreScroll()
274
+ }}
275
+ onScroll={handleScroll}
276
+ >
277
+ <div class="px-6 pt-4 pb-10 flex flex-col gap-10">
278
+ <div class="grid grid-cols-1 @[32rem]:grid-cols-2 gap-4">
279
+ <For each={stats}>
280
+ {(stat) => <Stat label={language.t(stat.label as Parameters<typeof language.t>[0])} value={stat.value()} />}
281
+ </For>
282
+ </div>
283
+
284
+ <Show when={breakdown().length > 0}>
285
+ <div class="flex flex-col gap-2">
286
+ <div class="text-12-regular text-text-weak">{language.t("context.breakdown.title")}</div>
287
+ <div class="h-2 w-full rounded-full bg-surface-base overflow-hidden flex">
288
+ <For each={breakdown()}>
289
+ {(segment) => (
290
+ <div
291
+ class="h-full"
292
+ style={{
293
+ width: `${segment.width}%`,
294
+ "background-color": BREAKDOWN_COLOR[segment.key],
295
+ }}
296
+ />
297
+ )}
298
+ </For>
299
+ </div>
300
+ <div class="flex flex-wrap gap-x-3 gap-y-1">
301
+ <For each={breakdown()}>
302
+ {(segment) => (
303
+ <div class="flex items-center gap-1 text-11-regular text-text-weak">
304
+ <div class="size-2 rounded-sm" style={{ "background-color": BREAKDOWN_COLOR[segment.key] }} />
305
+ <div>{breakdownLabel(segment.key)}</div>
306
+ <div class="text-text-weaker">{segment.percent.toLocaleString(language.intl())}%</div>
307
+ </div>
308
+ )}
309
+ </For>
310
+ </div>
311
+ <div class="hidden text-11-regular text-text-weaker">{language.t("context.breakdown.note")}</div>
312
+ </div>
313
+ </Show>
314
+
315
+ <Show when={systemPrompt()}>
316
+ {(prompt) => (
317
+ <div class="flex flex-col gap-2">
318
+ <div class="text-12-regular text-text-weak">{language.t("context.systemPrompt.title")}</div>
319
+ <div class="border border-border-base rounded-md bg-surface-base px-3 py-2">
320
+ <Markdown text={prompt()} class="text-12-regular" />
321
+ </div>
322
+ </div>
323
+ )}
324
+ </Show>
325
+
326
+ <div class="flex flex-col gap-2">
327
+ <div class="text-12-regular text-text-weak">{language.t("context.rawMessages.title")}</div>
328
+ <Accordion multiple>
329
+ <For each={messages()}>
330
+ {(message) => (
331
+ <RawMessage message={message} getParts={getParts} onRendered={restoreScroll} time={formatter().time} />
332
+ )}
333
+ </For>
334
+ </Accordion>
335
+ </div>
336
+ </div>
337
+ </ScrollView>
338
+ )
339
+ }