@ouhuang/pi-web-ui 0.65.1

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 (371) hide show
  1. package/CHANGELOG.md +349 -0
  2. package/README.md +601 -0
  3. package/dist/ChatPanel.d.ts +29 -0
  4. package/dist/ChatPanel.d.ts.map +1 -0
  5. package/dist/ChatPanel.js +194 -0
  6. package/dist/ChatPanel.js.map +1 -0
  7. package/dist/app.css +2 -0
  8. package/dist/components/AgentInterface.d.ts +40 -0
  9. package/dist/components/AgentInterface.d.ts.map +1 -0
  10. package/dist/components/AgentInterface.js +388 -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 +111 -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 +84 -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 +110 -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 +63 -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 +416 -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 +105 -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 +362 -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 +171 -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 +511 -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 +58 -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 +38 -0
  65. package/dist/components/sandbox/ArtifactsRuntimeProvider.d.ts.map +1 -0
  66. package/dist/components/sandbox/ArtifactsRuntimeProvider.js +189 -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 +64 -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 +161 -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 +97 -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 +168 -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 +79 -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 +28 -0
  105. package/dist/dialogs/ModelSelector.d.ts.map +1 -0
  106. package/dist/dialogs/ModelSelector.js +368 -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 +147 -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 +191 -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 +154 -0
  119. package/dist/dialogs/SessionListDialog.js.map +1 -0
  120. package/dist/dialogs/SettingsDialog.d.ts +31 -0
  121. package/dist/dialogs/SettingsDialog.d.ts.map +1 -0
  122. package/dist/dialogs/SettingsDialog.js +231 -0
  123. package/dist/dialogs/SettingsDialog.js.map +1 -0
  124. package/dist/index.d.ts +68 -0
  125. package/dist/index.d.ts.map +1 -0
  126. package/dist/index.js +71 -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 +38 -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 +166 -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 +26 -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 +11 -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 +95 -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 +208 -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 +216 -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 +117 -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 +189 -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 +120 -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 +82 -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 +184 -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 +78 -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 +144 -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 +272 -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 +659 -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 +120 -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/main.ts +421 -0
  294. package/example/tsconfig.json +23 -0
  295. package/example/vite.config.ts +6 -0
  296. package/package.json +51 -0
  297. package/scripts/count-prompt-tokens.ts +88 -0
  298. package/src/ChatPanel.ts +209 -0
  299. package/src/app.css +68 -0
  300. package/src/components/AgentInterface.ts +403 -0
  301. package/src/components/AttachmentTile.ts +107 -0
  302. package/src/components/ConsoleBlock.ts +72 -0
  303. package/src/components/CustomProviderCard.ts +100 -0
  304. package/src/components/ExpandableSection.ts +46 -0
  305. package/src/components/Input.ts +113 -0
  306. package/src/components/MessageEditor.ts +402 -0
  307. package/src/components/MessageList.ts +98 -0
  308. package/src/components/Messages.ts +384 -0
  309. package/src/components/ProviderKeyInput.ts +153 -0
  310. package/src/components/SandboxedIframe.ts +626 -0
  311. package/src/components/StreamingMessageContainer.ts +103 -0
  312. package/src/components/ThinkingBlock.ts +43 -0
  313. package/src/components/message-renderer-registry.ts +28 -0
  314. package/src/components/sandbox/ArtifactsRuntimeProvider.ts +220 -0
  315. package/src/components/sandbox/AttachmentsRuntimeProvider.ts +66 -0
  316. package/src/components/sandbox/ConsoleRuntimeProvider.ts +186 -0
  317. package/src/components/sandbox/FileDownloadRuntimeProvider.ts +110 -0
  318. package/src/components/sandbox/RuntimeMessageBridge.ts +82 -0
  319. package/src/components/sandbox/RuntimeMessageRouter.ts +216 -0
  320. package/src/components/sandbox/SandboxRuntimeProvider.ts +52 -0
  321. package/src/dialogs/ApiKeyPromptDialog.ts +75 -0
  322. package/src/dialogs/AttachmentOverlay.ts +636 -0
  323. package/src/dialogs/CustomProviderDialog.ts +274 -0
  324. package/src/dialogs/ModelSelector.ts +367 -0
  325. package/src/dialogs/PersistentStorageDialog.ts +144 -0
  326. package/src/dialogs/ProvidersModelsTab.ts +212 -0
  327. package/src/dialogs/SessionListDialog.ts +150 -0
  328. package/src/dialogs/SettingsDialog.ts +218 -0
  329. package/src/index.ts +120 -0
  330. package/src/prompts/prompts.ts +282 -0
  331. package/src/storage/app-storage.ts +60 -0
  332. package/src/storage/backends/indexeddb-storage-backend.ts +193 -0
  333. package/src/storage/store.ts +33 -0
  334. package/src/storage/stores/custom-providers-store.ts +62 -0
  335. package/src/storage/stores/provider-keys-store.ts +33 -0
  336. package/src/storage/stores/sessions-store.ts +136 -0
  337. package/src/storage/stores/settings-store.ts +34 -0
  338. package/src/storage/types.ts +206 -0
  339. package/src/tools/artifacts/ArtifactElement.ts +14 -0
  340. package/src/tools/artifacts/ArtifactPill.ts +26 -0
  341. package/src/tools/artifacts/Console.ts +93 -0
  342. package/src/tools/artifacts/DocxArtifact.ts +213 -0
  343. package/src/tools/artifacts/ExcelArtifact.ts +231 -0
  344. package/src/tools/artifacts/GenericArtifact.ts +117 -0
  345. package/src/tools/artifacts/HtmlArtifact.ts +195 -0
  346. package/src/tools/artifacts/ImageArtifact.ts +116 -0
  347. package/src/tools/artifacts/MarkdownArtifact.ts +82 -0
  348. package/src/tools/artifacts/PdfArtifact.ts +201 -0
  349. package/src/tools/artifacts/SvgArtifact.ts +78 -0
  350. package/src/tools/artifacts/TextArtifact.ts +148 -0
  351. package/src/tools/artifacts/artifacts-tool-renderer.ts +310 -0
  352. package/src/tools/artifacts/artifacts.ts +713 -0
  353. package/src/tools/artifacts/index.ts +7 -0
  354. package/src/tools/extract-document.ts +275 -0
  355. package/src/tools/index.ts +46 -0
  356. package/src/tools/javascript-repl.ts +293 -0
  357. package/src/tools/renderer-registry.ts +130 -0
  358. package/src/tools/renderers/BashRenderer.ts +52 -0
  359. package/src/tools/renderers/CalculateRenderer.ts +58 -0
  360. package/src/tools/renderers/DefaultRenderer.ts +103 -0
  361. package/src/tools/renderers/GetCurrentTimeRenderer.ts +92 -0
  362. package/src/tools/types.ts +15 -0
  363. package/src/utils/attachment-utils.ts +472 -0
  364. package/src/utils/auth-token.ts +22 -0
  365. package/src/utils/format.ts +42 -0
  366. package/src/utils/i18n.ts +653 -0
  367. package/src/utils/model-discovery.ts +277 -0
  368. package/src/utils/proxy-utils.ts +139 -0
  369. package/src/utils/test-sessions.ts +2357 -0
  370. package/tsconfig.build.json +26 -0
  371. package/tsconfig.json +13 -0
@@ -0,0 +1,402 @@
1
+ import { icon } from "@mariozechner/mini-lit";
2
+ import { Button } from "@mariozechner/mini-lit/dist/Button.js";
3
+ import { Select, type SelectOption } from "@mariozechner/mini-lit/dist/Select.js";
4
+ import type { Model } from "@ouhuang/pi-ai";
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 "@ouhuang/pi-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
+ const level = value as "off" | "minimal" | "low" | "medium" | "high";
337
+ this.thinkingLevel = level;
338
+ this.onThinkingChange?.(level);
339
+ },
340
+ width: "80px",
341
+ size: "sm",
342
+ variant: "ghost",
343
+ fitContent: true,
344
+ })}
345
+ `
346
+ : ""
347
+ }
348
+ </div>
349
+
350
+ <!-- Model selector and send on the right -->
351
+ <div class="flex gap-2 items-center">
352
+ ${
353
+ this.showModelSelector && this.currentModel
354
+ ? html`
355
+ ${Button({
356
+ variant: "ghost",
357
+ size: "sm",
358
+ onClick: () => {
359
+ // Focus textarea before opening model selector so focus returns there
360
+ this.textareaRef.value?.focus();
361
+ // Wait for next frame to ensure focus takes effect before dialog captures it
362
+ requestAnimationFrame(() => {
363
+ this.onModelSelect?.();
364
+ });
365
+ },
366
+ children: html`
367
+ ${icon(Sparkles, "sm")}
368
+ <span class="ml-1">${this.currentModel.id}</span>
369
+ `,
370
+ className: "h-8 text-xs truncate",
371
+ })}
372
+ `
373
+ : ""
374
+ }
375
+ ${
376
+ this.isStreaming
377
+ ? html`
378
+ ${Button({
379
+ variant: "ghost",
380
+ size: "icon",
381
+ onClick: this.onAbort,
382
+ children: icon(Square, "sm"),
383
+ className: "h-8 w-8",
384
+ })}
385
+ `
386
+ : html`
387
+ ${Button({
388
+ variant: "ghost",
389
+ size: "icon",
390
+ onClick: this.handleSend,
391
+ disabled: (!this.value.trim() && this.attachments.length === 0) || this.processingFiles,
392
+ children: html`<div style="transform: rotate(-45deg)">${icon(Send, "sm")}</div>`,
393
+ className: "h-8 w-8",
394
+ })}
395
+ `
396
+ }
397
+ </div>
398
+ </div>
399
+ </div>
400
+ `;
401
+ }
402
+ }
@@ -0,0 +1,98 @@
1
+ import type { AgentMessage, AgentTool } from "@ouhuang/pi-agent-core";
2
+ import type {
3
+ AssistantMessage as AssistantMessageType,
4
+ ToolResultMessage as ToolResultMessageType,
5
+ } from "@ouhuang/pi-ai";
6
+ import { html, LitElement, type TemplateResult } from "lit";
7
+ import { property } from "lit/decorators.js";
8
+ import { repeat } from "lit/directives/repeat.js";
9
+ import { renderMessage } from "./message-renderer-registry.js";
10
+
11
+ export class MessageList extends LitElement {
12
+ @property({ type: Array }) messages: AgentMessage[] = [];
13
+ @property({ type: Array }) tools: AgentTool[] = [];
14
+ @property({ type: Object }) pendingToolCalls?: ReadonlySet<string>;
15
+ @property({ type: Boolean }) isStreaming: boolean = false;
16
+ @property({ attribute: false }) onCostClick?: () => void;
17
+
18
+ protected override createRenderRoot(): HTMLElement | DocumentFragment {
19
+ return this;
20
+ }
21
+
22
+ override connectedCallback(): void {
23
+ super.connectedCallback();
24
+ this.style.display = "block";
25
+ }
26
+
27
+ private buildRenderItems() {
28
+ // Map tool results by call id for quick lookup
29
+ const resultByCallId = new Map<string, ToolResultMessageType>();
30
+ for (const message of this.messages) {
31
+ if (message.role === "toolResult") {
32
+ resultByCallId.set(message.toolCallId, message);
33
+ }
34
+ }
35
+
36
+ const items: Array<{ key: string; template: TemplateResult }> = [];
37
+ let index = 0;
38
+ for (const msg of this.messages) {
39
+ // Skip artifact messages - they're for session persistence only, not UI display
40
+ if (msg.role === "artifact") {
41
+ continue;
42
+ }
43
+
44
+ // Try custom renderer first
45
+ const customTemplate = renderMessage(msg);
46
+ if (customTemplate) {
47
+ items.push({ key: `msg:${index}`, template: customTemplate });
48
+ index++;
49
+ continue;
50
+ }
51
+
52
+ // Fall back to built-in renderers
53
+ if (msg.role === "user" || msg.role === "user-with-attachments") {
54
+ items.push({
55
+ key: `msg:${index}`,
56
+ template: html`<user-message .message=${msg}></user-message>`,
57
+ });
58
+ index++;
59
+ } else if (msg.role === "assistant") {
60
+ const amsg = msg as AssistantMessageType;
61
+ items.push({
62
+ key: `msg:${index}`,
63
+ template: html`<assistant-message
64
+ .message=${amsg}
65
+ .tools=${this.tools}
66
+ .isStreaming=${false}
67
+ .pendingToolCalls=${this.pendingToolCalls}
68
+ .toolResultsById=${resultByCallId}
69
+ .hideToolCalls=${false}
70
+ .hidePendingToolCalls=${this.isStreaming}
71
+ .onCostClick=${this.onCostClick}
72
+ ></assistant-message>`,
73
+ });
74
+ index++;
75
+ } else {
76
+ // Skip standalone toolResult messages; they are rendered via paired tool-message above
77
+ // Skip unknown roles
78
+ }
79
+ }
80
+ return items;
81
+ }
82
+
83
+ override render() {
84
+ const items = this.buildRenderItems();
85
+ return html`<div class="flex flex-col gap-3">
86
+ ${repeat(
87
+ items,
88
+ (it) => it.key,
89
+ (it) => it.template,
90
+ )}
91
+ </div>`;
92
+ }
93
+ }
94
+
95
+ // Register custom element
96
+ if (!customElements.get("message-list")) {
97
+ customElements.define("message-list", MessageList);
98
+ }