@draht/web-ui 2026.3.2-2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (372) hide show
  1. package/CHANGELOG.md +276 -0
  2. package/README.md +601 -0
  3. package/dist/ChatPanel.d.ts +28 -0
  4. package/dist/ChatPanel.d.ts.map +1 -0
  5. package/dist/ChatPanel.js +193 -0
  6. package/dist/ChatPanel.js.map +1 -0
  7. package/dist/app.css +2 -0
  8. package/dist/components/AgentInterface.d.ts +39 -0
  9. package/dist/components/AgentInterface.d.ts.map +1 -0
  10. package/dist/components/AgentInterface.js +384 -0
  11. package/dist/components/AgentInterface.js.map +1 -0
  12. package/dist/components/AttachmentTile.d.ts +12 -0
  13. package/dist/components/AttachmentTile.d.ts.map +1 -0
  14. package/dist/components/AttachmentTile.js +110 -0
  15. package/dist/components/AttachmentTile.js.map +1 -0
  16. package/dist/components/ConsoleBlock.d.ts +12 -0
  17. package/dist/components/ConsoleBlock.d.ts.map +1 -0
  18. package/dist/components/ConsoleBlock.js +81 -0
  19. package/dist/components/ConsoleBlock.js.map +1 -0
  20. package/dist/components/CustomProviderCard.d.ts +17 -0
  21. package/dist/components/CustomProviderCard.d.ts.map +1 -0
  22. package/dist/components/CustomProviderCard.js +112 -0
  23. package/dist/components/CustomProviderCard.js.map +1 -0
  24. package/dist/components/ExpandableSection.d.ts +15 -0
  25. package/dist/components/ExpandableSection.d.ts.map +1 -0
  26. package/dist/components/ExpandableSection.js +61 -0
  27. package/dist/components/ExpandableSection.js.map +1 -0
  28. package/dist/components/Input.d.ts +26 -0
  29. package/dist/components/Input.d.ts.map +1 -0
  30. package/dist/components/Input.js +57 -0
  31. package/dist/components/Input.js.map +1 -0
  32. package/dist/components/MessageEditor.d.ts +44 -0
  33. package/dist/components/MessageEditor.d.ts.map +1 -0
  34. package/dist/components/MessageEditor.js +418 -0
  35. package/dist/components/MessageEditor.js.map +1 -0
  36. package/dist/components/MessageList.d.ts +14 -0
  37. package/dist/components/MessageList.d.ts.map +1 -0
  38. package/dist/components/MessageList.js +104 -0
  39. package/dist/components/MessageList.js.map +1 -0
  40. package/dist/components/Messages.d.ts +95 -0
  41. package/dist/components/Messages.d.ts.map +1 -0
  42. package/dist/components/Messages.js +363 -0
  43. package/dist/components/Messages.js.map +1 -0
  44. package/dist/components/ProviderKeyInput.d.ts +16 -0
  45. package/dist/components/ProviderKeyInput.d.ts.map +1 -0
  46. package/dist/components/ProviderKeyInput.js +168 -0
  47. package/dist/components/ProviderKeyInput.js.map +1 -0
  48. package/dist/components/SandboxedIframe.d.ts +85 -0
  49. package/dist/components/SandboxedIframe.d.ts.map +1 -0
  50. package/dist/components/SandboxedIframe.js +518 -0
  51. package/dist/components/SandboxedIframe.js.map +1 -0
  52. package/dist/components/StreamingMessageContainer.d.ts +19 -0
  53. package/dist/components/StreamingMessageContainer.d.ts.map +1 -0
  54. package/dist/components/StreamingMessageContainer.js +117 -0
  55. package/dist/components/StreamingMessageContainer.js.map +1 -0
  56. package/dist/components/ThinkingBlock.d.ts +11 -0
  57. package/dist/components/ThinkingBlock.d.ts.map +1 -0
  58. package/dist/components/ThinkingBlock.js +56 -0
  59. package/dist/components/ThinkingBlock.js.map +1 -0
  60. package/dist/components/message-renderer-registry.d.ts +12 -0
  61. package/dist/components/message-renderer-registry.d.ts.map +1 -0
  62. package/dist/components/message-renderer-registry.js +12 -0
  63. package/dist/components/message-renderer-registry.js.map +1 -0
  64. package/dist/components/sandbox/ArtifactsRuntimeProvider.d.ts +35 -0
  65. package/dist/components/sandbox/ArtifactsRuntimeProvider.d.ts.map +1 -0
  66. package/dist/components/sandbox/ArtifactsRuntimeProvider.js +192 -0
  67. package/dist/components/sandbox/ArtifactsRuntimeProvider.js.map +1 -0
  68. package/dist/components/sandbox/AttachmentsRuntimeProvider.d.ts +17 -0
  69. package/dist/components/sandbox/AttachmentsRuntimeProvider.d.ts.map +1 -0
  70. package/dist/components/sandbox/AttachmentsRuntimeProvider.js +65 -0
  71. package/dist/components/sandbox/AttachmentsRuntimeProvider.js.map +1 -0
  72. package/dist/components/sandbox/ConsoleRuntimeProvider.d.ts +42 -0
  73. package/dist/components/sandbox/ConsoleRuntimeProvider.d.ts.map +1 -0
  74. package/dist/components/sandbox/ConsoleRuntimeProvider.js +159 -0
  75. package/dist/components/sandbox/ConsoleRuntimeProvider.js.map +1 -0
  76. package/dist/components/sandbox/FileDownloadRuntimeProvider.d.ts +30 -0
  77. package/dist/components/sandbox/FileDownloadRuntimeProvider.d.ts.map +1 -0
  78. package/dist/components/sandbox/FileDownloadRuntimeProvider.js +95 -0
  79. package/dist/components/sandbox/FileDownloadRuntimeProvider.js.map +1 -0
  80. package/dist/components/sandbox/RuntimeMessageBridge.d.ts +19 -0
  81. package/dist/components/sandbox/RuntimeMessageBridge.d.ts.map +1 -0
  82. package/dist/components/sandbox/RuntimeMessageBridge.js +74 -0
  83. package/dist/components/sandbox/RuntimeMessageBridge.js.map +1 -0
  84. package/dist/components/sandbox/RuntimeMessageRouter.d.ts +65 -0
  85. package/dist/components/sandbox/RuntimeMessageRouter.d.ts.map +1 -0
  86. package/dist/components/sandbox/RuntimeMessageRouter.js +166 -0
  87. package/dist/components/sandbox/RuntimeMessageRouter.js.map +1 -0
  88. package/dist/components/sandbox/SandboxRuntimeProvider.d.ts +48 -0
  89. package/dist/components/sandbox/SandboxRuntimeProvider.d.ts.map +1 -0
  90. package/dist/components/sandbox/SandboxRuntimeProvider.js +2 -0
  91. package/dist/components/sandbox/SandboxRuntimeProvider.js.map +1 -0
  92. package/dist/dialogs/ApiKeyPromptDialog.d.ts +15 -0
  93. package/dist/dialogs/ApiKeyPromptDialog.d.ts.map +1 -0
  94. package/dist/dialogs/ApiKeyPromptDialog.js +77 -0
  95. package/dist/dialogs/ApiKeyPromptDialog.js.map +1 -0
  96. package/dist/dialogs/AttachmentOverlay.d.ts +32 -0
  97. package/dist/dialogs/AttachmentOverlay.d.ts.map +1 -0
  98. package/dist/dialogs/AttachmentOverlay.js +576 -0
  99. package/dist/dialogs/AttachmentOverlay.js.map +1 -0
  100. package/dist/dialogs/CustomProviderDialog.d.ts +25 -0
  101. package/dist/dialogs/CustomProviderDialog.d.ts.map +1 -0
  102. package/dist/dialogs/CustomProviderDialog.js +270 -0
  103. package/dist/dialogs/CustomProviderDialog.js.map +1 -0
  104. package/dist/dialogs/ModelSelector.d.ts +27 -0
  105. package/dist/dialogs/ModelSelector.d.ts.map +1 -0
  106. package/dist/dialogs/ModelSelector.js +320 -0
  107. package/dist/dialogs/ModelSelector.js.map +1 -0
  108. package/dist/dialogs/PersistentStorageDialog.d.ts +17 -0
  109. package/dist/dialogs/PersistentStorageDialog.d.ts.map +1 -0
  110. package/dist/dialogs/PersistentStorageDialog.js +144 -0
  111. package/dist/dialogs/PersistentStorageDialog.js.map +1 -0
  112. package/dist/dialogs/ProvidersModelsTab.d.ts +20 -0
  113. package/dist/dialogs/ProvidersModelsTab.d.ts.map +1 -0
  114. package/dist/dialogs/ProvidersModelsTab.js +188 -0
  115. package/dist/dialogs/ProvidersModelsTab.js.map +1 -0
  116. package/dist/dialogs/SessionListDialog.d.ts +19 -0
  117. package/dist/dialogs/SessionListDialog.d.ts.map +1 -0
  118. package/dist/dialogs/SessionListDialog.js +152 -0
  119. package/dist/dialogs/SessionListDialog.js.map +1 -0
  120. package/dist/dialogs/SettingsDialog.d.ts +30 -0
  121. package/dist/dialogs/SettingsDialog.d.ts.map +1 -0
  122. package/dist/dialogs/SettingsDialog.js +222 -0
  123. package/dist/dialogs/SettingsDialog.js.map +1 -0
  124. package/dist/index.d.ts +67 -0
  125. package/dist/index.d.ts.map +1 -0
  126. package/dist/index.js +70 -0
  127. package/dist/index.js.map +1 -0
  128. package/dist/prompts/prompts.d.ts +11 -0
  129. package/dist/prompts/prompts.d.ts.map +1 -0
  130. package/dist/prompts/prompts.js +272 -0
  131. package/dist/prompts/prompts.js.map +1 -0
  132. package/dist/storage/app-storage.d.ts +33 -0
  133. package/dist/storage/app-storage.d.ts.map +1 -0
  134. package/dist/storage/app-storage.js +43 -0
  135. package/dist/storage/app-storage.js.map +1 -0
  136. package/dist/storage/backends/indexeddb-storage-backend.d.ts +27 -0
  137. package/dist/storage/backends/indexeddb-storage-backend.d.ts.map +1 -0
  138. package/dist/storage/backends/indexeddb-storage-backend.js +167 -0
  139. package/dist/storage/backends/indexeddb-storage-backend.js.map +1 -0
  140. package/dist/storage/store.d.ts +23 -0
  141. package/dist/storage/store.d.ts.map +1 -0
  142. package/dist/storage/store.js +24 -0
  143. package/dist/storage/store.js.map +1 -0
  144. package/dist/storage/stores/custom-providers-store.d.ts +25 -0
  145. package/dist/storage/stores/custom-providers-store.d.ts.map +1 -0
  146. package/dist/storage/stores/custom-providers-store.js +35 -0
  147. package/dist/storage/stores/custom-providers-store.js.map +1 -0
  148. package/dist/storage/stores/provider-keys-store.d.ts +14 -0
  149. package/dist/storage/stores/provider-keys-store.d.ts.map +1 -0
  150. package/dist/storage/stores/provider-keys-store.js +27 -0
  151. package/dist/storage/stores/provider-keys-store.js.map +1 -0
  152. package/dist/storage/stores/sessions-store.d.ts +32 -0
  153. package/dist/storage/stores/sessions-store.d.ts.map +1 -0
  154. package/dist/storage/stores/sessions-store.js +113 -0
  155. package/dist/storage/stores/sessions-store.js.map +1 -0
  156. package/dist/storage/stores/settings-store.d.ts +14 -0
  157. package/dist/storage/stores/settings-store.d.ts.map +1 -0
  158. package/dist/storage/stores/settings-store.js +28 -0
  159. package/dist/storage/stores/settings-store.js.map +1 -0
  160. package/dist/storage/types.d.ts +176 -0
  161. package/dist/storage/types.d.ts.map +1 -0
  162. package/dist/storage/types.js +2 -0
  163. package/dist/storage/types.js.map +1 -0
  164. package/dist/tools/artifacts/ArtifactElement.d.ts +9 -0
  165. package/dist/tools/artifacts/ArtifactElement.d.ts.map +1 -0
  166. package/dist/tools/artifacts/ArtifactElement.js +8 -0
  167. package/dist/tools/artifacts/ArtifactElement.js.map +1 -0
  168. package/dist/tools/artifacts/ArtifactPill.d.ts +4 -0
  169. package/dist/tools/artifacts/ArtifactPill.d.ts.map +1 -0
  170. package/dist/tools/artifacts/ArtifactPill.js +23 -0
  171. package/dist/tools/artifacts/ArtifactPill.js.map +1 -0
  172. package/dist/tools/artifacts/Console.d.ts +18 -0
  173. package/dist/tools/artifacts/Console.d.ts.map +1 -0
  174. package/dist/tools/artifacts/Console.js +92 -0
  175. package/dist/tools/artifacts/Console.js.map +1 -0
  176. package/dist/tools/artifacts/DocxArtifact.d.ts +22 -0
  177. package/dist/tools/artifacts/DocxArtifact.d.ts.map +1 -0
  178. package/dist/tools/artifacts/DocxArtifact.js +205 -0
  179. package/dist/tools/artifacts/DocxArtifact.js.map +1 -0
  180. package/dist/tools/artifacts/ExcelArtifact.d.ts +24 -0
  181. package/dist/tools/artifacts/ExcelArtifact.d.ts.map +1 -0
  182. package/dist/tools/artifacts/ExcelArtifact.js +213 -0
  183. package/dist/tools/artifacts/ExcelArtifact.js.map +1 -0
  184. package/dist/tools/artifacts/GenericArtifact.d.ts +19 -0
  185. package/dist/tools/artifacts/GenericArtifact.d.ts.map +1 -0
  186. package/dist/tools/artifacts/GenericArtifact.js +114 -0
  187. package/dist/tools/artifacts/GenericArtifact.js.map +1 -0
  188. package/dist/tools/artifacts/HtmlArtifact.d.ts +27 -0
  189. package/dist/tools/artifacts/HtmlArtifact.d.ts.map +1 -0
  190. package/dist/tools/artifacts/HtmlArtifact.js +187 -0
  191. package/dist/tools/artifacts/HtmlArtifact.js.map +1 -0
  192. package/dist/tools/artifacts/ImageArtifact.d.ts +20 -0
  193. package/dist/tools/artifacts/ImageArtifact.d.ts.map +1 -0
  194. package/dist/tools/artifacts/ImageArtifact.js +117 -0
  195. package/dist/tools/artifacts/ImageArtifact.js.map +1 -0
  196. package/dist/tools/artifacts/MarkdownArtifact.d.ts +19 -0
  197. package/dist/tools/artifacts/MarkdownArtifact.d.ts.map +1 -0
  198. package/dist/tools/artifacts/MarkdownArtifact.js +79 -0
  199. package/dist/tools/artifacts/MarkdownArtifact.js.map +1 -0
  200. package/dist/tools/artifacts/PdfArtifact.d.ts +25 -0
  201. package/dist/tools/artifacts/PdfArtifact.d.ts.map +1 -0
  202. package/dist/tools/artifacts/PdfArtifact.js +181 -0
  203. package/dist/tools/artifacts/PdfArtifact.js.map +1 -0
  204. package/dist/tools/artifacts/SvgArtifact.d.ts +18 -0
  205. package/dist/tools/artifacts/SvgArtifact.d.ts.map +1 -0
  206. package/dist/tools/artifacts/SvgArtifact.js +75 -0
  207. package/dist/tools/artifacts/SvgArtifact.js.map +1 -0
  208. package/dist/tools/artifacts/TextArtifact.d.ts +19 -0
  209. package/dist/tools/artifacts/TextArtifact.d.ts.map +1 -0
  210. package/dist/tools/artifacts/TextArtifact.js +141 -0
  211. package/dist/tools/artifacts/TextArtifact.js.map +1 -0
  212. package/dist/tools/artifacts/artifacts-tool-renderer.d.ts +11 -0
  213. package/dist/tools/artifacts/artifacts-tool-renderer.d.ts.map +1 -0
  214. package/dist/tools/artifacts/artifacts-tool-renderer.js +273 -0
  215. package/dist/tools/artifacts/artifacts-tool-renderer.js.map +1 -0
  216. package/dist/tools/artifacts/artifacts.d.ts +63 -0
  217. package/dist/tools/artifacts/artifacts.d.ts.map +1 -0
  218. package/dist/tools/artifacts/artifacts.js +664 -0
  219. package/dist/tools/artifacts/artifacts.js.map +1 -0
  220. package/dist/tools/artifacts/index.d.ts +8 -0
  221. package/dist/tools/artifacts/index.d.ts.map +1 -0
  222. package/dist/tools/artifacts/index.js +8 -0
  223. package/dist/tools/artifacts/index.js.map +1 -0
  224. package/dist/tools/extract-document.d.ts +24 -0
  225. package/dist/tools/extract-document.d.ts.map +1 -0
  226. package/dist/tools/extract-document.js +216 -0
  227. package/dist/tools/extract-document.js.map +1 -0
  228. package/dist/tools/index.d.ts +16 -0
  229. package/dist/tools/index.d.ts.map +1 -0
  230. package/dist/tools/index.js +33 -0
  231. package/dist/tools/index.js.map +1 -0
  232. package/dist/tools/javascript-repl.d.ts +44 -0
  233. package/dist/tools/javascript-repl.d.ts.map +1 -0
  234. package/dist/tools/javascript-repl.js +224 -0
  235. package/dist/tools/javascript-repl.js.map +1 -0
  236. package/dist/tools/renderer-registry.d.ts +23 -0
  237. package/dist/tools/renderer-registry.d.ts.map +1 -0
  238. package/dist/tools/renderer-registry.js +107 -0
  239. package/dist/tools/renderer-registry.js.map +1 -0
  240. package/dist/tools/renderers/BashRenderer.d.ts +10 -0
  241. package/dist/tools/renderers/BashRenderer.d.ts.map +1 -0
  242. package/dist/tools/renderers/BashRenderer.js +42 -0
  243. package/dist/tools/renderers/BashRenderer.js.map +1 -0
  244. package/dist/tools/renderers/CalculateRenderer.d.ts +10 -0
  245. package/dist/tools/renderers/CalculateRenderer.d.ts.map +1 -0
  246. package/dist/tools/renderers/CalculateRenderer.js +45 -0
  247. package/dist/tools/renderers/CalculateRenderer.js.map +1 -0
  248. package/dist/tools/renderers/DefaultRenderer.d.ts +6 -0
  249. package/dist/tools/renderers/DefaultRenderer.d.ts.map +1 -0
  250. package/dist/tools/renderers/DefaultRenderer.js +94 -0
  251. package/dist/tools/renderers/DefaultRenderer.js.map +1 -0
  252. package/dist/tools/renderers/GetCurrentTimeRenderer.d.ts +10 -0
  253. package/dist/tools/renderers/GetCurrentTimeRenderer.d.ts.map +1 -0
  254. package/dist/tools/renderers/GetCurrentTimeRenderer.js +72 -0
  255. package/dist/tools/renderers/GetCurrentTimeRenderer.js.map +1 -0
  256. package/dist/tools/types.d.ts +10 -0
  257. package/dist/tools/types.d.ts.map +1 -0
  258. package/dist/tools/types.js +2 -0
  259. package/dist/tools/types.js.map +1 -0
  260. package/dist/utils/attachment-utils.d.ts +19 -0
  261. package/dist/utils/attachment-utils.d.ts.map +1 -0
  262. package/dist/utils/attachment-utils.js +415 -0
  263. package/dist/utils/attachment-utils.js.map +1 -0
  264. package/dist/utils/auth-token.d.ts +3 -0
  265. package/dist/utils/auth-token.d.ts.map +1 -0
  266. package/dist/utils/auth-token.js +19 -0
  267. package/dist/utils/auth-token.js.map +1 -0
  268. package/dist/utils/format.d.ts +6 -0
  269. package/dist/utils/format.d.ts.map +1 -0
  270. package/dist/utils/format.js +47 -0
  271. package/dist/utils/format.js.map +1 -0
  272. package/dist/utils/i18n.d.ts +636 -0
  273. package/dist/utils/i18n.d.ts.map +1 -0
  274. package/dist/utils/i18n.js +418 -0
  275. package/dist/utils/i18n.js.map +1 -0
  276. package/dist/utils/model-discovery.d.ts +38 -0
  277. package/dist/utils/model-discovery.d.ts.map +1 -0
  278. package/dist/utils/model-discovery.js +243 -0
  279. package/dist/utils/model-discovery.js.map +1 -0
  280. package/dist/utils/proxy-utils.d.ts +45 -0
  281. package/dist/utils/proxy-utils.d.ts.map +1 -0
  282. package/dist/utils/proxy-utils.js +116 -0
  283. package/dist/utils/proxy-utils.js.map +1 -0
  284. package/dist/utils/test-sessions.d.ts +359 -0
  285. package/dist/utils/test-sessions.d.ts.map +1 -0
  286. package/dist/utils/test-sessions.js +2325 -0
  287. package/dist/utils/test-sessions.js.map +1 -0
  288. package/example/README.md +61 -0
  289. package/example/index.html +13 -0
  290. package/example/package.json +25 -0
  291. package/example/src/app.css +1 -0
  292. package/example/src/custom-messages.ts +99 -0
  293. package/example/src/env.d.ts +1 -0
  294. package/example/src/main.ts +421 -0
  295. package/example/tsconfig.json +23 -0
  296. package/example/vite.config.ts +6 -0
  297. package/package.json +53 -0
  298. package/scripts/count-prompt-tokens.ts +88 -0
  299. package/src/ChatPanel.ts +207 -0
  300. package/src/app.css +68 -0
  301. package/src/components/AgentInterface.ts +388 -0
  302. package/src/components/AttachmentTile.ts +107 -0
  303. package/src/components/ConsoleBlock.ts +72 -0
  304. package/src/components/CustomProviderCard.ts +100 -0
  305. package/src/components/ExpandableSection.ts +46 -0
  306. package/src/components/Input.ts +113 -0
  307. package/src/components/MessageEditor.ts +400 -0
  308. package/src/components/MessageList.ts +95 -0
  309. package/src/components/Messages.ts +383 -0
  310. package/src/components/ProviderKeyInput.ts +153 -0
  311. package/src/components/SandboxedIframe.ts +626 -0
  312. package/src/components/StreamingMessageContainer.ts +103 -0
  313. package/src/components/ThinkingBlock.ts +43 -0
  314. package/src/components/message-renderer-registry.ts +28 -0
  315. package/src/components/sandbox/ArtifactsRuntimeProvider.ts +219 -0
  316. package/src/components/sandbox/AttachmentsRuntimeProvider.ts +66 -0
  317. package/src/components/sandbox/ConsoleRuntimeProvider.ts +186 -0
  318. package/src/components/sandbox/FileDownloadRuntimeProvider.ts +110 -0
  319. package/src/components/sandbox/RuntimeMessageBridge.ts +82 -0
  320. package/src/components/sandbox/RuntimeMessageRouter.ts +216 -0
  321. package/src/components/sandbox/SandboxRuntimeProvider.ts +52 -0
  322. package/src/dialogs/ApiKeyPromptDialog.ts +75 -0
  323. package/src/dialogs/AttachmentOverlay.ts +636 -0
  324. package/src/dialogs/CustomProviderDialog.ts +274 -0
  325. package/src/dialogs/ModelSelector.ts +313 -0
  326. package/src/dialogs/PersistentStorageDialog.ts +144 -0
  327. package/src/dialogs/ProvidersModelsTab.ts +212 -0
  328. package/src/dialogs/SessionListDialog.ts +150 -0
  329. package/src/dialogs/SettingsDialog.ts +214 -0
  330. package/src/index.ts +119 -0
  331. package/src/prompts/prompts.ts +282 -0
  332. package/src/storage/app-storage.ts +60 -0
  333. package/src/storage/backends/indexeddb-storage-backend.ts +193 -0
  334. package/src/storage/store.ts +33 -0
  335. package/src/storage/stores/custom-providers-store.ts +62 -0
  336. package/src/storage/stores/provider-keys-store.ts +33 -0
  337. package/src/storage/stores/sessions-store.ts +136 -0
  338. package/src/storage/stores/settings-store.ts +34 -0
  339. package/src/storage/types.ts +206 -0
  340. package/src/tools/artifacts/ArtifactElement.ts +14 -0
  341. package/src/tools/artifacts/ArtifactPill.ts +26 -0
  342. package/src/tools/artifacts/Console.ts +93 -0
  343. package/src/tools/artifacts/DocxArtifact.ts +213 -0
  344. package/src/tools/artifacts/ExcelArtifact.ts +231 -0
  345. package/src/tools/artifacts/GenericArtifact.ts +117 -0
  346. package/src/tools/artifacts/HtmlArtifact.ts +195 -0
  347. package/src/tools/artifacts/ImageArtifact.ts +116 -0
  348. package/src/tools/artifacts/MarkdownArtifact.ts +82 -0
  349. package/src/tools/artifacts/PdfArtifact.ts +201 -0
  350. package/src/tools/artifacts/SvgArtifact.ts +78 -0
  351. package/src/tools/artifacts/TextArtifact.ts +148 -0
  352. package/src/tools/artifacts/artifacts-tool-renderer.ts +310 -0
  353. package/src/tools/artifacts/artifacts.ts +713 -0
  354. package/src/tools/artifacts/index.ts +7 -0
  355. package/src/tools/extract-document.ts +275 -0
  356. package/src/tools/index.ts +46 -0
  357. package/src/tools/javascript-repl.ts +293 -0
  358. package/src/tools/renderer-registry.ts +130 -0
  359. package/src/tools/renderers/BashRenderer.ts +52 -0
  360. package/src/tools/renderers/CalculateRenderer.ts +58 -0
  361. package/src/tools/renderers/DefaultRenderer.ts +103 -0
  362. package/src/tools/renderers/GetCurrentTimeRenderer.ts +92 -0
  363. package/src/tools/types.ts +15 -0
  364. package/src/utils/attachment-utils.ts +472 -0
  365. package/src/utils/auth-token.ts +22 -0
  366. package/src/utils/format.ts +42 -0
  367. package/src/utils/i18n.ts +653 -0
  368. package/src/utils/model-discovery.ts +277 -0
  369. package/src/utils/proxy-utils.ts +134 -0
  370. package/src/utils/test-sessions.ts +2357 -0
  371. package/tsconfig.build.json +20 -0
  372. package/tsconfig.json +7 -0
@@ -0,0 +1,400 @@
1
+ import type { Model } from "@draht/ai";
2
+ import { icon } from "@mariozechner/mini-lit";
3
+ import { Button } from "@mariozechner/mini-lit/dist/Button.js";
4
+ import { Select, type SelectOption } from "@mariozechner/mini-lit/dist/Select.js";
5
+ import { html, LitElement } from "lit";
6
+ import { customElement, property, state } from "lit/decorators.js";
7
+ import { createRef, ref } from "lit/directives/ref.js";
8
+ import { Brain, Loader2, Paperclip, Send, Sparkles, Square } from "lucide";
9
+ import { type Attachment, loadAttachment } from "../utils/attachment-utils.js";
10
+ import { i18n } from "../utils/i18n.js";
11
+ import "./AttachmentTile.js";
12
+ import type { ThinkingLevel } from "@draht/agent-core";
13
+
14
+ @customElement("message-editor")
15
+ export class MessageEditor extends LitElement {
16
+ private _value = "";
17
+ private textareaRef = createRef<HTMLTextAreaElement>();
18
+
19
+ @property()
20
+ get value() {
21
+ return this._value;
22
+ }
23
+
24
+ set value(val: string) {
25
+ const oldValue = this._value;
26
+ this._value = val;
27
+ this.requestUpdate("value", oldValue);
28
+ }
29
+
30
+ @property() isStreaming = false;
31
+ @property() currentModel?: Model<any>;
32
+ @property() thinkingLevel: ThinkingLevel = "off";
33
+ @property() showAttachmentButton = true;
34
+ @property() showModelSelector = true;
35
+ @property() showThinkingSelector = true;
36
+ @property() onInput?: (value: string) => void;
37
+ @property() onSend?: (input: string, attachments: Attachment[]) => void;
38
+ @property() onAbort?: () => void;
39
+ @property() onModelSelect?: () => void;
40
+ @property() onThinkingChange?: (level: "off" | "minimal" | "low" | "medium" | "high") => void;
41
+ @property() onFilesChange?: (files: Attachment[]) => void;
42
+ @property() attachments: Attachment[] = [];
43
+ @property() maxFiles = 10;
44
+ @property() maxFileSize = 20 * 1024 * 1024; // 20MB
45
+ @property() acceptedTypes =
46
+ "image/*,application/pdf,.docx,.pptx,.xlsx,.xls,.txt,.md,.json,.xml,.html,.css,.js,.ts,.jsx,.tsx,.yml,.yaml";
47
+
48
+ @state() processingFiles = false;
49
+ @state() isDragging = false;
50
+ private fileInputRef = createRef<HTMLInputElement>();
51
+
52
+ protected override createRenderRoot(): HTMLElement | DocumentFragment {
53
+ return this;
54
+ }
55
+
56
+ private handleTextareaInput = (e: Event) => {
57
+ const textarea = e.target as HTMLTextAreaElement;
58
+ this.value = textarea.value;
59
+ this.onInput?.(this.value);
60
+ };
61
+
62
+ private handleKeyDown = (e: KeyboardEvent) => {
63
+ if (e.key === "Enter" && !e.shiftKey) {
64
+ e.preventDefault();
65
+ if (!this.isStreaming && !this.processingFiles && (this.value.trim() || this.attachments.length > 0)) {
66
+ this.handleSend();
67
+ }
68
+ } else if (e.key === "Escape" && this.isStreaming) {
69
+ e.preventDefault();
70
+ this.onAbort?.();
71
+ }
72
+ };
73
+
74
+ private handlePaste = async (e: ClipboardEvent) => {
75
+ const items = e.clipboardData?.items;
76
+ if (!items) return;
77
+
78
+ const imageFiles: File[] = [];
79
+
80
+ // Check for image items in clipboard
81
+ for (let i = 0; i < items.length; i++) {
82
+ const item = items[i];
83
+ if (item.type.startsWith("image/")) {
84
+ const file = item.getAsFile();
85
+ if (file) {
86
+ imageFiles.push(file);
87
+ }
88
+ }
89
+ }
90
+
91
+ // If we found images, process them
92
+ if (imageFiles.length > 0) {
93
+ e.preventDefault(); // Prevent default paste behavior
94
+
95
+ if (imageFiles.length + this.attachments.length > this.maxFiles) {
96
+ alert(`Maximum ${this.maxFiles} files allowed`);
97
+ return;
98
+ }
99
+
100
+ this.processingFiles = true;
101
+ const newAttachments: Attachment[] = [];
102
+
103
+ for (const file of imageFiles) {
104
+ try {
105
+ if (file.size > this.maxFileSize) {
106
+ alert(`Image exceeds maximum size of ${Math.round(this.maxFileSize / 1024 / 1024)}MB`);
107
+ continue;
108
+ }
109
+
110
+ const attachment = await loadAttachment(file);
111
+ newAttachments.push(attachment);
112
+ } catch (error) {
113
+ console.error("Error processing pasted image:", error);
114
+ alert(`Failed to process pasted image: ${String(error)}`);
115
+ }
116
+ }
117
+
118
+ this.attachments = [...this.attachments, ...newAttachments];
119
+ this.onFilesChange?.(this.attachments);
120
+ this.processingFiles = false;
121
+ }
122
+ };
123
+
124
+ private handleSend = () => {
125
+ this.onSend?.(this.value, this.attachments);
126
+ };
127
+
128
+ private handleAttachmentClick = () => {
129
+ this.fileInputRef.value?.click();
130
+ };
131
+
132
+ private async handleFilesSelected(e: Event) {
133
+ const input = e.target as HTMLInputElement;
134
+ const files = Array.from(input.files || []);
135
+ if (files.length === 0) return;
136
+
137
+ if (files.length + this.attachments.length > this.maxFiles) {
138
+ alert(`Maximum ${this.maxFiles} files allowed`);
139
+ input.value = "";
140
+ return;
141
+ }
142
+
143
+ this.processingFiles = true;
144
+ const newAttachments: Attachment[] = [];
145
+
146
+ for (const file of files) {
147
+ try {
148
+ if (file.size > this.maxFileSize) {
149
+ alert(`${file.name} exceeds maximum size of ${Math.round(this.maxFileSize / 1024 / 1024)}MB`);
150
+ continue;
151
+ }
152
+
153
+ const attachment = await loadAttachment(file);
154
+ newAttachments.push(attachment);
155
+ } catch (error) {
156
+ console.error(`Error processing ${file.name}:`, error);
157
+ alert(`Failed to process ${file.name}: ${String(error)}`);
158
+ }
159
+ }
160
+
161
+ this.attachments = [...this.attachments, ...newAttachments];
162
+ this.onFilesChange?.(this.attachments);
163
+ this.processingFiles = false;
164
+ input.value = ""; // Reset input
165
+ }
166
+
167
+ private removeFile(fileId: string) {
168
+ this.attachments = this.attachments.filter((f) => f.id !== fileId);
169
+ this.onFilesChange?.(this.attachments);
170
+ }
171
+
172
+ private handleDragOver = (e: DragEvent) => {
173
+ e.preventDefault();
174
+ e.stopPropagation();
175
+ if (!this.isDragging) {
176
+ this.isDragging = true;
177
+ }
178
+ };
179
+
180
+ private handleDragLeave = (e: DragEvent) => {
181
+ e.preventDefault();
182
+ e.stopPropagation();
183
+ // Only set isDragging to false if we're leaving the entire component
184
+ const rect = (e.currentTarget as HTMLElement).getBoundingClientRect();
185
+ const x = e.clientX;
186
+ const y = e.clientY;
187
+ if (x <= rect.left || x >= rect.right || y <= rect.top || y >= rect.bottom) {
188
+ this.isDragging = false;
189
+ }
190
+ };
191
+
192
+ private handleDrop = async (e: DragEvent) => {
193
+ e.preventDefault();
194
+ e.stopPropagation();
195
+ this.isDragging = false;
196
+
197
+ const files = Array.from(e.dataTransfer?.files || []);
198
+ if (files.length === 0) return;
199
+
200
+ if (files.length + this.attachments.length > this.maxFiles) {
201
+ alert(`Maximum ${this.maxFiles} files allowed`);
202
+ return;
203
+ }
204
+
205
+ this.processingFiles = true;
206
+ const newAttachments: Attachment[] = [];
207
+
208
+ for (const file of files) {
209
+ try {
210
+ if (file.size > this.maxFileSize) {
211
+ alert(`${file.name} exceeds maximum size of ${Math.round(this.maxFileSize / 1024 / 1024)}MB`);
212
+ continue;
213
+ }
214
+
215
+ const attachment = await loadAttachment(file);
216
+ newAttachments.push(attachment);
217
+ } catch (error) {
218
+ console.error(`Error processing ${file.name}:`, error);
219
+ alert(`Failed to process ${file.name}: ${String(error)}`);
220
+ }
221
+ }
222
+
223
+ this.attachments = [...this.attachments, ...newAttachments];
224
+ this.onFilesChange?.(this.attachments);
225
+ this.processingFiles = false;
226
+ };
227
+
228
+ override firstUpdated() {
229
+ const textarea = this.textareaRef.value;
230
+ if (textarea) {
231
+ textarea.focus();
232
+ }
233
+ }
234
+
235
+ override render() {
236
+ // Check if current model supports thinking/reasoning
237
+ const model = this.currentModel;
238
+ const supportsThinking = model?.reasoning === true; // Models with reasoning:true support thinking
239
+
240
+ return html`
241
+ <div
242
+ class="bg-card rounded-xl border shadow-sm relative ${this.isDragging ? "border-primary border-2 bg-primary/5" : "border-border"}"
243
+ @dragover=${this.handleDragOver}
244
+ @dragleave=${this.handleDragLeave}
245
+ @drop=${this.handleDrop}
246
+ >
247
+ <!-- Drag overlay -->
248
+ ${
249
+ this.isDragging
250
+ ? html`
251
+ <div class="absolute inset-0 bg-primary/10 rounded-xl pointer-events-none z-10 flex items-center justify-center">
252
+ <div class="text-primary font-medium">${i18n("Drop files here")}</div>
253
+ </div>
254
+ `
255
+ : ""
256
+ }
257
+
258
+ <!-- Attachments -->
259
+ ${
260
+ this.attachments.length > 0
261
+ ? html`
262
+ <div class="px-4 pt-3 pb-2 flex flex-wrap gap-2">
263
+ ${this.attachments.map(
264
+ (attachment) => html`
265
+ <attachment-tile
266
+ .attachment=${attachment}
267
+ .showDelete=${true}
268
+ .onDelete=${() => this.removeFile(attachment.id)}
269
+ ></attachment-tile>
270
+ `,
271
+ )}
272
+ </div>
273
+ `
274
+ : ""
275
+ }
276
+
277
+ <textarea
278
+ class="w-full bg-transparent p-4 text-foreground placeholder-muted-foreground outline-none resize-none overflow-y-auto"
279
+ placeholder=${i18n("Type a message...")}
280
+ rows="1"
281
+ style="max-height: 200px; field-sizing: content; min-height: 1lh; height: auto;"
282
+ .value=${this.value}
283
+ @input=${this.handleTextareaInput}
284
+ @keydown=${this.handleKeyDown}
285
+ @paste=${this.handlePaste}
286
+ ${ref(this.textareaRef)}
287
+ ></textarea>
288
+
289
+ <!-- Hidden file input -->
290
+ <input
291
+ type="file"
292
+ ${ref(this.fileInputRef)}
293
+ @change=${this.handleFilesSelected}
294
+ accept=${this.acceptedTypes}
295
+ multiple
296
+ style="display: none;"
297
+ />
298
+
299
+ <!-- Button Row -->
300
+ <div class="px-2 pb-2 flex items-center justify-between">
301
+ <!-- Left side - attachment and thinking selector -->
302
+ <div class="flex gap-2 items-center">
303
+ ${
304
+ this.showAttachmentButton
305
+ ? this.processingFiles
306
+ ? html`
307
+ <div class="h-8 w-8 flex items-center justify-center">
308
+ ${icon(Loader2, "sm", "animate-spin text-muted-foreground")}
309
+ </div>
310
+ `
311
+ : html`
312
+ ${Button({
313
+ variant: "ghost",
314
+ size: "icon",
315
+ className: "h-8 w-8",
316
+ onClick: this.handleAttachmentClick,
317
+ children: icon(Paperclip, "sm"),
318
+ })}
319
+ `
320
+ : ""
321
+ }
322
+ ${
323
+ supportsThinking && this.showThinkingSelector
324
+ ? html`
325
+ ${Select({
326
+ value: this.thinkingLevel,
327
+ placeholder: i18n("Off"),
328
+ options: [
329
+ { value: "off", label: i18n("Off"), icon: icon(Brain, "sm") },
330
+ { value: "minimal", label: i18n("Minimal"), icon: icon(Brain, "sm") },
331
+ { value: "low", label: i18n("Low"), icon: icon(Brain, "sm") },
332
+ { value: "medium", label: i18n("Medium"), icon: icon(Brain, "sm") },
333
+ { value: "high", label: i18n("High"), icon: icon(Brain, "sm") },
334
+ ] as SelectOption[],
335
+ onChange: (value: string) => {
336
+ this.onThinkingChange?.(value as "off" | "minimal" | "low" | "medium" | "high");
337
+ },
338
+ width: "80px",
339
+ size: "sm",
340
+ variant: "ghost",
341
+ fitContent: true,
342
+ })}
343
+ `
344
+ : ""
345
+ }
346
+ </div>
347
+
348
+ <!-- Model selector and send on the right -->
349
+ <div class="flex gap-2 items-center">
350
+ ${
351
+ this.showModelSelector && this.currentModel
352
+ ? html`
353
+ ${Button({
354
+ variant: "ghost",
355
+ size: "sm",
356
+ onClick: () => {
357
+ // Focus textarea before opening model selector so focus returns there
358
+ this.textareaRef.value?.focus();
359
+ // Wait for next frame to ensure focus takes effect before dialog captures it
360
+ requestAnimationFrame(() => {
361
+ this.onModelSelect?.();
362
+ });
363
+ },
364
+ children: html`
365
+ ${icon(Sparkles, "sm")}
366
+ <span class="ml-1">${this.currentModel.id}</span>
367
+ `,
368
+ className: "h-8 text-xs truncate",
369
+ })}
370
+ `
371
+ : ""
372
+ }
373
+ ${
374
+ this.isStreaming
375
+ ? html`
376
+ ${Button({
377
+ variant: "ghost",
378
+ size: "icon",
379
+ onClick: this.onAbort,
380
+ children: icon(Square, "sm"),
381
+ className: "h-8 w-8",
382
+ })}
383
+ `
384
+ : html`
385
+ ${Button({
386
+ variant: "ghost",
387
+ size: "icon",
388
+ onClick: this.handleSend,
389
+ disabled: (!this.value.trim() && this.attachments.length === 0) || this.processingFiles,
390
+ children: html`<div style="transform: rotate(-45deg)">${icon(Send, "sm")}</div>`,
391
+ className: "h-8 w-8",
392
+ })}
393
+ `
394
+ }
395
+ </div>
396
+ </div>
397
+ </div>
398
+ `;
399
+ }
400
+ }
@@ -0,0 +1,95 @@
1
+ import type { AgentMessage, AgentTool } from "@draht/agent-core";
2
+ import type { AssistantMessage as AssistantMessageType, ToolResultMessage as ToolResultMessageType } from "@draht/ai";
3
+ import { html, LitElement, type TemplateResult } from "lit";
4
+ import { property } from "lit/decorators.js";
5
+ import { repeat } from "lit/directives/repeat.js";
6
+ import { renderMessage } from "./message-renderer-registry.js";
7
+
8
+ export class MessageList extends LitElement {
9
+ @property({ type: Array }) messages: AgentMessage[] = [];
10
+ @property({ type: Array }) tools: AgentTool[] = [];
11
+ @property({ type: Object }) pendingToolCalls?: Set<string>;
12
+ @property({ type: Boolean }) isStreaming: boolean = false;
13
+ @property({ attribute: false }) onCostClick?: () => void;
14
+
15
+ protected override createRenderRoot(): HTMLElement | DocumentFragment {
16
+ return this;
17
+ }
18
+
19
+ override connectedCallback(): void {
20
+ super.connectedCallback();
21
+ this.style.display = "block";
22
+ }
23
+
24
+ private buildRenderItems() {
25
+ // Map tool results by call id for quick lookup
26
+ const resultByCallId = new Map<string, ToolResultMessageType>();
27
+ for (const message of this.messages) {
28
+ if (message.role === "toolResult") {
29
+ resultByCallId.set(message.toolCallId, message);
30
+ }
31
+ }
32
+
33
+ const items: Array<{ key: string; template: TemplateResult }> = [];
34
+ let index = 0;
35
+ for (const msg of this.messages) {
36
+ // Skip artifact messages - they're for session persistence only, not UI display
37
+ if (msg.role === "artifact") {
38
+ continue;
39
+ }
40
+
41
+ // Try custom renderer first
42
+ const customTemplate = renderMessage(msg);
43
+ if (customTemplate) {
44
+ items.push({ key: `msg:${index}`, template: customTemplate });
45
+ index++;
46
+ continue;
47
+ }
48
+
49
+ // Fall back to built-in renderers
50
+ if (msg.role === "user" || msg.role === "user-with-attachments") {
51
+ items.push({
52
+ key: `msg:${index}`,
53
+ template: html`<user-message .message=${msg}></user-message>`,
54
+ });
55
+ index++;
56
+ } else if (msg.role === "assistant") {
57
+ const amsg = msg as AssistantMessageType;
58
+ items.push({
59
+ key: `msg:${index}`,
60
+ template: html`<assistant-message
61
+ .message=${amsg}
62
+ .tools=${this.tools}
63
+ .isStreaming=${false}
64
+ .pendingToolCalls=${this.pendingToolCalls}
65
+ .toolResultsById=${resultByCallId}
66
+ .hideToolCalls=${false}
67
+ .hidePendingToolCalls=${this.isStreaming}
68
+ .onCostClick=${this.onCostClick}
69
+ ></assistant-message>`,
70
+ });
71
+ index++;
72
+ } else {
73
+ // Skip standalone toolResult messages; they are rendered via paired tool-message above
74
+ // Skip unknown roles
75
+ }
76
+ }
77
+ return items;
78
+ }
79
+
80
+ override render() {
81
+ const items = this.buildRenderItems();
82
+ return html`<div class="flex flex-col gap-3">
83
+ ${repeat(
84
+ items,
85
+ (it) => it.key,
86
+ (it) => it.template,
87
+ )}
88
+ </div>`;
89
+ }
90
+ }
91
+
92
+ // Register custom element
93
+ if (!customElements.get("message-list")) {
94
+ customElements.define("message-list", MessageList);
95
+ }