@eminent337/aery 0.1.11 → 1.0.0

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 (547) hide show
  1. package/CHANGELOG.md +898 -898
  2. package/README.md +2 -32
  3. package/docs/compaction.md +14 -14
  4. package/docs/custom-provider.md +9 -9
  5. package/docs/development.md +2 -2
  6. package/docs/extensions.md +25 -25
  7. package/docs/json.md +4 -4
  8. package/docs/packages.md +2 -2
  9. package/docs/providers.md +1 -1
  10. package/docs/rpc.md +1 -1
  11. package/docs/sdk.md +22 -22
  12. package/docs/session.md +7 -7
  13. package/docs/termux.md +1 -1
  14. package/docs/themes.md +2 -2
  15. package/docs/tui.md +7 -7
  16. package/examples/extensions/dynamic-resources/dynamic.json +1 -1
  17. package/package.json +5 -5
  18. package/dist/bun/cli.d.ts +0 -3
  19. package/dist/bun/cli.d.ts.map +0 -1
  20. package/dist/bun/cli.js +0 -7
  21. package/dist/bun/cli.js.map +0 -1
  22. package/dist/bun/register-bedrock.d.ts +0 -2
  23. package/dist/bun/register-bedrock.d.ts.map +0 -1
  24. package/dist/bun/register-bedrock.js +0 -4
  25. package/dist/bun/register-bedrock.js.map +0 -1
  26. package/dist/cli/args.d.ts +0 -52
  27. package/dist/cli/args.d.ts.map +0 -1
  28. package/dist/cli/args.js +0 -321
  29. package/dist/cli/args.js.map +0 -1
  30. package/dist/cli/config-selector.d.ts +0 -14
  31. package/dist/cli/config-selector.d.ts.map +0 -1
  32. package/dist/cli/config-selector.js +0 -31
  33. package/dist/cli/config-selector.js.map +0 -1
  34. package/dist/cli/file-processor.d.ts +0 -15
  35. package/dist/cli/file-processor.d.ts.map +0 -1
  36. package/dist/cli/file-processor.js +0 -83
  37. package/dist/cli/file-processor.js.map +0 -1
  38. package/dist/cli/initial-message.d.ts +0 -18
  39. package/dist/cli/initial-message.d.ts.map +0 -1
  40. package/dist/cli/initial-message.js +0 -22
  41. package/dist/cli/initial-message.js.map +0 -1
  42. package/dist/cli/list-models.d.ts +0 -9
  43. package/dist/cli/list-models.d.ts.map +0 -1
  44. package/dist/cli/list-models.js +0 -97
  45. package/dist/cli/list-models.js.map +0 -1
  46. package/dist/cli/session-picker.d.ts +0 -9
  47. package/dist/cli/session-picker.d.ts.map +0 -1
  48. package/dist/cli/session-picker.js +0 -35
  49. package/dist/cli/session-picker.js.map +0 -1
  50. package/dist/cli.d.ts +0 -3
  51. package/dist/cli.d.ts.map +0 -1
  52. package/dist/cli.js +0 -15
  53. package/dist/cli.js.map +0 -1
  54. package/dist/config.d.ts +0 -77
  55. package/dist/config.d.ts.map +0 -1
  56. package/dist/config.js +0 -221
  57. package/dist/config.js.map +0 -1
  58. package/dist/core/agent-session-runtime.d.ts +0 -99
  59. package/dist/core/agent-session-runtime.d.ts.map +0 -1
  60. package/dist/core/agent-session-runtime.js +0 -261
  61. package/dist/core/agent-session-runtime.js.map +0 -1
  62. package/dist/core/agent-session-services.d.ts +0 -85
  63. package/dist/core/agent-session-services.d.ts.map +0 -1
  64. package/dist/core/agent-session-services.js +0 -116
  65. package/dist/core/agent-session-services.js.map +0 -1
  66. package/dist/core/agent-session.d.ts +0 -590
  67. package/dist/core/agent-session.d.ts.map +0 -1
  68. package/dist/core/agent-session.js +0 -2505
  69. package/dist/core/agent-session.js.map +0 -1
  70. package/dist/core/auth-storage.d.ts +0 -132
  71. package/dist/core/auth-storage.d.ts.map +0 -1
  72. package/dist/core/auth-storage.js +0 -422
  73. package/dist/core/auth-storage.js.map +0 -1
  74. package/dist/core/bash-executor.d.ts +0 -32
  75. package/dist/core/bash-executor.d.ts.map +0 -1
  76. package/dist/core/bash-executor.js +0 -108
  77. package/dist/core/bash-executor.js.map +0 -1
  78. package/dist/core/compaction/branch-summarization.d.ts +0 -88
  79. package/dist/core/compaction/branch-summarization.d.ts.map +0 -1
  80. package/dist/core/compaction/branch-summarization.js +0 -243
  81. package/dist/core/compaction/branch-summarization.js.map +0 -1
  82. package/dist/core/compaction/compaction.d.ts +0 -121
  83. package/dist/core/compaction/compaction.d.ts.map +0 -1
  84. package/dist/core/compaction/compaction.js +0 -615
  85. package/dist/core/compaction/compaction.js.map +0 -1
  86. package/dist/core/compaction/index.d.ts +0 -7
  87. package/dist/core/compaction/index.d.ts.map +0 -1
  88. package/dist/core/compaction/index.js +0 -7
  89. package/dist/core/compaction/index.js.map +0 -1
  90. package/dist/core/compaction/utils.d.ts +0 -38
  91. package/dist/core/compaction/utils.d.ts.map +0 -1
  92. package/dist/core/compaction/utils.js +0 -153
  93. package/dist/core/compaction/utils.js.map +0 -1
  94. package/dist/core/defaults.d.ts +0 -3
  95. package/dist/core/defaults.d.ts.map +0 -1
  96. package/dist/core/defaults.js +0 -2
  97. package/dist/core/defaults.js.map +0 -1
  98. package/dist/core/diagnostics.d.ts +0 -15
  99. package/dist/core/diagnostics.d.ts.map +0 -1
  100. package/dist/core/diagnostics.js +0 -2
  101. package/dist/core/diagnostics.js.map +0 -1
  102. package/dist/core/event-bus.d.ts +0 -9
  103. package/dist/core/event-bus.d.ts.map +0 -1
  104. package/dist/core/event-bus.js +0 -25
  105. package/dist/core/event-bus.js.map +0 -1
  106. package/dist/core/exec.d.ts +0 -29
  107. package/dist/core/exec.d.ts.map +0 -1
  108. package/dist/core/exec.js +0 -75
  109. package/dist/core/exec.js.map +0 -1
  110. package/dist/core/export-html/ansi-to-html.d.ts +0 -22
  111. package/dist/core/export-html/ansi-to-html.d.ts.map +0 -1
  112. package/dist/core/export-html/ansi-to-html.js +0 -249
  113. package/dist/core/export-html/ansi-to-html.js.map +0 -1
  114. package/dist/core/export-html/index.d.ts +0 -37
  115. package/dist/core/export-html/index.d.ts.map +0 -1
  116. package/dist/core/export-html/index.js +0 -224
  117. package/dist/core/export-html/index.js.map +0 -1
  118. package/dist/core/export-html/template.css +0 -1017
  119. package/dist/core/export-html/template.html +0 -55
  120. package/dist/core/export-html/template.js +0 -1729
  121. package/dist/core/export-html/tool-renderer.d.ts +0 -40
  122. package/dist/core/export-html/tool-renderer.d.ts.map +0 -1
  123. package/dist/core/export-html/tool-renderer.js +0 -95
  124. package/dist/core/export-html/tool-renderer.js.map +0 -1
  125. package/dist/core/export-html/vendor/highlight.min.js +0 -1213
  126. package/dist/core/export-html/vendor/marked.min.js +0 -6
  127. package/dist/core/extensions/index.d.ts +0 -12
  128. package/dist/core/extensions/index.d.ts.map +0 -1
  129. package/dist/core/extensions/index.js +0 -9
  130. package/dist/core/extensions/index.js.map +0 -1
  131. package/dist/core/extensions/loader.d.ts +0 -25
  132. package/dist/core/extensions/loader.d.ts.map +0 -1
  133. package/dist/core/extensions/loader.js +0 -436
  134. package/dist/core/extensions/loader.js.map +0 -1
  135. package/dist/core/extensions/runner.d.ts +0 -150
  136. package/dist/core/extensions/runner.d.ts.map +0 -1
  137. package/dist/core/extensions/runner.js +0 -706
  138. package/dist/core/extensions/runner.js.map +0 -1
  139. package/dist/core/extensions/types.d.ts +0 -1111
  140. package/dist/core/extensions/types.d.ts.map +0 -1
  141. package/dist/core/extensions/types.js +0 -45
  142. package/dist/core/extensions/types.js.map +0 -1
  143. package/dist/core/extensions/wrapper.d.ts +0 -20
  144. package/dist/core/extensions/wrapper.d.ts.map +0 -1
  145. package/dist/core/extensions/wrapper.js +0 -22
  146. package/dist/core/extensions/wrapper.js.map +0 -1
  147. package/dist/core/footer-data-provider.d.ts +0 -48
  148. package/dist/core/footer-data-provider.d.ts.map +0 -1
  149. package/dist/core/footer-data-provider.js +0 -314
  150. package/dist/core/footer-data-provider.js.map +0 -1
  151. package/dist/core/index.d.ts +0 -12
  152. package/dist/core/index.d.ts.map +0 -1
  153. package/dist/core/index.js +0 -12
  154. package/dist/core/index.js.map +0 -1
  155. package/dist/core/keybindings.d.ts +0 -353
  156. package/dist/core/keybindings.d.ts.map +0 -1
  157. package/dist/core/keybindings.js +0 -295
  158. package/dist/core/keybindings.js.map +0 -1
  159. package/dist/core/messages.d.ts +0 -77
  160. package/dist/core/messages.d.ts.map +0 -1
  161. package/dist/core/messages.js +0 -123
  162. package/dist/core/messages.js.map +0 -1
  163. package/dist/core/model-registry.d.ts +0 -132
  164. package/dist/core/model-registry.d.ts.map +0 -1
  165. package/dist/core/model-registry.js +0 -635
  166. package/dist/core/model-registry.js.map +0 -1
  167. package/dist/core/model-resolver.d.ts +0 -110
  168. package/dist/core/model-resolver.d.ts.map +0 -1
  169. package/dist/core/model-resolver.js +0 -486
  170. package/dist/core/model-resolver.js.map +0 -1
  171. package/dist/core/output-guard.d.ts +0 -6
  172. package/dist/core/output-guard.d.ts.map +0 -1
  173. package/dist/core/output-guard.js +0 -59
  174. package/dist/core/output-guard.js.map +0 -1
  175. package/dist/core/package-manager.d.ts +0 -194
  176. package/dist/core/package-manager.d.ts.map +0 -1
  177. package/dist/core/package-manager.js +0 -1882
  178. package/dist/core/package-manager.js.map +0 -1
  179. package/dist/core/prompt-templates.d.ts +0 -52
  180. package/dist/core/prompt-templates.d.ts.map +0 -1
  181. package/dist/core/prompt-templates.js +0 -250
  182. package/dist/core/prompt-templates.js.map +0 -1
  183. package/dist/core/resolve-config-value.d.ts +0 -23
  184. package/dist/core/resolve-config-value.d.ts.map +0 -1
  185. package/dist/core/resolve-config-value.js +0 -126
  186. package/dist/core/resolve-config-value.js.map +0 -1
  187. package/dist/core/resource-loader.d.ts +0 -194
  188. package/dist/core/resource-loader.d.ts.map +0 -1
  189. package/dist/core/resource-loader.js +0 -726
  190. package/dist/core/resource-loader.js.map +0 -1
  191. package/dist/core/sdk.d.ts +0 -99
  192. package/dist/core/sdk.d.ts.map +0 -1
  193. package/dist/core/sdk.js +0 -261
  194. package/dist/core/sdk.js.map +0 -1
  195. package/dist/core/session-cwd.d.ts +0 -19
  196. package/dist/core/session-cwd.d.ts.map +0 -1
  197. package/dist/core/session-cwd.js +0 -38
  198. package/dist/core/session-cwd.js.map +0 -1
  199. package/dist/core/session-manager.d.ts +0 -333
  200. package/dist/core/session-manager.d.ts.map +0 -1
  201. package/dist/core/session-manager.js +0 -1109
  202. package/dist/core/session-manager.js.map +0 -1
  203. package/dist/core/settings-manager.d.ts +0 -240
  204. package/dist/core/settings-manager.d.ts.map +0 -1
  205. package/dist/core/settings-manager.js +0 -711
  206. package/dist/core/settings-manager.js.map +0 -1
  207. package/dist/core/skills.d.ts +0 -60
  208. package/dist/core/skills.d.ts.map +0 -1
  209. package/dist/core/skills.js +0 -409
  210. package/dist/core/skills.js.map +0 -1
  211. package/dist/core/slash-commands.d.ts +0 -14
  212. package/dist/core/slash-commands.d.ts.map +0 -1
  213. package/dist/core/slash-commands.js +0 -24
  214. package/dist/core/slash-commands.js.map +0 -1
  215. package/dist/core/source-info.d.ts +0 -18
  216. package/dist/core/source-info.d.ts.map +0 -1
  217. package/dist/core/source-info.js +0 -19
  218. package/dist/core/source-info.js.map +0 -1
  219. package/dist/core/system-prompt.d.ts +0 -28
  220. package/dist/core/system-prompt.d.ts.map +0 -1
  221. package/dist/core/system-prompt.js +0 -120
  222. package/dist/core/system-prompt.js.map +0 -1
  223. package/dist/core/telemetry.d.ts +0 -3
  224. package/dist/core/telemetry.d.ts.map +0 -1
  225. package/dist/core/telemetry.js +0 -9
  226. package/dist/core/telemetry.js.map +0 -1
  227. package/dist/core/timings.d.ts +0 -8
  228. package/dist/core/timings.d.ts.map +0 -1
  229. package/dist/core/timings.js +0 -31
  230. package/dist/core/timings.js.map +0 -1
  231. package/dist/core/tools/bash.d.ts +0 -64
  232. package/dist/core/tools/bash.d.ts.map +0 -1
  233. package/dist/core/tools/bash.js +0 -355
  234. package/dist/core/tools/bash.js.map +0 -1
  235. package/dist/core/tools/edit-diff.d.ts +0 -85
  236. package/dist/core/tools/edit-diff.d.ts.map +0 -1
  237. package/dist/core/tools/edit-diff.js +0 -337
  238. package/dist/core/tools/edit-diff.js.map +0 -1
  239. package/dist/core/tools/edit.d.ts +0 -49
  240. package/dist/core/tools/edit.d.ts.map +0 -1
  241. package/dist/core/tools/edit.js +0 -323
  242. package/dist/core/tools/edit.js.map +0 -1
  243. package/dist/core/tools/file-mutation-queue.d.ts +0 -6
  244. package/dist/core/tools/file-mutation-queue.d.ts.map +0 -1
  245. package/dist/core/tools/file-mutation-queue.js +0 -37
  246. package/dist/core/tools/file-mutation-queue.js.map +0 -1
  247. package/dist/core/tools/find.d.ts +0 -35
  248. package/dist/core/tools/find.d.ts.map +0 -1
  249. package/dist/core/tools/find.js +0 -298
  250. package/dist/core/tools/find.js.map +0 -1
  251. package/dist/core/tools/grep.d.ts +0 -37
  252. package/dist/core/tools/grep.d.ts.map +0 -1
  253. package/dist/core/tools/grep.js +0 -304
  254. package/dist/core/tools/grep.js.map +0 -1
  255. package/dist/core/tools/index.d.ts +0 -40
  256. package/dist/core/tools/index.d.ts.map +0 -1
  257. package/dist/core/tools/index.js +0 -112
  258. package/dist/core/tools/index.js.map +0 -1
  259. package/dist/core/tools/ls.d.ts +0 -37
  260. package/dist/core/tools/ls.d.ts.map +0 -1
  261. package/dist/core/tools/ls.js +0 -169
  262. package/dist/core/tools/ls.js.map +0 -1
  263. package/dist/core/tools/path-utils.d.ts +0 -8
  264. package/dist/core/tools/path-utils.d.ts.map +0 -1
  265. package/dist/core/tools/path-utils.js +0 -81
  266. package/dist/core/tools/path-utils.js.map +0 -1
  267. package/dist/core/tools/read.d.ts +0 -35
  268. package/dist/core/tools/read.d.ts.map +0 -1
  269. package/dist/core/tools/read.js +0 -232
  270. package/dist/core/tools/read.js.map +0 -1
  271. package/dist/core/tools/render-utils.d.ts +0 -21
  272. package/dist/core/tools/render-utils.d.ts.map +0 -1
  273. package/dist/core/tools/render-utils.js +0 -49
  274. package/dist/core/tools/render-utils.js.map +0 -1
  275. package/dist/core/tools/tool-definition-wrapper.d.ts +0 -14
  276. package/dist/core/tools/tool-definition-wrapper.d.ts.map +0 -1
  277. package/dist/core/tools/tool-definition-wrapper.js +0 -34
  278. package/dist/core/tools/tool-definition-wrapper.js.map +0 -1
  279. package/dist/core/tools/truncate.d.ts +0 -70
  280. package/dist/core/tools/truncate.d.ts.map +0 -1
  281. package/dist/core/tools/truncate.js +0 -205
  282. package/dist/core/tools/truncate.js.map +0 -1
  283. package/dist/core/tools/write.d.ts +0 -26
  284. package/dist/core/tools/write.d.ts.map +0 -1
  285. package/dist/core/tools/write.js +0 -213
  286. package/dist/core/tools/write.js.map +0 -1
  287. package/dist/index.d.ts +0 -28
  288. package/dist/index.d.ts.map +0 -1
  289. package/dist/index.js +0 -41
  290. package/dist/index.js.map +0 -1
  291. package/dist/main.d.ts +0 -12
  292. package/dist/main.d.ts.map +0 -1
  293. package/dist/main.js +0 -589
  294. package/dist/main.js.map +0 -1
  295. package/dist/migrations.d.ts +0 -33
  296. package/dist/migrations.d.ts.map +0 -1
  297. package/dist/migrations.js +0 -281
  298. package/dist/migrations.js.map +0 -1
  299. package/dist/modes/index.d.ts +0 -9
  300. package/dist/modes/index.d.ts.map +0 -1
  301. package/dist/modes/index.js +0 -8
  302. package/dist/modes/index.js.map +0 -1
  303. package/dist/modes/interactive/assets/clankolas.png +0 -0
  304. package/dist/modes/interactive/components/armin.d.ts +0 -34
  305. package/dist/modes/interactive/components/armin.d.ts.map +0 -1
  306. package/dist/modes/interactive/components/armin.js +0 -333
  307. package/dist/modes/interactive/components/armin.js.map +0 -1
  308. package/dist/modes/interactive/components/assistant-message.d.ts +0 -20
  309. package/dist/modes/interactive/components/assistant-message.d.ts.map +0 -1
  310. package/dist/modes/interactive/components/assistant-message.js +0 -121
  311. package/dist/modes/interactive/components/assistant-message.js.map +0 -1
  312. package/dist/modes/interactive/components/bash-execution.d.ts +0 -34
  313. package/dist/modes/interactive/components/bash-execution.d.ts.map +0 -1
  314. package/dist/modes/interactive/components/bash-execution.js +0 -175
  315. package/dist/modes/interactive/components/bash-execution.js.map +0 -1
  316. package/dist/modes/interactive/components/bordered-loader.d.ts +0 -16
  317. package/dist/modes/interactive/components/bordered-loader.d.ts.map +0 -1
  318. package/dist/modes/interactive/components/bordered-loader.js +0 -54
  319. package/dist/modes/interactive/components/bordered-loader.js.map +0 -1
  320. package/dist/modes/interactive/components/branch-summary-message.d.ts +0 -16
  321. package/dist/modes/interactive/components/branch-summary-message.d.ts.map +0 -1
  322. package/dist/modes/interactive/components/branch-summary-message.js +0 -44
  323. package/dist/modes/interactive/components/branch-summary-message.js.map +0 -1
  324. package/dist/modes/interactive/components/compaction-summary-message.d.ts +0 -16
  325. package/dist/modes/interactive/components/compaction-summary-message.d.ts.map +0 -1
  326. package/dist/modes/interactive/components/compaction-summary-message.js +0 -45
  327. package/dist/modes/interactive/components/compaction-summary-message.js.map +0 -1
  328. package/dist/modes/interactive/components/config-selector.d.ts +0 -71
  329. package/dist/modes/interactive/components/config-selector.d.ts.map +0 -1
  330. package/dist/modes/interactive/components/config-selector.js +0 -479
  331. package/dist/modes/interactive/components/config-selector.js.map +0 -1
  332. package/dist/modes/interactive/components/countdown-timer.d.ts +0 -14
  333. package/dist/modes/interactive/components/countdown-timer.d.ts.map +0 -1
  334. package/dist/modes/interactive/components/countdown-timer.js +0 -33
  335. package/dist/modes/interactive/components/countdown-timer.js.map +0 -1
  336. package/dist/modes/interactive/components/custom-editor.d.ts +0 -21
  337. package/dist/modes/interactive/components/custom-editor.d.ts.map +0 -1
  338. package/dist/modes/interactive/components/custom-editor.js +0 -70
  339. package/dist/modes/interactive/components/custom-editor.js.map +0 -1
  340. package/dist/modes/interactive/components/custom-message.d.ts +0 -20
  341. package/dist/modes/interactive/components/custom-message.d.ts.map +0 -1
  342. package/dist/modes/interactive/components/custom-message.js +0 -79
  343. package/dist/modes/interactive/components/custom-message.js.map +0 -1
  344. package/dist/modes/interactive/components/daxnuts.d.ts +0 -23
  345. package/dist/modes/interactive/components/daxnuts.d.ts.map +0 -1
  346. package/dist/modes/interactive/components/daxnuts.js +0 -140
  347. package/dist/modes/interactive/components/daxnuts.js.map +0 -1
  348. package/dist/modes/interactive/components/diff.d.ts +0 -12
  349. package/dist/modes/interactive/components/diff.d.ts.map +0 -1
  350. package/dist/modes/interactive/components/diff.js +0 -133
  351. package/dist/modes/interactive/components/diff.js.map +0 -1
  352. package/dist/modes/interactive/components/dynamic-border.d.ts +0 -15
  353. package/dist/modes/interactive/components/dynamic-border.d.ts.map +0 -1
  354. package/dist/modes/interactive/components/dynamic-border.js +0 -21
  355. package/dist/modes/interactive/components/dynamic-border.js.map +0 -1
  356. package/dist/modes/interactive/components/earendil-announcement.d.ts +0 -5
  357. package/dist/modes/interactive/components/earendil-announcement.d.ts.map +0 -1
  358. package/dist/modes/interactive/components/earendil-announcement.js +0 -40
  359. package/dist/modes/interactive/components/earendil-announcement.js.map +0 -1
  360. package/dist/modes/interactive/components/extension-editor.d.ts +0 -20
  361. package/dist/modes/interactive/components/extension-editor.d.ts.map +0 -1
  362. package/dist/modes/interactive/components/extension-editor.js +0 -111
  363. package/dist/modes/interactive/components/extension-editor.js.map +0 -1
  364. package/dist/modes/interactive/components/extension-input.d.ts +0 -23
  365. package/dist/modes/interactive/components/extension-input.d.ts.map +0 -1
  366. package/dist/modes/interactive/components/extension-input.js +0 -61
  367. package/dist/modes/interactive/components/extension-input.js.map +0 -1
  368. package/dist/modes/interactive/components/extension-selector.d.ts +0 -24
  369. package/dist/modes/interactive/components/extension-selector.d.ts.map +0 -1
  370. package/dist/modes/interactive/components/extension-selector.js +0 -78
  371. package/dist/modes/interactive/components/extension-selector.js.map +0 -1
  372. package/dist/modes/interactive/components/footer.d.ts +0 -27
  373. package/dist/modes/interactive/components/footer.d.ts.map +0 -1
  374. package/dist/modes/interactive/components/footer.js +0 -201
  375. package/dist/modes/interactive/components/footer.js.map +0 -1
  376. package/dist/modes/interactive/components/index.d.ts +0 -32
  377. package/dist/modes/interactive/components/index.d.ts.map +0 -1
  378. package/dist/modes/interactive/components/index.js +0 -33
  379. package/dist/modes/interactive/components/index.js.map +0 -1
  380. package/dist/modes/interactive/components/keybinding-hints.d.ts +0 -8
  381. package/dist/modes/interactive/components/keybinding-hints.d.ts.map +0 -1
  382. package/dist/modes/interactive/components/keybinding-hints.js +0 -22
  383. package/dist/modes/interactive/components/keybinding-hints.js.map +0 -1
  384. package/dist/modes/interactive/components/login-dialog.d.ts +0 -42
  385. package/dist/modes/interactive/components/login-dialog.d.ts.map +0 -1
  386. package/dist/modes/interactive/components/login-dialog.js +0 -145
  387. package/dist/modes/interactive/components/login-dialog.js.map +0 -1
  388. package/dist/modes/interactive/components/model-selector.d.ts +0 -47
  389. package/dist/modes/interactive/components/model-selector.d.ts.map +0 -1
  390. package/dist/modes/interactive/components/model-selector.js +0 -278
  391. package/dist/modes/interactive/components/model-selector.js.map +0 -1
  392. package/dist/modes/interactive/components/oauth-selector.d.ts +0 -19
  393. package/dist/modes/interactive/components/oauth-selector.d.ts.map +0 -1
  394. package/dist/modes/interactive/components/oauth-selector.js +0 -97
  395. package/dist/modes/interactive/components/oauth-selector.js.map +0 -1
  396. package/dist/modes/interactive/components/scoped-models-selector.d.ts +0 -42
  397. package/dist/modes/interactive/components/scoped-models-selector.d.ts.map +0 -1
  398. package/dist/modes/interactive/components/scoped-models-selector.js +0 -290
  399. package/dist/modes/interactive/components/scoped-models-selector.js.map +0 -1
  400. package/dist/modes/interactive/components/session-selector-search.d.ts +0 -23
  401. package/dist/modes/interactive/components/session-selector-search.d.ts.map +0 -1
  402. package/dist/modes/interactive/components/session-selector-search.js +0 -155
  403. package/dist/modes/interactive/components/session-selector-search.js.map +0 -1
  404. package/dist/modes/interactive/components/session-selector.d.ts +0 -96
  405. package/dist/modes/interactive/components/session-selector.d.ts.map +0 -1
  406. package/dist/modes/interactive/components/session-selector.js +0 -865
  407. package/dist/modes/interactive/components/session-selector.js.map +0 -1
  408. package/dist/modes/interactive/components/settings-selector.d.ts +0 -60
  409. package/dist/modes/interactive/components/settings-selector.d.ts.map +0 -1
  410. package/dist/modes/interactive/components/settings-selector.js +0 -311
  411. package/dist/modes/interactive/components/settings-selector.js.map +0 -1
  412. package/dist/modes/interactive/components/show-images-selector.d.ts +0 -10
  413. package/dist/modes/interactive/components/show-images-selector.d.ts.map +0 -1
  414. package/dist/modes/interactive/components/show-images-selector.js +0 -39
  415. package/dist/modes/interactive/components/show-images-selector.js.map +0 -1
  416. package/dist/modes/interactive/components/skill-invocation-message.d.ts +0 -17
  417. package/dist/modes/interactive/components/skill-invocation-message.d.ts.map +0 -1
  418. package/dist/modes/interactive/components/skill-invocation-message.js +0 -47
  419. package/dist/modes/interactive/components/skill-invocation-message.js.map +0 -1
  420. package/dist/modes/interactive/components/theme-selector.d.ts +0 -11
  421. package/dist/modes/interactive/components/theme-selector.d.ts.map +0 -1
  422. package/dist/modes/interactive/components/theme-selector.js +0 -50
  423. package/dist/modes/interactive/components/theme-selector.js.map +0 -1
  424. package/dist/modes/interactive/components/thinking-selector.d.ts +0 -11
  425. package/dist/modes/interactive/components/thinking-selector.d.ts.map +0 -1
  426. package/dist/modes/interactive/components/thinking-selector.js +0 -51
  427. package/dist/modes/interactive/components/thinking-selector.js.map +0 -1
  428. package/dist/modes/interactive/components/tool-execution.d.ts +0 -60
  429. package/dist/modes/interactive/components/tool-execution.d.ts.map +0 -1
  430. package/dist/modes/interactive/components/tool-execution.js +0 -289
  431. package/dist/modes/interactive/components/tool-execution.js.map +0 -1
  432. package/dist/modes/interactive/components/tree-selector.d.ts +0 -89
  433. package/dist/modes/interactive/components/tree-selector.d.ts.map +0 -1
  434. package/dist/modes/interactive/components/tree-selector.js +0 -1092
  435. package/dist/modes/interactive/components/tree-selector.js.map +0 -1
  436. package/dist/modes/interactive/components/user-message-selector.d.ts +0 -30
  437. package/dist/modes/interactive/components/user-message-selector.d.ts.map +0 -1
  438. package/dist/modes/interactive/components/user-message-selector.js +0 -114
  439. package/dist/modes/interactive/components/user-message-selector.js.map +0 -1
  440. package/dist/modes/interactive/components/user-message.d.ts +0 -10
  441. package/dist/modes/interactive/components/user-message.d.ts.map +0 -1
  442. package/dist/modes/interactive/components/user-message.js +0 -29
  443. package/dist/modes/interactive/components/user-message.js.map +0 -1
  444. package/dist/modes/interactive/components/visual-truncate.d.ts +0 -24
  445. package/dist/modes/interactive/components/visual-truncate.d.ts.map +0 -1
  446. package/dist/modes/interactive/components/visual-truncate.js +0 -33
  447. package/dist/modes/interactive/components/visual-truncate.js.map +0 -1
  448. package/dist/modes/interactive/interactive-mode.d.ts +0 -343
  449. package/dist/modes/interactive/interactive-mode.d.ts.map +0 -1
  450. package/dist/modes/interactive/interactive-mode.js +0 -4315
  451. package/dist/modes/interactive/interactive-mode.js.map +0 -1
  452. package/dist/modes/interactive/theme/aery.json +0 -84
  453. package/dist/modes/interactive/theme/dark.json +0 -85
  454. package/dist/modes/interactive/theme/light.json +0 -84
  455. package/dist/modes/interactive/theme/theme-schema.json +0 -335
  456. package/dist/modes/interactive/theme/theme.d.ts +0 -81
  457. package/dist/modes/interactive/theme/theme.d.ts.map +0 -1
  458. package/dist/modes/interactive/theme/theme.js +0 -979
  459. package/dist/modes/interactive/theme/theme.js.map +0 -1
  460. package/dist/modes/print-mode.d.ts +0 -28
  461. package/dist/modes/print-mode.d.ts.map +0 -1
  462. package/dist/modes/print-mode.js +0 -141
  463. package/dist/modes/print-mode.js.map +0 -1
  464. package/dist/modes/rpc/jsonl.d.ts +0 -17
  465. package/dist/modes/rpc/jsonl.d.ts.map +0 -1
  466. package/dist/modes/rpc/jsonl.js +0 -49
  467. package/dist/modes/rpc/jsonl.js.map +0 -1
  468. package/dist/modes/rpc/rpc-client.d.ts +0 -224
  469. package/dist/modes/rpc/rpc-client.d.ts.map +0 -1
  470. package/dist/modes/rpc/rpc-client.js +0 -410
  471. package/dist/modes/rpc/rpc-client.js.map +0 -1
  472. package/dist/modes/rpc/rpc-mode.d.ts +0 -20
  473. package/dist/modes/rpc/rpc-mode.d.ts.map +0 -1
  474. package/dist/modes/rpc/rpc-mode.js +0 -601
  475. package/dist/modes/rpc/rpc-mode.js.map +0 -1
  476. package/dist/modes/rpc/rpc-types.d.ts +0 -419
  477. package/dist/modes/rpc/rpc-types.d.ts.map +0 -1
  478. package/dist/modes/rpc/rpc-types.js +0 -8
  479. package/dist/modes/rpc/rpc-types.js.map +0 -1
  480. package/dist/package-manager-cli.d.ts +0 -4
  481. package/dist/package-manager-cli.d.ts.map +0 -1
  482. package/dist/package-manager-cli.js +0 -234
  483. package/dist/package-manager-cli.js.map +0 -1
  484. package/dist/utils/changelog.d.ts +0 -21
  485. package/dist/utils/changelog.d.ts.map +0 -1
  486. package/dist/utils/changelog.js +0 -87
  487. package/dist/utils/changelog.js.map +0 -1
  488. package/dist/utils/child-process.d.ts +0 -11
  489. package/dist/utils/child-process.d.ts.map +0 -1
  490. package/dist/utils/child-process.js +0 -78
  491. package/dist/utils/child-process.js.map +0 -1
  492. package/dist/utils/clipboard-image.d.ts +0 -11
  493. package/dist/utils/clipboard-image.d.ts.map +0 -1
  494. package/dist/utils/clipboard-image.js +0 -245
  495. package/dist/utils/clipboard-image.js.map +0 -1
  496. package/dist/utils/clipboard-native.d.ts +0 -8
  497. package/dist/utils/clipboard-native.d.ts.map +0 -1
  498. package/dist/utils/clipboard-native.js +0 -14
  499. package/dist/utils/clipboard-native.js.map +0 -1
  500. package/dist/utils/clipboard.d.ts +0 -2
  501. package/dist/utils/clipboard.d.ts.map +0 -1
  502. package/dist/utils/clipboard.js +0 -78
  503. package/dist/utils/clipboard.js.map +0 -1
  504. package/dist/utils/exif-orientation.d.ts +0 -5
  505. package/dist/utils/exif-orientation.d.ts.map +0 -1
  506. package/dist/utils/exif-orientation.js +0 -158
  507. package/dist/utils/exif-orientation.js.map +0 -1
  508. package/dist/utils/frontmatter.d.ts +0 -8
  509. package/dist/utils/frontmatter.d.ts.map +0 -1
  510. package/dist/utils/frontmatter.js +0 -26
  511. package/dist/utils/frontmatter.js.map +0 -1
  512. package/dist/utils/git.d.ts +0 -26
  513. package/dist/utils/git.d.ts.map +0 -1
  514. package/dist/utils/git.js +0 -163
  515. package/dist/utils/git.js.map +0 -1
  516. package/dist/utils/image-convert.d.ts +0 -9
  517. package/dist/utils/image-convert.d.ts.map +0 -1
  518. package/dist/utils/image-convert.js +0 -39
  519. package/dist/utils/image-convert.js.map +0 -1
  520. package/dist/utils/image-resize.d.ts +0 -36
  521. package/dist/utils/image-resize.d.ts.map +0 -1
  522. package/dist/utils/image-resize.js +0 -137
  523. package/dist/utils/image-resize.js.map +0 -1
  524. package/dist/utils/mime.d.ts +0 -2
  525. package/dist/utils/mime.d.ts.map +0 -1
  526. package/dist/utils/mime.js +0 -26
  527. package/dist/utils/mime.js.map +0 -1
  528. package/dist/utils/paths.d.ts +0 -7
  529. package/dist/utils/paths.d.ts.map +0 -1
  530. package/dist/utils/paths.js +0 -19
  531. package/dist/utils/paths.js.map +0 -1
  532. package/dist/utils/photon.d.ts +0 -21
  533. package/dist/utils/photon.d.ts.map +0 -1
  534. package/dist/utils/photon.js +0 -121
  535. package/dist/utils/photon.js.map +0 -1
  536. package/dist/utils/shell.d.ts +0 -29
  537. package/dist/utils/shell.d.ts.map +0 -1
  538. package/dist/utils/shell.js +0 -203
  539. package/dist/utils/shell.js.map +0 -1
  540. package/dist/utils/sleep.d.ts +0 -5
  541. package/dist/utils/sleep.d.ts.map +0 -1
  542. package/dist/utils/sleep.js +0 -17
  543. package/dist/utils/sleep.js.map +0 -1
  544. package/dist/utils/tools-manager.d.ts +0 -3
  545. package/dist/utils/tools-manager.d.ts.map +0 -1
  546. package/dist/utils/tools-manager.js +0 -252
  547. package/dist/utils/tools-manager.js.map +0 -1
@@ -1,1729 +0,0 @@
1
- (function() {
2
- 'use strict';
3
-
4
- // ============================================================
5
- // DATA LOADING
6
- // ============================================================
7
-
8
- const base64 = document.getElementById('session-data').textContent;
9
- const binary = atob(base64);
10
- const bytes = new Uint8Array(binary.length);
11
- for (let i = 0; i < binary.length; i++) {
12
- bytes[i] = binary.charCodeAt(i);
13
- }
14
- const data = JSON.parse(new TextDecoder('utf-8').decode(bytes));
15
- const { header, entries, leafId: defaultLeafId, systemPrompt, tools, renderedTools } = data;
16
-
17
- // ============================================================
18
- // URL PARAMETER HANDLING
19
- // ============================================================
20
-
21
- // Parse URL parameters for deep linking: leafId and targetId
22
- // Check for injected params (when loaded in iframe via srcdoc) or use window.location
23
- const injectedParams = document.querySelector('meta[name="pi-url-params"]');
24
- const searchString = injectedParams ? injectedParams.content : window.location.search.substring(1);
25
- const urlParams = new URLSearchParams(searchString);
26
- const urlLeafId = urlParams.get('leafId');
27
- const urlTargetId = urlParams.get('targetId');
28
- // Use URL leafId if provided, otherwise fall back to session default
29
- const leafId = urlLeafId || defaultLeafId;
30
-
31
- // ============================================================
32
- // DATA STRUCTURES
33
- // ============================================================
34
-
35
- // Entry lookup by ID
36
- const byId = new Map();
37
- for (const entry of entries) {
38
- byId.set(entry.id, entry);
39
- }
40
-
41
- // Tool call lookup (toolCallId -> {name, arguments})
42
- const toolCallMap = new Map();
43
- for (const entry of entries) {
44
- if (entry.type === 'message' && entry.message.role === 'assistant') {
45
- const content = entry.message.content;
46
- if (Array.isArray(content)) {
47
- for (const block of content) {
48
- if (block.type === 'toolCall') {
49
- toolCallMap.set(block.id, { name: block.name, arguments: block.arguments });
50
- }
51
- }
52
- }
53
- }
54
- }
55
-
56
- // Label lookup (entryId -> label string)
57
- // Labels are stored in 'label' entries that reference their target via targetId
58
- const labelMap = new Map();
59
- for (const entry of entries) {
60
- if (entry.type === 'label' && entry.targetId && entry.label) {
61
- labelMap.set(entry.targetId, entry.label);
62
- }
63
- }
64
-
65
- // ============================================================
66
- // TREE DATA PREPARATION (no DOM, pure data)
67
- // ============================================================
68
-
69
- /**
70
- * Build tree structure from flat entries.
71
- * Returns array of root nodes, each with { entry, children, label }.
72
- */
73
- function buildTree() {
74
- const nodeMap = new Map();
75
- const roots = [];
76
-
77
- // Create nodes
78
- for (const entry of entries) {
79
- nodeMap.set(entry.id, {
80
- entry,
81
- children: [],
82
- label: labelMap.get(entry.id)
83
- });
84
- }
85
-
86
- // Build parent-child relationships
87
- for (const entry of entries) {
88
- const node = nodeMap.get(entry.id);
89
- if (entry.parentId === null || entry.parentId === undefined || entry.parentId === entry.id) {
90
- roots.push(node);
91
- } else {
92
- const parent = nodeMap.get(entry.parentId);
93
- if (parent) {
94
- parent.children.push(node);
95
- } else {
96
- roots.push(node);
97
- }
98
- }
99
- }
100
-
101
- // Sort children by timestamp
102
- function sortChildren(node) {
103
- node.children.sort((a, b) =>
104
- new Date(a.entry.timestamp).getTime() - new Date(b.entry.timestamp).getTime()
105
- );
106
- node.children.forEach(sortChildren);
107
- }
108
- roots.forEach(sortChildren);
109
-
110
- return roots;
111
- }
112
-
113
- /**
114
- * Build set of entry IDs on path from root to target.
115
- */
116
- function buildActivePathIds(targetId) {
117
- const ids = new Set();
118
- let current = byId.get(targetId);
119
- while (current) {
120
- ids.add(current.id);
121
- // Stop if no parent or self-referencing (root)
122
- if (!current.parentId || current.parentId === current.id) {
123
- break;
124
- }
125
- current = byId.get(current.parentId);
126
- }
127
- return ids;
128
- }
129
-
130
- /**
131
- * Get array of entries from root to target (the conversation path).
132
- */
133
- function getPath(targetId) {
134
- const path = [];
135
- let current = byId.get(targetId);
136
- while (current) {
137
- path.unshift(current);
138
- // Stop if no parent or self-referencing (root)
139
- if (!current.parentId || current.parentId === current.id) {
140
- break;
141
- }
142
- current = byId.get(current.parentId);
143
- }
144
- return path;
145
- }
146
-
147
- // Tree node lookup for finding leaves
148
- let treeNodeMap = null;
149
-
150
- /**
151
- * Find the newest leaf node reachable from a given node.
152
- * This allows clicking any node in a branch to show the full branch.
153
- * Children are sorted by timestamp, so the newest is always last.
154
- */
155
- function findNewestLeaf(nodeId) {
156
- // Build tree node map lazily
157
- if (!treeNodeMap) {
158
- treeNodeMap = new Map();
159
- const tree = buildTree();
160
- function mapNodes(node) {
161
- treeNodeMap.set(node.entry.id, node);
162
- node.children.forEach(mapNodes);
163
- }
164
- tree.forEach(mapNodes);
165
- }
166
-
167
- const node = treeNodeMap.get(nodeId);
168
- if (!node) return nodeId;
169
-
170
- // Follow the newest (last) child at each level
171
- let current = node;
172
- while (current.children.length > 0) {
173
- current = current.children[current.children.length - 1];
174
- }
175
- return current.entry.id;
176
- }
177
-
178
- /**
179
- * Flatten tree into list with indentation and connector info.
180
- * Returns array of { node, indent, showConnector, isLast, gutters, isVirtualRootChild, multipleRoots }.
181
- * Matches tree-selector.ts logic exactly.
182
- */
183
- function flattenTree(roots, activePathIds) {
184
- const result = [];
185
- const multipleRoots = roots.length > 1;
186
-
187
- // Mark which subtrees contain the active leaf
188
- const containsActive = new Map();
189
- function markActive(node) {
190
- let has = activePathIds.has(node.entry.id);
191
- for (const child of node.children) {
192
- if (markActive(child)) has = true;
193
- }
194
- containsActive.set(node, has);
195
- return has;
196
- }
197
- roots.forEach(markActive);
198
-
199
- // Stack: [node, indent, justBranched, showConnector, isLast, gutters, isVirtualRootChild]
200
- const stack = [];
201
-
202
- // Add roots (prioritize branch containing active leaf)
203
- const orderedRoots = [...roots].sort((a, b) =>
204
- Number(containsActive.get(b)) - Number(containsActive.get(a))
205
- );
206
- for (let i = orderedRoots.length - 1; i >= 0; i--) {
207
- const isLast = i === orderedRoots.length - 1;
208
- stack.push([orderedRoots[i], multipleRoots ? 1 : 0, multipleRoots, multipleRoots, isLast, [], multipleRoots]);
209
- }
210
-
211
- while (stack.length > 0) {
212
- const [node, indent, justBranched, showConnector, isLast, gutters, isVirtualRootChild] = stack.pop();
213
-
214
- result.push({ node, indent, showConnector, isLast, gutters, isVirtualRootChild, multipleRoots });
215
-
216
- const children = node.children;
217
- const multipleChildren = children.length > 1;
218
-
219
- // Order children (active branch first)
220
- const orderedChildren = [...children].sort((a, b) =>
221
- Number(containsActive.get(b)) - Number(containsActive.get(a))
222
- );
223
-
224
- // Calculate child indent (matches tree-selector.ts)
225
- let childIndent;
226
- if (multipleChildren) {
227
- // Parent branches: children get +1
228
- childIndent = indent + 1;
229
- } else if (justBranched && indent > 0) {
230
- // First generation after a branch: +1 for visual grouping
231
- childIndent = indent + 1;
232
- } else {
233
- // Single-child chain: stay flat
234
- childIndent = indent;
235
- }
236
-
237
- // Build gutters for children
238
- const connectorDisplayed = showConnector && !isVirtualRootChild;
239
- const currentDisplayIndent = multipleRoots ? Math.max(0, indent - 1) : indent;
240
- const connectorPosition = Math.max(0, currentDisplayIndent - 1);
241
- const childGutters = connectorDisplayed
242
- ? [...gutters, { position: connectorPosition, show: !isLast }]
243
- : gutters;
244
-
245
- // Add children in reverse order for stack
246
- for (let i = orderedChildren.length - 1; i >= 0; i--) {
247
- const childIsLast = i === orderedChildren.length - 1;
248
- stack.push([orderedChildren[i], childIndent, multipleChildren, multipleChildren, childIsLast, childGutters, false]);
249
- }
250
- }
251
-
252
- return result;
253
- }
254
-
255
- /**
256
- * Build ASCII prefix string for tree node.
257
- */
258
- function buildTreePrefix(flatNode) {
259
- const { indent, showConnector, isLast, gutters, isVirtualRootChild, multipleRoots } = flatNode;
260
- const displayIndent = multipleRoots ? Math.max(0, indent - 1) : indent;
261
- const connector = showConnector && !isVirtualRootChild ? (isLast ? '└─ ' : '├─ ') : '';
262
- const connectorPosition = connector ? displayIndent - 1 : -1;
263
-
264
- const totalChars = displayIndent * 3;
265
- const prefixChars = [];
266
- for (let i = 0; i < totalChars; i++) {
267
- const level = Math.floor(i / 3);
268
- const posInLevel = i % 3;
269
-
270
- const gutter = gutters.find(g => g.position === level);
271
- if (gutter) {
272
- prefixChars.push(posInLevel === 0 ? (gutter.show ? '│' : ' ') : ' ');
273
- } else if (connector && level === connectorPosition) {
274
- if (posInLevel === 0) {
275
- prefixChars.push(isLast ? '└' : '├');
276
- } else if (posInLevel === 1) {
277
- prefixChars.push('─');
278
- } else {
279
- prefixChars.push(' ');
280
- }
281
- } else {
282
- prefixChars.push(' ');
283
- }
284
- }
285
- return prefixChars.join('');
286
- }
287
-
288
- // ============================================================
289
- // FILTERING (pure data)
290
- // ============================================================
291
-
292
- let filterMode = 'default';
293
- let searchQuery = '';
294
-
295
- function hasTextContent(content) {
296
- if (typeof content === 'string') return content.trim().length > 0;
297
- if (Array.isArray(content)) {
298
- for (const c of content) {
299
- if (c.type === 'text' && c.text && c.text.trim().length > 0) return true;
300
- }
301
- }
302
- return false;
303
- }
304
-
305
- function extractContent(content) {
306
- if (typeof content === 'string') return content;
307
- if (Array.isArray(content)) {
308
- return content
309
- .filter(c => c.type === 'text' && c.text)
310
- .map(c => c.text)
311
- .join('');
312
- }
313
- return '';
314
- }
315
-
316
- function getSearchableText(entry, label) {
317
- const parts = [];
318
- if (label) parts.push(label);
319
-
320
- switch (entry.type) {
321
- case 'message': {
322
- const msg = entry.message;
323
- parts.push(msg.role);
324
- if (msg.content) parts.push(extractContent(msg.content));
325
- if (msg.role === 'bashExecution' && msg.command) parts.push(msg.command);
326
- break;
327
- }
328
- case 'custom_message':
329
- parts.push(entry.customType);
330
- parts.push(typeof entry.content === 'string' ? entry.content : extractContent(entry.content));
331
- break;
332
- case 'compaction':
333
- parts.push('compaction');
334
- break;
335
- case 'branch_summary':
336
- parts.push('branch summary', entry.summary);
337
- break;
338
- case 'model_change':
339
- parts.push('model', entry.modelId);
340
- break;
341
- case 'thinking_level_change':
342
- parts.push('thinking', entry.thinkingLevel);
343
- break;
344
- }
345
-
346
- return parts.join(' ').toLowerCase();
347
- }
348
-
349
- /**
350
- * Filter flat nodes based on current filterMode and searchQuery.
351
- */
352
- function filterNodes(flatNodes, currentLeafId) {
353
- const searchTokens = searchQuery.toLowerCase().split(/\s+/).filter(Boolean);
354
-
355
- const filtered = flatNodes.filter(flatNode => {
356
- const entry = flatNode.node.entry;
357
- const label = flatNode.node.label;
358
- const isCurrentLeaf = entry.id === currentLeafId;
359
-
360
- // Always show current leaf
361
- if (isCurrentLeaf) return true;
362
-
363
- // Hide assistant messages with only tool calls (no text) unless error/aborted
364
- if (entry.type === 'message' && entry.message.role === 'assistant') {
365
- const msg = entry.message;
366
- const hasText = hasTextContent(msg.content);
367
- const isErrorOrAborted = msg.stopReason && msg.stopReason !== 'stop' && msg.stopReason !== 'toolUse';
368
- if (!hasText && !isErrorOrAborted) return false;
369
- }
370
-
371
- // Apply filter mode
372
- const isSettingsEntry = ['label', 'custom', 'model_change', 'thinking_level_change'].includes(entry.type);
373
- let passesFilter = true;
374
-
375
- switch (filterMode) {
376
- case 'user-only':
377
- passesFilter = entry.type === 'message' && entry.message.role === 'user';
378
- break;
379
- case 'no-tools':
380
- passesFilter = !isSettingsEntry && !(entry.type === 'message' && entry.message.role === 'toolResult');
381
- break;
382
- case 'labeled-only':
383
- passesFilter = label !== undefined;
384
- break;
385
- case 'all':
386
- passesFilter = true;
387
- break;
388
- default: // 'default'
389
- passesFilter = !isSettingsEntry;
390
- break;
391
- }
392
-
393
- if (!passesFilter) return false;
394
-
395
- // Apply search filter
396
- if (searchTokens.length > 0) {
397
- const nodeText = getSearchableText(entry, label);
398
- if (!searchTokens.every(t => nodeText.includes(t))) return false;
399
- }
400
-
401
- return true;
402
- });
403
-
404
- // Recalculate visual structure based on visible tree
405
- recalculateVisualStructure(filtered, flatNodes);
406
-
407
- return filtered;
408
- }
409
-
410
- /**
411
- * Recompute indentation/connectors for the filtered view
412
- *
413
- * Filtering can hide intermediate entries; descendants attach to the nearest visible ancestor.
414
- * Keep indentation semantics aligned with flattenTree() so single-child chains don't drift right.
415
- */
416
- function recalculateVisualStructure(filteredNodes, allFlatNodes) {
417
- if (filteredNodes.length === 0) return;
418
-
419
- const visibleIds = new Set(filteredNodes.map(n => n.node.entry.id));
420
-
421
- // Build entry map for parent lookup (using full tree)
422
- const entryMap = new Map();
423
- for (const flatNode of allFlatNodes) {
424
- entryMap.set(flatNode.node.entry.id, flatNode);
425
- }
426
-
427
- // Find nearest visible ancestor for a node
428
- function findVisibleAncestor(nodeId) {
429
- let currentId = entryMap.get(nodeId)?.node.entry.parentId;
430
- while (currentId != null) {
431
- if (visibleIds.has(currentId)) {
432
- return currentId;
433
- }
434
- currentId = entryMap.get(currentId)?.node.entry.parentId;
435
- }
436
- return null;
437
- }
438
-
439
- // Build visible tree structure
440
- const visibleParent = new Map();
441
- const visibleChildren = new Map();
442
- visibleChildren.set(null, []); // root-level nodes
443
-
444
- for (const flatNode of filteredNodes) {
445
- const nodeId = flatNode.node.entry.id;
446
- const ancestorId = findVisibleAncestor(nodeId);
447
- visibleParent.set(nodeId, ancestorId);
448
-
449
- if (!visibleChildren.has(ancestorId)) {
450
- visibleChildren.set(ancestorId, []);
451
- }
452
- visibleChildren.get(ancestorId).push(nodeId);
453
- }
454
-
455
- // Update multipleRoots based on visible roots
456
- const visibleRootIds = visibleChildren.get(null);
457
- const multipleRoots = visibleRootIds.length > 1;
458
-
459
- // Build a map for quick lookup: nodeId → FlatNode
460
- const filteredNodeMap = new Map();
461
- for (const flatNode of filteredNodes) {
462
- filteredNodeMap.set(flatNode.node.entry.id, flatNode);
463
- }
464
-
465
- // DFS traversal of visible tree, applying same indentation rules as flattenTree()
466
- // Stack items: [nodeId, indent, justBranched, showConnector, isLast, gutters, isVirtualRootChild]
467
- const stack = [];
468
-
469
- // Add visible roots in reverse order (to process in forward order via stack)
470
- for (let i = visibleRootIds.length - 1; i >= 0; i--) {
471
- const isLast = i === visibleRootIds.length - 1;
472
- stack.push([
473
- visibleRootIds[i],
474
- multipleRoots ? 1 : 0,
475
- multipleRoots,
476
- multipleRoots,
477
- isLast,
478
- [],
479
- multipleRoots
480
- ]);
481
- }
482
-
483
- while (stack.length > 0) {
484
- const [nodeId, indent, justBranched, showConnector, isLast, gutters, isVirtualRootChild] = stack.pop();
485
-
486
- const flatNode = filteredNodeMap.get(nodeId);
487
- if (!flatNode) continue;
488
-
489
- // Update this node's visual properties
490
- flatNode.indent = indent;
491
- flatNode.showConnector = showConnector;
492
- flatNode.isLast = isLast;
493
- flatNode.gutters = gutters;
494
- flatNode.isVirtualRootChild = isVirtualRootChild;
495
- flatNode.multipleRoots = multipleRoots;
496
-
497
- // Get visible children of this node
498
- const children = visibleChildren.get(nodeId) || [];
499
- const multipleChildren = children.length > 1;
500
-
501
- // Calculate child indent using same rules as flattenTree():
502
- // - Parent branches (multiple children): children get +1
503
- // - Just branched and indent > 0: children get +1 for visual grouping
504
- // - Single-child chain: stay flat
505
- let childIndent;
506
- if (multipleChildren) {
507
- childIndent = indent + 1;
508
- } else if (justBranched && indent > 0) {
509
- childIndent = indent + 1;
510
- } else {
511
- childIndent = indent;
512
- }
513
-
514
- // Build gutters for children (same logic as flattenTree)
515
- const connectorDisplayed = showConnector && !isVirtualRootChild;
516
- const currentDisplayIndent = multipleRoots ? Math.max(0, indent - 1) : indent;
517
- const connectorPosition = Math.max(0, currentDisplayIndent - 1);
518
- const childGutters = connectorDisplayed
519
- ? [...gutters, { position: connectorPosition, show: !isLast }]
520
- : gutters;
521
-
522
- // Add children in reverse order (to process in forward order via stack)
523
- for (let i = children.length - 1; i >= 0; i--) {
524
- const childIsLast = i === children.length - 1;
525
- stack.push([
526
- children[i],
527
- childIndent,
528
- multipleChildren,
529
- multipleChildren,
530
- childIsLast,
531
- childGutters,
532
- false
533
- ]);
534
- }
535
- }
536
- }
537
-
538
- // ============================================================
539
- // TREE DISPLAY TEXT (pure data -> string)
540
- // ============================================================
541
-
542
- function shortenPath(p) {
543
- if (typeof p !== 'string') return '';
544
- if (p.startsWith('/Users/')) {
545
- const parts = p.split('/');
546
- if (parts.length > 2) return '~' + p.slice(('/Users/' + parts[2]).length);
547
- }
548
- if (p.startsWith('/home/')) {
549
- const parts = p.split('/');
550
- if (parts.length > 2) return '~' + p.slice(('/home/' + parts[2]).length);
551
- }
552
- return p;
553
- }
554
-
555
- function formatToolCall(name, args) {
556
- switch (name) {
557
- case 'read': {
558
- const path = shortenPath(String(args.path || args.file_path || ''));
559
- const offset = args.offset;
560
- const limit = args.limit;
561
- let display = path;
562
- if (offset !== undefined || limit !== undefined) {
563
- const start = offset ?? 1;
564
- const end = limit !== undefined ? start + limit - 1 : '';
565
- display += `:${start}${end ? `-${end}` : ''}`;
566
- }
567
- return `[read: ${display}]`;
568
- }
569
- case 'write':
570
- return `[write: ${shortenPath(String(args.path || args.file_path || ''))}]`;
571
- case 'edit':
572
- return `[edit: ${shortenPath(String(args.path || args.file_path || ''))}]`;
573
- case 'bash': {
574
- const rawCmd = String(args.command || '');
575
- const cmd = rawCmd.replace(/[\n\t]/g, ' ').trim().slice(0, 50);
576
- return `[bash: ${cmd}${rawCmd.length > 50 ? '...' : ''}]`;
577
- }
578
- case 'grep':
579
- return `[grep: /${args.pattern || ''}/ in ${shortenPath(String(args.path || '.'))}]`;
580
- case 'find':
581
- return `[find: ${args.pattern || ''} in ${shortenPath(String(args.path || '.'))}]`;
582
- case 'ls':
583
- return `[ls: ${shortenPath(String(args.path || '.'))}]`;
584
- default: {
585
- const argsStr = JSON.stringify(args).slice(0, 40);
586
- return `[${name}: ${argsStr}${JSON.stringify(args).length > 40 ? '...' : ''}]`;
587
- }
588
- }
589
- }
590
-
591
- function escapeHtml(text) {
592
- const div = document.createElement('div');
593
- div.textContent = text;
594
- return div.innerHTML;
595
- }
596
-
597
- /**
598
- * Truncate string to maxLen chars, append "..." if truncated.
599
- */
600
- function truncate(s, maxLen = 100) {
601
- if (s.length <= maxLen) return s;
602
- return s.slice(0, maxLen) + '...';
603
- }
604
-
605
- /**
606
- * Get display text for tree node (returns HTML string).
607
- */
608
- function getTreeNodeDisplayHtml(entry, label) {
609
- const normalize = s => s.replace(/[\n\t]/g, ' ').trim();
610
- const labelHtml = label ? `<span class="tree-label">[${escapeHtml(label)}]</span> ` : '';
611
-
612
- switch (entry.type) {
613
- case 'message': {
614
- const msg = entry.message;
615
- if (msg.role === 'user') {
616
- const content = truncate(normalize(extractContent(msg.content)));
617
- return labelHtml + `<span class="tree-role-user">user:</span> ${escapeHtml(content)}`;
618
- }
619
- if (msg.role === 'assistant') {
620
- const textContent = truncate(normalize(extractContent(msg.content)));
621
- if (textContent) {
622
- return labelHtml + `<span class="tree-role-assistant">assistant:</span> ${escapeHtml(textContent)}`;
623
- }
624
- if (msg.stopReason === 'aborted') {
625
- return labelHtml + `<span class="tree-role-assistant">assistant:</span> <span class="tree-muted">(aborted)</span>`;
626
- }
627
- if (msg.errorMessage) {
628
- return labelHtml + `<span class="tree-role-assistant">assistant:</span> <span class="tree-error">${escapeHtml(truncate(msg.errorMessage))}</span>`;
629
- }
630
- return labelHtml + `<span class="tree-role-assistant">assistant:</span> <span class="tree-muted">(no text)</span>`;
631
- }
632
- if (msg.role === 'toolResult') {
633
- const toolCall = msg.toolCallId ? toolCallMap.get(msg.toolCallId) : null;
634
- if (toolCall) {
635
- return labelHtml + `<span class="tree-role-tool">${escapeHtml(formatToolCall(toolCall.name, toolCall.arguments))}</span>`;
636
- }
637
- return labelHtml + `<span class="tree-role-tool">[${msg.toolName || 'tool'}]</span>`;
638
- }
639
- if (msg.role === 'bashExecution') {
640
- const cmd = truncate(normalize(msg.command || ''));
641
- return labelHtml + `<span class="tree-role-tool">[bash]:</span> ${escapeHtml(cmd)}`;
642
- }
643
- return labelHtml + `<span class="tree-muted">[${msg.role}]</span>`;
644
- }
645
- case 'compaction':
646
- return labelHtml + `<span class="tree-compaction">[compaction: ${Math.round(entry.tokensBefore/1000)}k tokens]</span>`;
647
- case 'branch_summary': {
648
- const summary = truncate(normalize(entry.summary || ''));
649
- return labelHtml + `<span class="tree-branch-summary">[branch summary]:</span> ${escapeHtml(summary)}`;
650
- }
651
- case 'custom_message': {
652
- const content = typeof entry.content === 'string' ? entry.content : extractContent(entry.content);
653
- return labelHtml + `<span class="tree-custom">[${escapeHtml(entry.customType)}]:</span> ${escapeHtml(truncate(normalize(content)))}`;
654
- }
655
- case 'model_change':
656
- return labelHtml + `<span class="tree-muted">[model: ${entry.modelId}]</span>`;
657
- case 'thinking_level_change':
658
- return labelHtml + `<span class="tree-muted">[thinking: ${entry.thinkingLevel}]</span>`;
659
- default:
660
- return labelHtml + `<span class="tree-muted">[${entry.type}]</span>`;
661
- }
662
- }
663
-
664
- // ============================================================
665
- // TREE RENDERING (DOM manipulation)
666
- // ============================================================
667
-
668
- let currentLeafId = leafId;
669
- let currentTargetId = urlTargetId || leafId;
670
- let treeRendered = false;
671
-
672
- function renderTree() {
673
- const tree = buildTree();
674
- const activePathIds = buildActivePathIds(currentLeafId);
675
- const flatNodes = flattenTree(tree, activePathIds);
676
- const filtered = filterNodes(flatNodes, currentLeafId);
677
- const container = document.getElementById('tree-container');
678
-
679
- // Full render only on first call or when filter/search changes
680
- if (!treeRendered) {
681
- container.innerHTML = '';
682
-
683
- for (const flatNode of filtered) {
684
- const entry = flatNode.node.entry;
685
- const isOnPath = activePathIds.has(entry.id);
686
- const isTarget = entry.id === currentTargetId;
687
-
688
- const div = document.createElement('div');
689
- div.className = 'tree-node';
690
- if (isOnPath) div.classList.add('in-path');
691
- if (isTarget) div.classList.add('active');
692
- div.dataset.id = entry.id;
693
-
694
- const prefix = buildTreePrefix(flatNode);
695
- const prefixSpan = document.createElement('span');
696
- prefixSpan.className = 'tree-prefix';
697
- prefixSpan.textContent = prefix;
698
-
699
- const marker = document.createElement('span');
700
- marker.className = 'tree-marker';
701
- marker.textContent = isOnPath ? '•' : ' ';
702
-
703
- const content = document.createElement('span');
704
- content.className = 'tree-content';
705
- content.innerHTML = getTreeNodeDisplayHtml(entry, flatNode.node.label);
706
-
707
- div.appendChild(prefixSpan);
708
- div.appendChild(marker);
709
- div.appendChild(content);
710
- // Navigate to the newest leaf through this node, but scroll to the clicked node
711
- div.addEventListener('click', () => {
712
- if (window.getSelection().toString()) return;
713
- const leafId = findNewestLeaf(entry.id);
714
- navigateTo(leafId, 'target', entry.id);
715
- });
716
-
717
- container.appendChild(div);
718
- }
719
-
720
- treeRendered = true;
721
- } else {
722
- // Just update markers and classes
723
- const nodes = container.querySelectorAll('.tree-node');
724
- for (const node of nodes) {
725
- const id = node.dataset.id;
726
- const isOnPath = activePathIds.has(id);
727
- const isTarget = id === currentTargetId;
728
-
729
- node.classList.toggle('in-path', isOnPath);
730
- node.classList.toggle('active', isTarget);
731
-
732
- const marker = node.querySelector('.tree-marker');
733
- if (marker) {
734
- marker.textContent = isOnPath ? '•' : ' ';
735
- }
736
- }
737
- }
738
-
739
- document.getElementById('tree-status').textContent = `${filtered.length} / ${flatNodes.length} entries`;
740
-
741
- // Scroll active node into view after layout
742
- setTimeout(() => {
743
- const activeNode = container.querySelector('.tree-node.active');
744
- if (activeNode) {
745
- activeNode.scrollIntoView({ block: 'nearest' });
746
- }
747
- }, 0);
748
- }
749
-
750
- function forceTreeRerender() {
751
- treeRendered = false;
752
- renderTree();
753
- }
754
-
755
- // ============================================================
756
- // MESSAGE RENDERING
757
- // ============================================================
758
-
759
- function formatTokens(count) {
760
- if (count < 1000) return count.toString();
761
- if (count < 10000) return (count / 1000).toFixed(1) + 'k';
762
- if (count < 1000000) return Math.round(count / 1000) + 'k';
763
- return (count / 1000000).toFixed(1) + 'M';
764
- }
765
-
766
- function formatTimestamp(ts) {
767
- if (!ts) return '';
768
- const date = new Date(ts);
769
- return date.toLocaleTimeString(undefined, { hour: '2-digit', minute: '2-digit', second: '2-digit' });
770
- }
771
-
772
- function replaceTabs(text) {
773
- return text.replace(/\t/g, ' ');
774
- }
775
-
776
- /** Safely coerce value to string for display. Returns null if invalid type. */
777
- function str(value) {
778
- if (typeof value === 'string') return value;
779
- if (value == null) return '';
780
- return null;
781
- }
782
-
783
- function getLanguageFromPath(filePath) {
784
- const ext = filePath.split('.').pop()?.toLowerCase();
785
- const extToLang = {
786
- ts: 'typescript', tsx: 'typescript', js: 'javascript', jsx: 'javascript',
787
- py: 'python', rb: 'ruby', rs: 'rust', go: 'go', java: 'java',
788
- c: 'c', cpp: 'cpp', h: 'c', hpp: 'cpp', cs: 'csharp',
789
- php: 'php', sh: 'bash', bash: 'bash', zsh: 'bash',
790
- sql: 'sql', html: 'html', css: 'css', scss: 'scss',
791
- json: 'json', yaml: 'yaml', yml: 'yaml', xml: 'xml',
792
- md: 'markdown', dockerfile: 'dockerfile'
793
- };
794
- return extToLang[ext];
795
- }
796
-
797
- function findToolResult(toolCallId) {
798
- for (const entry of entries) {
799
- if (entry.type === 'message' && entry.message.role === 'toolResult') {
800
- if (entry.message.toolCallId === toolCallId) {
801
- return entry.message;
802
- }
803
- }
804
- }
805
- return null;
806
- }
807
-
808
- function formatExpandableOutput(text, maxLines, lang) {
809
- text = replaceTabs(text);
810
- const lines = text.split('\n');
811
- const displayLines = lines.slice(0, maxLines);
812
- const remaining = lines.length - maxLines;
813
-
814
- if (lang) {
815
- let highlighted;
816
- try {
817
- highlighted = hljs.highlight(text, { language: lang }).value;
818
- } catch {
819
- highlighted = escapeHtml(text);
820
- }
821
-
822
- if (remaining > 0) {
823
- const previewCode = displayLines.join('\n');
824
- let previewHighlighted;
825
- try {
826
- previewHighlighted = hljs.highlight(previewCode, { language: lang }).value;
827
- } catch {
828
- previewHighlighted = escapeHtml(previewCode);
829
- }
830
-
831
- return `<div class="tool-output expandable" onclick="if(window.getSelection().toString())return;this.classList.toggle('expanded')">
832
- <div class="output-preview"><pre><code class="hljs">${previewHighlighted}</code></pre>
833
- <div class="expand-hint">... (${remaining} more lines)</div></div>
834
- <div class="output-full"><pre><code class="hljs">${highlighted}</code></pre></div></div>`;
835
- }
836
-
837
- return `<div class="tool-output"><pre><code class="hljs">${highlighted}</code></pre></div>`;
838
- }
839
-
840
- // Plain text output
841
- if (remaining > 0) {
842
- let out = '<div class="tool-output expandable" onclick="if(window.getSelection().toString())return;this.classList.toggle(\'expanded\')">';
843
- out += '<div class="output-preview">';
844
- for (const line of displayLines) {
845
- out += `<div>${escapeHtml(replaceTabs(line))}</div>`;
846
- }
847
- out += `<div class="expand-hint">... (${remaining} more lines)</div></div>`;
848
- out += '<div class="output-full">';
849
- for (const line of lines) {
850
- out += `<div>${escapeHtml(replaceTabs(line))}</div>`;
851
- }
852
- out += '</div></div>';
853
- return out;
854
- }
855
-
856
- let out = '<div class="tool-output">';
857
- for (const line of displayLines) {
858
- out += `<div>${escapeHtml(replaceTabs(line))}</div>`;
859
- }
860
- out += '</div>';
861
- return out;
862
- }
863
-
864
- function renderToolCall(call) {
865
- const result = findToolResult(call.id);
866
- const isError = result?.isError || false;
867
- const statusClass = result ? (isError ? 'error' : 'success') : 'pending';
868
-
869
- const getResultText = () => {
870
- if (!result) return '';
871
- const textBlocks = result.content.filter(c => c.type === 'text');
872
- return textBlocks.map(c => c.text).join('\n');
873
- };
874
-
875
- const getResultImages = () => {
876
- if (!result) return [];
877
- return result.content.filter(c => c.type === 'image');
878
- };
879
-
880
- const renderResultImages = () => {
881
- const images = getResultImages();
882
- if (images.length === 0) return '';
883
- return '<div class="tool-images">' +
884
- images.map(img => `<img src="data:${img.mimeType};base64,${img.data}" class="tool-image" />`).join('') +
885
- '</div>';
886
- };
887
-
888
- let html = `<div class="tool-execution ${statusClass}">`;
889
- const args = call.arguments || {};
890
- const name = call.name;
891
-
892
- const invalidArg = '<span class="tool-error">[invalid arg]</span>';
893
-
894
- switch (name) {
895
- case 'bash': {
896
- const command = str(args.command);
897
- const cmdDisplay = command === null ? invalidArg : escapeHtml(command || '...');
898
- html += `<div class="tool-command">$ ${cmdDisplay}</div>`;
899
- if (result) {
900
- const output = getResultText().trim();
901
- if (output) html += formatExpandableOutput(output, 5);
902
- }
903
- break;
904
- }
905
- case 'read': {
906
- const filePath = str(args.file_path ?? args.path);
907
- const offset = args.offset;
908
- const limit = args.limit;
909
-
910
- let pathHtml = filePath === null ? invalidArg : escapeHtml(shortenPath(filePath || ''));
911
- if (filePath !== null && (offset !== undefined || limit !== undefined)) {
912
- const startLine = offset ?? 1;
913
- const endLine = limit !== undefined ? startLine + limit - 1 : '';
914
- pathHtml += `<span class="line-numbers">:${startLine}${endLine ? '-' + endLine : ''}</span>`;
915
- }
916
-
917
- html += `<div class="tool-header"><span class="tool-name">read</span> <span class="tool-path">${pathHtml}</span></div>`;
918
- if (result) {
919
- html += renderResultImages();
920
- const output = getResultText();
921
- const lang = filePath ? getLanguageFromPath(filePath) : null;
922
- if (output) html += formatExpandableOutput(output, 10, lang);
923
- }
924
- break;
925
- }
926
- case 'write': {
927
- const filePath = str(args.file_path ?? args.path);
928
- const content = str(args.content);
929
-
930
- html += `<div class="tool-header"><span class="tool-name">write</span> <span class="tool-path">${filePath === null ? invalidArg : escapeHtml(shortenPath(filePath || ''))}</span>`;
931
- if (content !== null && content) {
932
- const lines = content.split('\n');
933
- if (lines.length > 10) html += ` <span class="line-count">(${lines.length} lines)</span>`;
934
- }
935
- html += '</div>';
936
-
937
- if (content === null) {
938
- html += `<div class="tool-error">[invalid content arg - expected string]</div>`;
939
- } else if (content) {
940
- const lang = filePath ? getLanguageFromPath(filePath) : null;
941
- html += formatExpandableOutput(content, 10, lang);
942
- }
943
- if (result) {
944
- const output = getResultText().trim();
945
- if (output) html += `<div class="tool-output"><div>${escapeHtml(output)}</div></div>`;
946
- }
947
- break;
948
- }
949
- case 'edit': {
950
- const filePath = str(args.file_path ?? args.path);
951
- html += `<div class="tool-header"><span class="tool-name">edit</span> <span class="tool-path">${filePath === null ? invalidArg : escapeHtml(shortenPath(filePath || ''))}</span></div>`;
952
-
953
- if (result?.details?.diff) {
954
- const diffLines = result.details.diff.split('\n');
955
- html += '<div class="tool-diff">';
956
- for (const line of diffLines) {
957
- const cls = line.match(/^\+/) ? 'diff-added' : line.match(/^-/) ? 'diff-removed' : 'diff-context';
958
- html += `<div class="${cls}">${escapeHtml(replaceTabs(line))}</div>`;
959
- }
960
- html += '</div>';
961
- } else if (result) {
962
- const output = getResultText().trim();
963
- if (output) html += `<div class="tool-output"><pre>${escapeHtml(output)}</pre></div>`;
964
- }
965
- break;
966
- }
967
- default: {
968
- // Check for pre-rendered custom tool HTML
969
- const rendered = renderedTools?.[call.id];
970
- if (rendered?.callHtml || rendered?.resultHtmlCollapsed || rendered?.resultHtmlExpanded) {
971
- // Custom tool with pre-rendered HTML from TUI renderer
972
- if (rendered.callHtml) {
973
- html += `<div class="tool-header ansi-rendered">${rendered.callHtml}</div>`;
974
- } else {
975
- html += `<div class="tool-header"><span class="tool-name">${escapeHtml(name)}</span></div>`;
976
- }
977
-
978
- if (rendered.resultHtmlCollapsed && rendered.resultHtmlExpanded && rendered.resultHtmlCollapsed !== rendered.resultHtmlExpanded) {
979
- // Both collapsed and expanded differ - render expandable section
980
- html += `<div class="tool-output expandable ansi-rendered" onclick="if(window.getSelection().toString())return;this.classList.toggle('expanded')">
981
- <div class="output-preview">${rendered.resultHtmlCollapsed}</div>
982
- <div class="output-full">${rendered.resultHtmlExpanded}</div>
983
- </div>`;
984
- } else if (rendered.resultHtmlExpanded) {
985
- // Only expanded exists (or collapsed is identical) - show directly
986
- html += `<div class="tool-output ansi-rendered">${rendered.resultHtmlExpanded}</div>`;
987
- } else if (result) {
988
- // No pre-rendered result HTML - fallback to JSON
989
- const output = getResultText();
990
- if (output) html += formatExpandableOutput(output, 10);
991
- }
992
- } else {
993
- // Fallback to JSON display (existing behavior)
994
- html += `<div class="tool-header"><span class="tool-name">${escapeHtml(name)}</span></div>`;
995
- html += `<div class="tool-output"><pre>${escapeHtml(JSON.stringify(args, null, 2))}</pre></div>`;
996
- if (result) {
997
- const output = getResultText();
998
- if (output) html += formatExpandableOutput(output, 10);
999
- }
1000
- }
1001
- }
1002
- }
1003
-
1004
- html += '</div>';
1005
- return html;
1006
- }
1007
-
1008
- /**
1009
- * Download the session data as a JSONL file.
1010
- * Reconstructs the original format: header line + entry lines.
1011
- */
1012
- window.downloadSessionJson = function() {
1013
- // Build JSONL content: header first, then all entries
1014
- const lines = [];
1015
- if (header) {
1016
- lines.push(JSON.stringify({ type: 'header', ...header }));
1017
- }
1018
- for (const entry of entries) {
1019
- lines.push(JSON.stringify(entry));
1020
- }
1021
- const jsonlContent = lines.join('\n');
1022
-
1023
- // Create download
1024
- const blob = new Blob([jsonlContent], { type: 'application/x-ndjson' });
1025
- const url = URL.createObjectURL(blob);
1026
- const a = document.createElement('a');
1027
- a.href = url;
1028
- a.download = `${header?.id || 'session'}.jsonl`;
1029
- document.body.appendChild(a);
1030
- a.click();
1031
- document.body.removeChild(a);
1032
- URL.revokeObjectURL(url);
1033
- }
1034
-
1035
- /**
1036
- * Build a shareable URL for a specific message.
1037
- * URL format: base?gistId&leafId=<leafId>&targetId=<entryId>
1038
- */
1039
- function buildShareUrl(entryId) {
1040
- // Check for injected base URL (used when loaded in iframe via srcdoc)
1041
- const baseUrlMeta = document.querySelector('meta[name="pi-share-base-url"]');
1042
- const baseUrl = baseUrlMeta ? baseUrlMeta.content : window.location.href.split('?')[0];
1043
-
1044
- const url = new URL(window.location.href);
1045
- // Find the gist ID (first query param without value, e.g., ?abc123)
1046
- const gistId = Array.from(url.searchParams.keys()).find(k => !url.searchParams.get(k));
1047
-
1048
- // Build the share URL
1049
- const params = new URLSearchParams();
1050
- params.set('leafId', currentLeafId);
1051
- params.set('targetId', entryId);
1052
-
1053
- // If we have an injected base URL (iframe context), use it directly
1054
- if (baseUrlMeta) {
1055
- return `${baseUrl}&${params.toString()}`;
1056
- }
1057
-
1058
- // Otherwise build from current location (direct file access)
1059
- url.search = gistId ? `?${gistId}&${params.toString()}` : `?${params.toString()}`;
1060
- return url.toString();
1061
- }
1062
-
1063
- /**
1064
- * Copy text to clipboard with visual feedback.
1065
- * Uses navigator.clipboard with fallback to execCommand for HTTP contexts.
1066
- */
1067
- async function copyToClipboard(text, button) {
1068
- let success = false;
1069
- try {
1070
- if (navigator.clipboard && navigator.clipboard.writeText) {
1071
- await navigator.clipboard.writeText(text);
1072
- success = true;
1073
- }
1074
- } catch (err) {
1075
- // Clipboard API failed, try fallback
1076
- }
1077
-
1078
- // Fallback for HTTP or when Clipboard API is unavailable
1079
- if (!success) {
1080
- try {
1081
- const textarea = document.createElement('textarea');
1082
- textarea.value = text;
1083
- textarea.style.position = 'fixed';
1084
- textarea.style.opacity = '0';
1085
- document.body.appendChild(textarea);
1086
- textarea.select();
1087
- success = document.execCommand('copy');
1088
- document.body.removeChild(textarea);
1089
- } catch (err) {
1090
- console.error('Failed to copy:', err);
1091
- }
1092
- }
1093
-
1094
- if (success && button) {
1095
- const originalHtml = button.innerHTML;
1096
- button.innerHTML = '✓';
1097
- button.classList.add('copied');
1098
- setTimeout(() => {
1099
- button.innerHTML = originalHtml;
1100
- button.classList.remove('copied');
1101
- }, 1500);
1102
- }
1103
- }
1104
-
1105
- /**
1106
- * Render the copy-link button HTML for a message.
1107
- */
1108
- function renderCopyLinkButton(entryId) {
1109
- return `<button class="copy-link-btn" data-entry-id="${entryId}" title="Copy link to this message">
1110
- <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
1111
- <path d="M10 13a5 5 0 0 0 7.54.54l3-3a5 5 0 0 0-7.07-7.07l-1.72 1.71"/>
1112
- <path d="M14 11a5 5 0 0 0-7.54-.54l-3 3a5 5 0 0 0 7.07 7.07l1.71-1.71"/>
1113
- </svg>
1114
- </button>`;
1115
- }
1116
-
1117
- function renderEntry(entry) {
1118
- const ts = formatTimestamp(entry.timestamp);
1119
- const tsHtml = ts ? `<div class="message-timestamp">${ts}</div>` : '';
1120
- const entryId = `entry-${entry.id}`;
1121
- const copyBtnHtml = renderCopyLinkButton(entry.id);
1122
-
1123
- if (entry.type === 'message') {
1124
- const msg = entry.message;
1125
-
1126
- if (msg.role === 'user') {
1127
- let html = `<div class="user-message" id="${entryId}">${copyBtnHtml}${tsHtml}`;
1128
- const content = msg.content;
1129
-
1130
- if (Array.isArray(content)) {
1131
- const images = content.filter(c => c.type === 'image');
1132
- if (images.length > 0) {
1133
- html += '<div class="message-images">';
1134
- for (const img of images) {
1135
- html += `<img src="data:${img.mimeType};base64,${img.data}" class="message-image" />`;
1136
- }
1137
- html += '</div>';
1138
- }
1139
- }
1140
-
1141
- const text = typeof content === 'string' ? content :
1142
- content.filter(c => c.type === 'text').map(c => c.text).join('\n');
1143
- if (text.trim()) {
1144
- html += `<div class="markdown-content">${safeMarkedParse(text)}</div>`;
1145
- }
1146
- html += '</div>';
1147
- return html;
1148
- }
1149
-
1150
- if (msg.role === 'assistant') {
1151
- let html = `<div class="assistant-message" id="${entryId}">${copyBtnHtml}${tsHtml}`;
1152
-
1153
- for (const block of msg.content) {
1154
- if (block.type === 'text' && block.text.trim()) {
1155
- html += `<div class="assistant-text markdown-content">${safeMarkedParse(block.text)}</div>`;
1156
- } else if (block.type === 'thinking' && block.thinking.trim()) {
1157
- html += `<div class="thinking-block">
1158
- <div class="thinking-text">${escapeHtml(block.thinking)}</div>
1159
- <div class="thinking-collapsed">Thinking ...</div>
1160
- </div>`;
1161
- }
1162
- }
1163
-
1164
- for (const block of msg.content) {
1165
- if (block.type === 'toolCall') {
1166
- html += renderToolCall(block);
1167
- }
1168
- }
1169
-
1170
- if (msg.stopReason === 'aborted') {
1171
- html += '<div class="error-text">Aborted</div>';
1172
- } else if (msg.stopReason === 'error') {
1173
- html += `<div class="error-text">Error: ${escapeHtml(msg.errorMessage || 'Unknown error')}</div>`;
1174
- }
1175
-
1176
- html += '</div>';
1177
- return html;
1178
- }
1179
-
1180
- if (msg.role === 'bashExecution') {
1181
- const isError = msg.cancelled || (msg.exitCode !== 0 && msg.exitCode !== null);
1182
- let html = `<div class="tool-execution ${isError ? 'error' : 'success'}" id="${entryId}">${tsHtml}`;
1183
- html += `<div class="tool-command">$ ${escapeHtml(msg.command)}</div>`;
1184
- if (msg.output) html += formatExpandableOutput(msg.output, 10);
1185
- if (msg.cancelled) {
1186
- html += '<div style="color: var(--warning)">(cancelled)</div>';
1187
- } else if (msg.exitCode !== 0 && msg.exitCode !== null) {
1188
- html += `<div style="color: var(--error)">(exit ${msg.exitCode})</div>`;
1189
- }
1190
- html += '</div>';
1191
- return html;
1192
- }
1193
-
1194
- if (msg.role === 'toolResult') return '';
1195
- }
1196
-
1197
- if (entry.type === 'model_change') {
1198
- return `<div class="model-change" id="${entryId}">${tsHtml}Switched to model: <span class="model-name">${escapeHtml(entry.provider)}/${escapeHtml(entry.modelId)}</span></div>`;
1199
- }
1200
-
1201
- if (entry.type === 'compaction') {
1202
- return `<div class="compaction" id="${entryId}" onclick="if(window.getSelection().toString())return;this.classList.toggle('expanded')">
1203
- <div class="compaction-label">[compaction]</div>
1204
- <div class="compaction-collapsed">Compacted from ${entry.tokensBefore.toLocaleString()} tokens</div>
1205
- <div class="compaction-content"><strong>Compacted from ${entry.tokensBefore.toLocaleString()} tokens</strong>\n\n${escapeHtml(entry.summary)}</div>
1206
- </div>`;
1207
- }
1208
-
1209
- if (entry.type === 'branch_summary') {
1210
- return `<div class="branch-summary" id="${entryId}">${tsHtml}
1211
- <div class="branch-summary-header">Branch Summary</div>
1212
- <div class="markdown-content">${safeMarkedParse(entry.summary)}</div>
1213
- </div>`;
1214
- }
1215
-
1216
- if (entry.type === 'custom_message' && entry.display) {
1217
- return `<div class="hook-message" id="${entryId}">${tsHtml}
1218
- <div class="hook-type">[${escapeHtml(entry.customType)}]</div>
1219
- <div class="markdown-content">${safeMarkedParse(typeof entry.content === 'string' ? entry.content : JSON.stringify(entry.content))}</div>
1220
- </div>`;
1221
- }
1222
-
1223
- return '';
1224
- }
1225
-
1226
- // ============================================================
1227
- // HEADER / STATS
1228
- // ============================================================
1229
-
1230
- function computeStats(entryList) {
1231
- let userMessages = 0, assistantMessages = 0, toolResults = 0;
1232
- let customMessages = 0, compactions = 0, branchSummaries = 0, toolCalls = 0;
1233
- const tokens = { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 };
1234
- const cost = { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 };
1235
- const models = new Set();
1236
-
1237
- for (const entry of entryList) {
1238
- if (entry.type === 'message') {
1239
- const msg = entry.message;
1240
- if (msg.role === 'user') userMessages++;
1241
- if (msg.role === 'assistant') {
1242
- assistantMessages++;
1243
- if (msg.model) models.add(msg.provider ? `${msg.provider}/${msg.model}` : msg.model);
1244
- if (msg.usage) {
1245
- tokens.input += msg.usage.input || 0;
1246
- tokens.output += msg.usage.output || 0;
1247
- tokens.cacheRead += msg.usage.cacheRead || 0;
1248
- tokens.cacheWrite += msg.usage.cacheWrite || 0;
1249
- if (msg.usage.cost) {
1250
- cost.input += msg.usage.cost.input || 0;
1251
- cost.output += msg.usage.cost.output || 0;
1252
- cost.cacheRead += msg.usage.cost.cacheRead || 0;
1253
- cost.cacheWrite += msg.usage.cost.cacheWrite || 0;
1254
- }
1255
- }
1256
- toolCalls += msg.content.filter(c => c.type === 'toolCall').length;
1257
- }
1258
- if (msg.role === 'toolResult') toolResults++;
1259
- } else if (entry.type === 'compaction') {
1260
- compactions++;
1261
- } else if (entry.type === 'branch_summary') {
1262
- branchSummaries++;
1263
- } else if (entry.type === 'custom_message') {
1264
- customMessages++;
1265
- }
1266
- }
1267
-
1268
- return { userMessages, assistantMessages, toolResults, customMessages, compactions, branchSummaries, toolCalls, tokens, cost, models: Array.from(models) };
1269
- }
1270
-
1271
- const globalStats = computeStats(entries);
1272
-
1273
- function renderHeader() {
1274
- const totalCost = globalStats.cost.input + globalStats.cost.output + globalStats.cost.cacheRead + globalStats.cost.cacheWrite;
1275
-
1276
- const tokenParts = [];
1277
- if (globalStats.tokens.input) tokenParts.push(`↑${formatTokens(globalStats.tokens.input)}`);
1278
- if (globalStats.tokens.output) tokenParts.push(`↓${formatTokens(globalStats.tokens.output)}`);
1279
- if (globalStats.tokens.cacheRead) tokenParts.push(`R${formatTokens(globalStats.tokens.cacheRead)}`);
1280
- if (globalStats.tokens.cacheWrite) tokenParts.push(`W${formatTokens(globalStats.tokens.cacheWrite)}`);
1281
-
1282
- const msgParts = [];
1283
- if (globalStats.userMessages) msgParts.push(`${globalStats.userMessages} user`);
1284
- if (globalStats.assistantMessages) msgParts.push(`${globalStats.assistantMessages} assistant`);
1285
- if (globalStats.toolResults) msgParts.push(`${globalStats.toolResults} tool results`);
1286
- if (globalStats.customMessages) msgParts.push(`${globalStats.customMessages} custom`);
1287
- if (globalStats.compactions) msgParts.push(`${globalStats.compactions} compactions`);
1288
- if (globalStats.branchSummaries) msgParts.push(`${globalStats.branchSummaries} branch summaries`);
1289
-
1290
- let html = `
1291
- <div class="header">
1292
- <h1>Session: ${escapeHtml(header?.id || 'unknown')}</h1>
1293
- <div class="help-bar">
1294
- <span class="help-hint">T toggle thinking · O toggle tools</span>
1295
- <div class="help-actions">
1296
- <button type="button" class="header-toggle-btn" data-action="toggle-thinking" title="Toggle thinking (T)">Toggle thinking</button>
1297
- <button type="button" class="header-toggle-btn" data-action="toggle-tools" title="Toggle tools (O)">Toggle tools</button>
1298
- <button type="button" class="download-json-btn" onclick="downloadSessionJson()" title="Download session as JSONL">↓ JSONL</button>
1299
- </div>
1300
- </div>
1301
- <div class="header-info">
1302
- <div class="info-item"><span class="info-label">Date:</span><span class="info-value">${header?.timestamp ? new Date(header.timestamp).toLocaleString() : 'unknown'}</span></div>
1303
- <div class="info-item"><span class="info-label">Models:</span><span class="info-value">${globalStats.models.join(', ') || 'unknown'}</span></div>
1304
- <div class="info-item"><span class="info-label">Messages:</span><span class="info-value">${msgParts.join(', ') || '0'}</span></div>
1305
- <div class="info-item"><span class="info-label">Tool Calls:</span><span class="info-value">${globalStats.toolCalls}</span></div>
1306
- <div class="info-item"><span class="info-label">Tokens:</span><span class="info-value">${tokenParts.join(' ') || '0'}</span></div>
1307
- <div class="info-item"><span class="info-label">Cost:</span><span class="info-value">$${totalCost.toFixed(3)}</span></div>
1308
- </div>
1309
- </div>`;
1310
-
1311
- // Render system prompt (user's base prompt, applies to all providers)
1312
- if (systemPrompt) {
1313
- const lines = systemPrompt.split('\n');
1314
- const previewLines = 10;
1315
- if (lines.length > previewLines) {
1316
- const preview = lines.slice(0, previewLines).join('\n');
1317
- const remaining = lines.length - previewLines;
1318
- html += `<div class="system-prompt expandable" onclick="if(window.getSelection().toString())return;this.classList.toggle('expanded')">
1319
- <div class="system-prompt-header">System Prompt</div>
1320
- <div class="system-prompt-preview">${escapeHtml(preview)}</div>
1321
- <div class="system-prompt-expand-hint">... (${remaining} more lines, click to expand)</div>
1322
- <div class="system-prompt-full">${escapeHtml(systemPrompt)}</div>
1323
- </div>`;
1324
- } else {
1325
- html += `<div class="system-prompt">
1326
- <div class="system-prompt-header">System Prompt</div>
1327
- <div class="system-prompt-full" style="display: block">${escapeHtml(systemPrompt)}</div>
1328
- </div>`;
1329
- }
1330
- }
1331
-
1332
- if (tools && tools.length > 0) {
1333
- html += `<div class="tools-list">
1334
- <div class="tools-header">Available Tools</div>
1335
- <div class="tools-content">
1336
- ${tools.map(t => {
1337
- const hasParams = t.parameters && typeof t.parameters === 'object' && t.parameters.properties && Object.keys(t.parameters.properties).length > 0;
1338
- if (!hasParams) {
1339
- return `<div class="tool-item"><span class="tool-item-name">${escapeHtml(t.name)}</span> - <span class="tool-item-desc">${escapeHtml(t.description)}</span></div>`;
1340
- }
1341
- const params = t.parameters;
1342
- const properties = params.properties;
1343
- const required = params.required || [];
1344
- let paramsHtml = '';
1345
- for (const [name, prop] of Object.entries(properties)) {
1346
- const isRequired = required.includes(name);
1347
- const typeStr = prop.type || 'any';
1348
- const reqLabel = isRequired ? '<span class="tool-param-required">required</span>' : '<span class="tool-param-optional">optional</span>';
1349
- paramsHtml += `<div class="tool-param"><span class="tool-param-name">${escapeHtml(name)}</span> <span class="tool-param-type">${escapeHtml(typeStr)}</span> ${reqLabel}`;
1350
- if (prop.description) {
1351
- paramsHtml += `<div class="tool-param-desc">${escapeHtml(prop.description)}</div>`;
1352
- }
1353
- paramsHtml += `</div>`;
1354
- }
1355
- return `<div class="tool-item" onclick="if(window.getSelection().toString())return;this.classList.toggle('params-expanded')"><span class="tool-item-name">${escapeHtml(t.name)}</span> - <span class="tool-item-desc">${escapeHtml(t.description)}</span> <span class="tool-params-hint"></span><div class="tool-params-content">${paramsHtml}</div></div>`;
1356
- }).join('')}
1357
- </div>
1358
- </div>`;
1359
- }
1360
-
1361
- return html;
1362
- }
1363
-
1364
- // ============================================================
1365
- // NAVIGATION
1366
- // ============================================================
1367
-
1368
- // Cache for rendered entry DOM nodes
1369
- const entryCache = new Map();
1370
-
1371
- function renderEntryToNode(entry) {
1372
- // Check cache first
1373
- if (entryCache.has(entry.id)) {
1374
- return entryCache.get(entry.id).cloneNode(true);
1375
- }
1376
-
1377
- // Render to HTML string, then parse to node
1378
- const html = renderEntry(entry);
1379
- if (!html) return null;
1380
-
1381
- const template = document.createElement('template');
1382
- template.innerHTML = html;
1383
- const node = template.content.firstElementChild;
1384
-
1385
- // Cache the node
1386
- if (node) {
1387
- entryCache.set(entry.id, node.cloneNode(true));
1388
- }
1389
- return node;
1390
- }
1391
-
1392
- function navigateTo(targetId, scrollMode = 'target', scrollToEntryId = null) {
1393
- currentLeafId = targetId;
1394
- currentTargetId = scrollToEntryId || targetId;
1395
- const path = getPath(targetId);
1396
-
1397
- renderTree();
1398
-
1399
- document.getElementById('header-container').innerHTML = renderHeader();
1400
- attachHeaderHandlers();
1401
-
1402
- // Build messages using cached DOM nodes
1403
- const messagesEl = document.getElementById('messages');
1404
- const fragment = document.createDocumentFragment();
1405
-
1406
- for (const entry of path) {
1407
- const node = renderEntryToNode(entry);
1408
- if (node) {
1409
- fragment.appendChild(node);
1410
- }
1411
- }
1412
-
1413
- messagesEl.innerHTML = '';
1414
- messagesEl.appendChild(fragment);
1415
-
1416
- // Attach click handlers for copy-link buttons
1417
- messagesEl.querySelectorAll('.copy-link-btn').forEach(btn => {
1418
- btn.addEventListener('click', (e) => {
1419
- e.stopPropagation();
1420
- const entryId = btn.dataset.entryId;
1421
- const shareUrl = buildShareUrl(entryId);
1422
- copyToClipboard(shareUrl, btn);
1423
- });
1424
- });
1425
-
1426
- // Use setTimeout(0) to ensure DOM is fully laid out before scrolling
1427
- setTimeout(() => {
1428
- const content = document.getElementById('content');
1429
- if (scrollMode === 'bottom') {
1430
- content.scrollTop = content.scrollHeight;
1431
- } else if (scrollMode === 'target') {
1432
- // If scrollToEntryId is provided, scroll to that specific entry
1433
- const scrollTargetId = scrollToEntryId || targetId;
1434
- const targetEl = document.getElementById(`entry-${scrollTargetId}`);
1435
- if (targetEl) {
1436
- targetEl.scrollIntoView({ block: 'center' });
1437
- // Briefly highlight the target message
1438
- if (scrollToEntryId) {
1439
- targetEl.classList.add('highlight');
1440
- setTimeout(() => targetEl.classList.remove('highlight'), 2000);
1441
- }
1442
- }
1443
- }
1444
- }, 0);
1445
- }
1446
-
1447
- // ============================================================
1448
- // INITIALIZATION
1449
- // ============================================================
1450
-
1451
- // Escape HTML tags in text (but not code blocks)
1452
- function escapeHtmlTags(text) {
1453
- return text.replace(/<(?=[a-zA-Z\/])/g, '&lt;');
1454
- }
1455
-
1456
- // Configure marked with syntax highlighting and HTML escaping for text
1457
- const strictStrikethroughRegex = /^(~~)(?=[^\s~])((?:\\.|[^\\])*?(?:\\.|[^\s~\\]))\1(?=[^~]|$)/;
1458
-
1459
- marked.use({
1460
- breaks: true,
1461
- gfm: true,
1462
- tokenizer: {
1463
- del(src) {
1464
- const match = strictStrikethroughRegex.exec(src);
1465
- if (!match) return undefined;
1466
- return {
1467
- type: 'del',
1468
- raw: match[0],
1469
- text: match[2],
1470
- tokens: this.lexer.inlineTokens(match[2])
1471
- };
1472
- }
1473
- },
1474
- renderer: {
1475
- // Code blocks: syntax highlight, no HTML escaping
1476
- code(token) {
1477
- const code = token.text;
1478
- const lang = token.lang;
1479
- let highlighted;
1480
- if (lang && hljs.getLanguage(lang)) {
1481
- try {
1482
- highlighted = hljs.highlight(code, { language: lang }).value;
1483
- } catch {
1484
- highlighted = escapeHtml(code);
1485
- }
1486
- } else {
1487
- // Auto-detect language if not specified
1488
- try {
1489
- highlighted = hljs.highlightAuto(code).value;
1490
- } catch {
1491
- highlighted = escapeHtml(code);
1492
- }
1493
- }
1494
- return `<pre><code class="hljs">${highlighted}</code></pre>`;
1495
- },
1496
- // Text content: escape HTML tags
1497
- text(token) {
1498
- return escapeHtmlTags(escapeHtml(token.text));
1499
- },
1500
- // Inline code: escape HTML
1501
- codespan(token) {
1502
- return `<code>${escapeHtml(token.text)}</code>`;
1503
- }
1504
- }
1505
- });
1506
-
1507
- // Simple marked parse (escaping handled in renderers)
1508
- function safeMarkedParse(text) {
1509
- return marked.parse(text);
1510
- }
1511
-
1512
- // Search input
1513
- const searchInput = document.getElementById('tree-search');
1514
- searchInput.addEventListener('input', (e) => {
1515
- searchQuery = e.target.value;
1516
- forceTreeRerender();
1517
- });
1518
-
1519
- // Filter buttons
1520
- document.querySelectorAll('.filter-btn').forEach(btn => {
1521
- btn.addEventListener('click', () => {
1522
- document.querySelectorAll('.filter-btn').forEach(b => b.classList.remove('active'));
1523
- btn.classList.add('active');
1524
- filterMode = btn.dataset.filter;
1525
- forceTreeRerender();
1526
- });
1527
- });
1528
-
1529
- // Sidebar toggle
1530
- const sidebar = document.getElementById('sidebar');
1531
- const overlay = document.getElementById('sidebar-overlay');
1532
- const hamburger = document.getElementById('hamburger');
1533
- const sidebarResizer = document.getElementById('sidebar-resizer');
1534
- const SIDEBAR_WIDTH_STORAGE_KEY = 'pi-share:v1:sidebar-width';
1535
- const MIN_CONTENT_WIDTH = 320;
1536
-
1537
- function isMobileLayout() {
1538
- return window.matchMedia('(max-width: 900px)').matches;
1539
- }
1540
-
1541
- function getSidebarBounds() {
1542
- const rootStyles = getComputedStyle(document.documentElement);
1543
- const minWidth = parseFloat(rootStyles.getPropertyValue('--sidebar-min-width')) || 240;
1544
- const maxWidth = parseFloat(rootStyles.getPropertyValue('--sidebar-max-width')) || 720;
1545
- const viewportMaxWidth = window.innerWidth - MIN_CONTENT_WIDTH;
1546
- return {
1547
- minWidth,
1548
- maxWidth: Math.max(minWidth, Math.min(maxWidth, viewportMaxWidth))
1549
- };
1550
- }
1551
-
1552
- function clampSidebarWidth(width) {
1553
- const { minWidth, maxWidth } = getSidebarBounds();
1554
- return Math.max(minWidth, Math.min(maxWidth, width));
1555
- }
1556
-
1557
- function applySidebarWidth(width) {
1558
- document.documentElement.style.setProperty('--sidebar-width', `${Math.round(clampSidebarWidth(width))}px`);
1559
- }
1560
-
1561
- function loadSidebarWidth() {
1562
- try {
1563
- const raw = localStorage.getItem(SIDEBAR_WIDTH_STORAGE_KEY);
1564
- if (raw === null) return null;
1565
- const width = Number(raw);
1566
- return Number.isFinite(width) ? width : null;
1567
- } catch {
1568
- return null;
1569
- }
1570
- }
1571
-
1572
- function saveSidebarWidth(width) {
1573
- try {
1574
- localStorage.setItem(SIDEBAR_WIDTH_STORAGE_KEY, String(Math.round(clampSidebarWidth(width))));
1575
- } catch {
1576
- // Ignore storage failures (e.g. private browsing restrictions)
1577
- }
1578
- }
1579
-
1580
- function setupSidebarResize() {
1581
- const savedWidth = loadSidebarWidth();
1582
- if (savedWidth !== null) {
1583
- applySidebarWidth(savedWidth);
1584
- }
1585
-
1586
- if (!sidebarResizer) return;
1587
-
1588
- let cleanupDrag = null;
1589
-
1590
- const stopDrag = (pointerId) => {
1591
- if (cleanupDrag) {
1592
- cleanupDrag(pointerId);
1593
- cleanupDrag = null;
1594
- }
1595
- };
1596
-
1597
- sidebarResizer.addEventListener('pointerdown', (e) => {
1598
- if (isMobileLayout()) return;
1599
-
1600
- e.preventDefault();
1601
- const startX = e.clientX;
1602
- const startWidth = sidebar.getBoundingClientRect().width;
1603
- document.body.classList.add('sidebar-resizing');
1604
- sidebarResizer.setPointerCapture?.(e.pointerId);
1605
-
1606
- const onPointerMove = (event) => {
1607
- applySidebarWidth(startWidth + (event.clientX - startX));
1608
- };
1609
-
1610
- cleanupDrag = (pointerIdToRelease) => {
1611
- document.body.classList.remove('sidebar-resizing');
1612
- sidebarResizer.releasePointerCapture?.(pointerIdToRelease);
1613
- window.removeEventListener('pointermove', onPointerMove);
1614
- window.removeEventListener('pointerup', onPointerUp);
1615
- window.removeEventListener('pointercancel', onPointerCancel);
1616
- saveSidebarWidth(sidebar.getBoundingClientRect().width);
1617
- };
1618
-
1619
- const onPointerUp = (event) => stopDrag(event.pointerId);
1620
- const onPointerCancel = (event) => stopDrag(event.pointerId);
1621
-
1622
- window.addEventListener('pointermove', onPointerMove);
1623
- window.addEventListener('pointerup', onPointerUp);
1624
- window.addEventListener('pointercancel', onPointerCancel);
1625
- });
1626
-
1627
- sidebarResizer.addEventListener('dblclick', () => {
1628
- if (isMobileLayout()) return;
1629
- applySidebarWidth(400);
1630
- saveSidebarWidth(400);
1631
- });
1632
-
1633
- window.addEventListener('resize', () => {
1634
- if (isMobileLayout()) return;
1635
- applySidebarWidth(sidebar.getBoundingClientRect().width);
1636
- });
1637
- }
1638
-
1639
- setupSidebarResize();
1640
-
1641
- hamburger.addEventListener('click', () => {
1642
- sidebar.classList.add('open');
1643
- overlay.classList.add('open');
1644
- hamburger.style.display = 'none';
1645
- });
1646
-
1647
- const closeSidebar = () => {
1648
- sidebar.classList.remove('open');
1649
- overlay.classList.remove('open');
1650
- hamburger.style.display = '';
1651
- };
1652
-
1653
- overlay.addEventListener('click', closeSidebar);
1654
- document.getElementById('sidebar-close').addEventListener('click', closeSidebar);
1655
-
1656
- // Toggle states
1657
- let thinkingExpanded = true;
1658
- let toolOutputsExpanded = false;
1659
-
1660
- const toggleThinking = () => {
1661
- thinkingExpanded = !thinkingExpanded;
1662
- document.querySelectorAll('.thinking-text').forEach(el => {
1663
- el.style.display = thinkingExpanded ? '' : 'none';
1664
- });
1665
- document.querySelectorAll('.thinking-collapsed').forEach(el => {
1666
- el.style.display = thinkingExpanded ? 'none' : 'block';
1667
- });
1668
- };
1669
-
1670
- const toggleToolOutputs = () => {
1671
- toolOutputsExpanded = !toolOutputsExpanded;
1672
- document.querySelectorAll('.tool-output.expandable').forEach(el => {
1673
- el.classList.toggle('expanded', toolOutputsExpanded);
1674
- });
1675
- document.querySelectorAll('.compaction').forEach(el => {
1676
- el.classList.toggle('expanded', toolOutputsExpanded);
1677
- });
1678
- };
1679
-
1680
- const attachHeaderHandlers = () => {
1681
- document.querySelector('[data-action="toggle-thinking"]')?.addEventListener('click', toggleThinking);
1682
- document.querySelector('[data-action="toggle-tools"]')?.addEventListener('click', toggleToolOutputs);
1683
- };
1684
-
1685
- const isEditableTarget = (element) => {
1686
- if (!element) return false;
1687
- const tagName = element.tagName;
1688
- if (tagName === 'INPUT' || tagName === 'TEXTAREA' || tagName === 'SELECT' || tagName === 'BUTTON') {
1689
- return true;
1690
- }
1691
- return element.isContentEditable || Boolean(element.closest?.('[contenteditable="true"]'));
1692
- };
1693
-
1694
- // Keyboard shortcuts
1695
- document.addEventListener('keydown', (e) => {
1696
- if (e.key === 'Escape') {
1697
- searchInput.value = '';
1698
- searchQuery = '';
1699
- navigateTo(leafId, 'bottom');
1700
- }
1701
-
1702
- if (isEditableTarget(document.activeElement)) {
1703
- return;
1704
- }
1705
-
1706
- const key = e.key.toLowerCase();
1707
- if (key === 't') {
1708
- e.preventDefault();
1709
- toggleThinking();
1710
- } else if (key === 'o') {
1711
- e.preventDefault();
1712
- toggleToolOutputs();
1713
- }
1714
- });
1715
-
1716
- // Initial render
1717
- // If URL has targetId, scroll to that specific message; otherwise stay at top
1718
- if (leafId) {
1719
- if (urlTargetId && byId.has(urlTargetId)) {
1720
- // Deep link: navigate to leaf and scroll to target message
1721
- navigateTo(leafId, 'target', urlTargetId);
1722
- } else {
1723
- navigateTo(leafId, 'none');
1724
- }
1725
- } else if (entries.length > 0) {
1726
- // Fallback: use last entry if no leafId
1727
- navigateTo(entries[entries.length - 1].id, 'none');
1728
- }
1729
- })();