@dxos/plugin-automation 0.7.5-main.ff8607b → 0.7.5-staging.2ff1350

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 (301) hide show
  1. package/dist/lib/browser/AutomationPanel-NCKK3CFN.mjs +8 -0
  2. package/dist/lib/browser/AutomationPanel-NCKK3CFN.mjs.map +7 -0
  3. package/dist/lib/browser/ChatContainer-AZPLZJ5H.mjs +12 -0
  4. package/dist/lib/browser/ChatContainer-AZPLZJ5H.mjs.map +7 -0
  5. package/dist/lib/browser/ai-client-T2PBJISL.mjs +22 -0
  6. package/dist/lib/browser/ai-client-T2PBJISL.mjs.map +7 -0
  7. package/dist/lib/browser/app-graph-builder-F22H4YBQ.mjs +111 -0
  8. package/dist/lib/browser/app-graph-builder-F22H4YBQ.mjs.map +7 -0
  9. package/dist/lib/browser/chunk-67KDIIM6.mjs +15 -0
  10. package/dist/lib/browser/chunk-67KDIIM6.mjs.map +7 -0
  11. package/dist/lib/browser/{chunk-DQ7ZSYJJ.mjs → chunk-HZSTXZWK.mjs} +3 -1
  12. package/dist/lib/browser/chunk-HZSTXZWK.mjs.map +7 -0
  13. package/dist/lib/browser/chunk-L2YLAKCS.mjs +151 -0
  14. package/dist/lib/browser/chunk-L2YLAKCS.mjs.map +7 -0
  15. package/dist/lib/{node-esm/AutomationPanel-MW42U6I4.mjs → browser/chunk-LJ7F4KDM.mjs} +26 -15
  16. package/dist/lib/browser/chunk-LJ7F4KDM.mjs.map +7 -0
  17. package/dist/lib/browser/chunk-M4MWUOUC.mjs +382 -0
  18. package/dist/lib/browser/chunk-M4MWUOUC.mjs.map +7 -0
  19. package/dist/lib/browser/chunk-U435IUIA.mjs +1775 -0
  20. package/dist/lib/browser/chunk-U435IUIA.mjs.map +7 -0
  21. package/dist/lib/browser/index.mjs +127 -42
  22. package/dist/lib/browser/index.mjs.map +4 -4
  23. package/dist/lib/browser/intent-resolver-MONKAKTY.mjs +30 -0
  24. package/dist/lib/browser/intent-resolver-MONKAKTY.mjs.map +7 -0
  25. package/dist/lib/browser/meta.json +1 -1
  26. package/dist/lib/browser/react-surface-7QWCWE4K.mjs +58 -0
  27. package/dist/lib/browser/react-surface-7QWCWE4K.mjs.map +7 -0
  28. package/dist/lib/browser/types/index.mjs +14 -3
  29. package/dist/lib/node/AutomationPanel-6COLKRIB.cjs +26 -0
  30. package/dist/lib/node/AutomationPanel-6COLKRIB.cjs.map +7 -0
  31. package/dist/lib/node/ChatContainer-SBEUFMQ3.cjs +33 -0
  32. package/dist/lib/node/ChatContainer-SBEUFMQ3.cjs.map +7 -0
  33. package/dist/lib/node/ai-client-L5GUSAER.cjs +38 -0
  34. package/dist/lib/node/ai-client-L5GUSAER.cjs.map +7 -0
  35. package/dist/lib/node/app-graph-builder-OCSH5R6U.cjs +123 -0
  36. package/dist/lib/node/app-graph-builder-OCSH5R6U.cjs.map +7 -0
  37. package/dist/lib/node/{chunk-TQEDPRY5.cjs → chunk-BAQRBGOJ.cjs} +7 -4
  38. package/dist/lib/node/chunk-BAQRBGOJ.cjs.map +7 -0
  39. package/dist/lib/node/chunk-BZFAEBME.cjs +1747 -0
  40. package/dist/lib/node/chunk-BZFAEBME.cjs.map +7 -0
  41. package/dist/lib/node/chunk-K3F5JNZ6.cjs +394 -0
  42. package/dist/lib/node/chunk-K3F5JNZ6.cjs.map +7 -0
  43. package/dist/lib/node/chunk-NUOYHYAF.cjs +174 -0
  44. package/dist/lib/node/chunk-NUOYHYAF.cjs.map +7 -0
  45. package/dist/lib/node/{AutomationPanel-H5WE4FIU.cjs → chunk-V4L7LTWM.cjs} +34 -18
  46. package/dist/lib/node/chunk-V4L7LTWM.cjs.map +7 -0
  47. package/dist/lib/node/chunk-VNBYGJ4K.cjs +34 -0
  48. package/dist/lib/node/chunk-VNBYGJ4K.cjs.map +7 -0
  49. package/dist/lib/node/index.cjs +157 -86
  50. package/dist/lib/node/index.cjs.map +4 -4
  51. package/dist/lib/node/intent-resolver-KQGTYVF2.cjs +45 -0
  52. package/dist/lib/node/intent-resolver-KQGTYVF2.cjs.map +7 -0
  53. package/dist/lib/node/meta.json +1 -1
  54. package/dist/lib/node/{react-surface-S6U5ISJ6.cjs → react-surface-IXKIJFRP.cjs} +31 -12
  55. package/dist/lib/node/react-surface-IXKIJFRP.cjs.map +7 -0
  56. package/dist/lib/node/types/index.cjs +17 -6
  57. package/dist/lib/node/types/index.cjs.map +2 -2
  58. package/dist/lib/node-esm/AutomationPanel-3MPIAO7R.mjs +9 -0
  59. package/dist/lib/node-esm/AutomationPanel-3MPIAO7R.mjs.map +7 -0
  60. package/dist/lib/node-esm/ChatContainer-JCDLYMDZ.mjs +13 -0
  61. package/dist/lib/node-esm/ChatContainer-JCDLYMDZ.mjs.map +7 -0
  62. package/dist/lib/node-esm/ai-client-45LZLW7V.mjs +23 -0
  63. package/dist/lib/node-esm/ai-client-45LZLW7V.mjs.map +7 -0
  64. package/dist/lib/node-esm/app-graph-builder-XSM7DETF.mjs +112 -0
  65. package/dist/lib/node-esm/app-graph-builder-XSM7DETF.mjs.map +7 -0
  66. package/dist/lib/node-esm/chunk-53FFD4VO.mjs +1776 -0
  67. package/dist/lib/node-esm/chunk-53FFD4VO.mjs.map +7 -0
  68. package/dist/lib/{browser/AutomationPanel-KT3YFV56.mjs → node-esm/chunk-AWAU4BHR.mjs} +27 -14
  69. package/dist/lib/node-esm/chunk-AWAU4BHR.mjs.map +7 -0
  70. package/dist/lib/node-esm/{chunk-JFHI22MF.mjs → chunk-IG4GSWZN.mjs} +3 -1
  71. package/dist/lib/node-esm/chunk-IG4GSWZN.mjs.map +7 -0
  72. package/dist/lib/node-esm/chunk-K5NNSGIV.mjs +152 -0
  73. package/dist/lib/node-esm/chunk-K5NNSGIV.mjs.map +7 -0
  74. package/dist/lib/node-esm/chunk-RAT3GSJP.mjs +383 -0
  75. package/dist/lib/node-esm/chunk-RAT3GSJP.mjs.map +7 -0
  76. package/dist/lib/node-esm/chunk-SS7U6GCX.mjs +16 -0
  77. package/dist/lib/node-esm/chunk-SS7U6GCX.mjs.map +7 -0
  78. package/dist/lib/node-esm/index.mjs +127 -42
  79. package/dist/lib/node-esm/index.mjs.map +4 -4
  80. package/dist/lib/node-esm/intent-resolver-XSZRLU6V.mjs +31 -0
  81. package/dist/lib/node-esm/intent-resolver-XSZRLU6V.mjs.map +7 -0
  82. package/dist/lib/node-esm/meta.json +1 -1
  83. package/dist/lib/node-esm/react-surface-FZAAAXGR.mjs +59 -0
  84. package/dist/lib/node-esm/react-surface-FZAAAXGR.mjs.map +7 -0
  85. package/dist/lib/node-esm/types/index.mjs +14 -3
  86. package/dist/types/src/AutomationPlugin.d.ts.map +1 -1
  87. package/dist/types/src/capabilities/ai-client.d.ts +5 -0
  88. package/dist/types/src/capabilities/ai-client.d.ts.map +1 -0
  89. package/dist/types/src/capabilities/app-graph-builder.d.ts +23 -22
  90. package/dist/types/src/capabilities/app-graph-builder.d.ts.map +1 -1
  91. package/dist/types/src/capabilities/capabilities.d.ts +5 -0
  92. package/dist/types/src/capabilities/capabilities.d.ts.map +1 -0
  93. package/dist/types/src/capabilities/index.d.ts +3 -0
  94. package/dist/types/src/capabilities/index.d.ts.map +1 -1
  95. package/dist/types/src/capabilities/intent-resolver.d.ts +4 -0
  96. package/dist/types/src/capabilities/intent-resolver.d.ts.map +1 -0
  97. package/dist/types/src/capabilities/react-surface.d.ts.map +1 -1
  98. package/dist/types/src/components/AmbientDialog/AmbientDialog.d.ts +7 -0
  99. package/dist/types/src/components/AmbientDialog/AmbientDialog.d.ts.map +1 -0
  100. package/dist/types/src/components/AmbientDialog/AmbientDialog.stories.d.ts +8 -0
  101. package/dist/types/src/components/AmbientDialog/AmbientDialog.stories.d.ts.map +1 -0
  102. package/dist/types/src/components/AmbientDialog/index.d.ts +2 -0
  103. package/dist/types/src/components/AmbientDialog/index.d.ts.map +1 -0
  104. package/dist/types/src/components/AssistantDialog/AssistantDialog.d.ts +6 -0
  105. package/dist/types/src/components/AssistantDialog/AssistantDialog.d.ts.map +1 -0
  106. package/dist/types/src/components/AssistantDialog/index.d.ts +2 -0
  107. package/dist/types/src/components/AssistantDialog/index.d.ts.map +1 -0
  108. package/dist/types/src/components/AutomationPanel/AutomationPanel.d.ts +1 -2
  109. package/dist/types/src/components/AutomationPanel/AutomationPanel.d.ts.map +1 -1
  110. package/dist/types/src/components/ChatContainer/ChatContainer.d.ts +6 -0
  111. package/dist/types/src/components/ChatContainer/ChatContainer.d.ts.map +1 -0
  112. package/dist/types/src/components/ChatContainer/index.d.ts +4 -0
  113. package/dist/types/src/components/ChatContainer/index.d.ts.map +1 -0
  114. package/dist/types/src/components/Prompt/Prompt.d.ts +13 -0
  115. package/dist/types/src/components/Prompt/Prompt.d.ts.map +1 -0
  116. package/dist/types/src/components/Prompt/Prompt.stories.d.ts +9 -0
  117. package/dist/types/src/components/Prompt/Prompt.stories.d.ts.map +1 -0
  118. package/dist/types/src/components/Prompt/PromptBar.d.ts +10 -0
  119. package/dist/types/src/components/Prompt/PromptBar.d.ts.map +1 -0
  120. package/dist/types/src/components/Prompt/autocomplete.d.ts +21 -0
  121. package/dist/types/src/components/Prompt/autocomplete.d.ts.map +1 -0
  122. package/dist/types/src/components/Prompt/index.d.ts +3 -0
  123. package/dist/types/src/components/Prompt/index.d.ts.map +1 -0
  124. package/dist/types/src/components/PromptEditor/PromptEditor.d.ts +1 -2
  125. package/dist/types/src/components/PromptEditor/PromptEditor.d.ts.map +1 -1
  126. package/dist/types/src/components/ServiceRegistry/ServiceRegistry.d.ts +5 -0
  127. package/dist/types/src/components/ServiceRegistry/ServiceRegistry.d.ts.map +1 -0
  128. package/dist/types/src/components/ServiceRegistry/ServiceRegistry.stories.d.ts +8 -0
  129. package/dist/types/src/components/ServiceRegistry/ServiceRegistry.stories.d.ts.map +1 -0
  130. package/dist/types/src/components/ServiceRegistry/index.d.ts +2 -0
  131. package/dist/types/src/components/ServiceRegistry/index.d.ts.map +1 -0
  132. package/dist/types/src/components/Thread/Thread.d.ts +15 -0
  133. package/dist/types/src/components/Thread/Thread.d.ts.map +1 -0
  134. package/dist/types/src/components/Thread/Thread.stories.d.ts +11 -0
  135. package/dist/types/src/components/Thread/Thread.stories.d.ts.map +1 -0
  136. package/dist/types/src/components/Thread/ThreadContainer.d.ts +8 -0
  137. package/dist/types/src/components/Thread/ThreadContainer.d.ts.map +1 -0
  138. package/dist/types/src/components/Thread/ThreadContainer.stories.d.ts +15 -0
  139. package/dist/types/src/components/Thread/ThreadContainer.stories.d.ts.map +1 -0
  140. package/dist/types/src/components/Thread/ThreadMessage.d.ts +13 -0
  141. package/dist/types/src/components/Thread/ThreadMessage.d.ts.map +1 -0
  142. package/dist/types/src/components/Thread/ToolInvocations.d.ts +13 -0
  143. package/dist/types/src/components/Thread/ToolInvocations.d.ts.map +1 -0
  144. package/dist/types/src/components/Thread/index.d.ts +3 -0
  145. package/dist/types/src/components/Thread/index.d.ts.map +1 -0
  146. package/dist/types/src/components/Thread/reducer.d.ts +12 -0
  147. package/dist/types/src/components/Thread/reducer.d.ts.map +1 -0
  148. package/dist/types/src/components/Toolbox/Toolbox.d.ts +19 -0
  149. package/dist/types/src/components/Toolbox/Toolbox.d.ts.map +1 -0
  150. package/dist/types/src/components/Toolbox/Toolbox.stories.d.ts +8 -0
  151. package/dist/types/src/components/Toolbox/Toolbox.stories.d.ts.map +1 -0
  152. package/dist/types/src/components/Toolbox/index.d.ts +2 -0
  153. package/dist/types/src/components/Toolbox/index.d.ts.map +1 -0
  154. package/dist/types/src/components/TriggerEditor/TriggerEditor.d.ts +1 -2
  155. package/dist/types/src/components/TriggerEditor/TriggerEditor.d.ts.map +1 -1
  156. package/dist/types/src/components/index.d.ts +13 -2
  157. package/dist/types/src/components/index.d.ts.map +1 -1
  158. package/dist/types/src/hooks/index.d.ts +5 -0
  159. package/dist/types/src/hooks/index.d.ts.map +1 -1
  160. package/dist/types/src/hooks/processor.d.ts +75 -0
  161. package/dist/types/src/hooks/processor.d.ts.map +1 -0
  162. package/dist/types/src/hooks/processor.test.d.ts +2 -0
  163. package/dist/types/src/hooks/processor.test.d.ts.map +1 -0
  164. package/dist/types/src/hooks/useChatProcessor.d.ts +7 -0
  165. package/dist/types/src/hooks/useChatProcessor.d.ts.map +1 -0
  166. package/dist/types/src/hooks/useMessageQueue.d.ts +41 -0
  167. package/dist/types/src/hooks/useMessageQueue.d.ts.map +1 -0
  168. package/dist/types/src/hooks/useServices.d.ts +7 -0
  169. package/dist/types/src/hooks/useServices.d.ts.map +1 -0
  170. package/dist/types/src/hooks/useTextInputEvents.d.ts +13 -0
  171. package/dist/types/src/hooks/useTextInputEvents.d.ts.map +1 -0
  172. package/dist/types/src/meta.d.ts +1 -0
  173. package/dist/types/src/meta.d.ts.map +1 -1
  174. package/dist/types/src/testing/index.d.ts +2 -1
  175. package/dist/types/src/testing/index.d.ts.map +1 -1
  176. package/dist/types/src/testing/{testing.d.ts → test-functions.d.ts} +3 -1
  177. package/dist/types/src/testing/test-functions.d.ts.map +1 -0
  178. package/dist/types/src/testing/test-services.d.ts +5 -0
  179. package/dist/types/src/testing/test-services.d.ts.map +1 -0
  180. package/dist/types/src/tools/function.d.ts +5 -0
  181. package/dist/types/src/tools/function.d.ts.map +1 -0
  182. package/dist/types/src/tools/index.d.ts +3 -0
  183. package/dist/types/src/tools/index.d.ts.map +1 -0
  184. package/dist/types/src/tools/openapi.d.ts +10 -0
  185. package/dist/types/src/tools/openapi.d.ts.map +1 -0
  186. package/dist/types/src/tools/openapi.test.d.ts +2 -0
  187. package/dist/types/src/tools/openapi.test.d.ts.map +1 -0
  188. package/dist/types/src/translations.d.ts +95 -2
  189. package/dist/types/src/translations.d.ts.map +1 -1
  190. package/dist/types/src/types/index.d.ts +2 -0
  191. package/dist/types/src/types/index.d.ts.map +1 -1
  192. package/dist/types/src/types/registry.d.ts +10 -0
  193. package/dist/types/src/types/registry.d.ts.map +1 -0
  194. package/dist/types/src/types/schema.d.ts +161 -3
  195. package/dist/types/src/types/schema.d.ts.map +1 -1
  196. package/dist/types/src/types/types.d.ts +19 -0
  197. package/dist/types/src/types/types.d.ts.map +1 -0
  198. package/package.json +59 -40
  199. package/src/AutomationPlugin.tsx +51 -14
  200. package/src/capabilities/ai-client.ts +19 -0
  201. package/src/capabilities/app-graph-builder.ts +83 -110
  202. package/src/capabilities/capabilities.ts +12 -0
  203. package/src/capabilities/index.ts +4 -0
  204. package/src/capabilities/intent-resolver.ts +26 -0
  205. package/src/capabilities/react-surface.tsx +21 -6
  206. package/src/components/AmbientDialog/AmbientDialog.stories.tsx +81 -0
  207. package/src/components/AmbientDialog/AmbientDialog.tsx +103 -0
  208. package/src/components/AmbientDialog/index.ts +5 -0
  209. package/src/components/AssistantDialog/AssistantDialog.tsx +23 -0
  210. package/src/components/AssistantDialog/index.ts +5 -0
  211. package/src/components/ChatContainer/ChatContainer.tsx +19 -0
  212. package/src/components/ChatContainer/index.ts +9 -0
  213. package/src/components/Prompt/Prompt.stories.tsx +79 -0
  214. package/src/components/Prompt/Prompt.tsx +79 -0
  215. package/src/components/Prompt/PromptBar.tsx +100 -0
  216. package/src/components/Prompt/autocomplete.ts +212 -0
  217. package/src/components/Prompt/index.ts +6 -0
  218. package/src/components/PromptEditor/PromptEditor.stories.tsx +3 -3
  219. package/src/components/PromptEditor/PromptEditor.tsx +3 -3
  220. package/src/components/ServiceRegistry/ServiceRegistry.stories.tsx +49 -0
  221. package/src/components/ServiceRegistry/ServiceRegistry.tsx +76 -0
  222. package/src/components/ServiceRegistry/index.ts +5 -0
  223. package/src/components/Thread/Thread.stories.tsx +210 -0
  224. package/src/components/Thread/Thread.tsx +106 -0
  225. package/src/components/Thread/ThreadContainer.stories.tsx +245 -0
  226. package/src/components/Thread/ThreadContainer.tsx +67 -0
  227. package/src/components/Thread/ThreadMessage.tsx +187 -0
  228. package/src/components/Thread/ToolInvocations.tsx +86 -0
  229. package/src/components/Thread/index.ts +6 -0
  230. package/src/components/Thread/reducer.ts +52 -0
  231. package/src/components/Toolbox/Toolbox.stories.tsx +66 -0
  232. package/src/components/Toolbox/Toolbox.tsx +89 -0
  233. package/src/components/Toolbox/index.ts +5 -0
  234. package/src/components/TriggerEditor/TriggerEditor.tsx +18 -11
  235. package/src/components/index.ts +10 -1
  236. package/src/hooks/index.ts +6 -0
  237. package/src/hooks/processor.test.ts +15 -0
  238. package/src/hooks/processor.ts +218 -0
  239. package/src/hooks/useChatProcessor.tsx +75 -0
  240. package/src/hooks/useMessageQueue.ts +23 -0
  241. package/src/hooks/useServices.ts +28 -0
  242. package/src/hooks/useTextInputEvents.ts +49 -0
  243. package/src/meta.ts +2 -0
  244. package/src/testing/index.ts +2 -1
  245. package/src/testing/{testing.ts → test-functions.ts} +9 -2
  246. package/src/testing/test-services.ts +131 -0
  247. package/src/tools/function.ts +47 -0
  248. package/src/tools/index.ts +6 -0
  249. package/src/tools/openapi.test.ts +224 -0
  250. package/src/tools/openapi.ts +331 -0
  251. package/src/translations.ts +17 -2
  252. package/src/types/index.ts +2 -0
  253. package/src/types/registry.ts +26 -0
  254. package/src/types/schema.ts +95 -1
  255. package/src/types/types.ts +22 -0
  256. package/src/typings.d.ts +9 -0
  257. package/dist/lib/browser/AssistantPanel-NTOUHOWA.mjs +0 -339
  258. package/dist/lib/browser/AssistantPanel-NTOUHOWA.mjs.map +0 -7
  259. package/dist/lib/browser/AutomationPanel-KT3YFV56.mjs.map +0 -7
  260. package/dist/lib/browser/app-graph-builder-6FSMLMT4.mjs +0 -131
  261. package/dist/lib/browser/app-graph-builder-6FSMLMT4.mjs.map +0 -7
  262. package/dist/lib/browser/chunk-4ODBJDXM.mjs +0 -10
  263. package/dist/lib/browser/chunk-4ODBJDXM.mjs.map +0 -7
  264. package/dist/lib/browser/chunk-DL6LB2NI.mjs +0 -43
  265. package/dist/lib/browser/chunk-DL6LB2NI.mjs.map +0 -7
  266. package/dist/lib/browser/chunk-DQ7ZSYJJ.mjs.map +0 -7
  267. package/dist/lib/browser/react-surface-5YTE5MRE.mjs +0 -34
  268. package/dist/lib/browser/react-surface-5YTE5MRE.mjs.map +0 -7
  269. package/dist/lib/node/AssistantPanel-U7GDWPRC.cjs +0 -359
  270. package/dist/lib/node/AssistantPanel-U7GDWPRC.cjs.map +0 -7
  271. package/dist/lib/node/AutomationPanel-H5WE4FIU.cjs.map +0 -7
  272. package/dist/lib/node/app-graph-builder-65TBXO3I.cjs +0 -147
  273. package/dist/lib/node/app-graph-builder-65TBXO3I.cjs.map +0 -7
  274. package/dist/lib/node/chunk-JNDMZQH7.cjs +0 -68
  275. package/dist/lib/node/chunk-JNDMZQH7.cjs.map +0 -7
  276. package/dist/lib/node/chunk-JUSXT4U5.cjs +0 -43
  277. package/dist/lib/node/chunk-JUSXT4U5.cjs.map +0 -7
  278. package/dist/lib/node/chunk-TQEDPRY5.cjs.map +0 -7
  279. package/dist/lib/node/react-surface-S6U5ISJ6.cjs.map +0 -7
  280. package/dist/lib/node-esm/AssistantPanel-M67P24GS.mjs +0 -340
  281. package/dist/lib/node-esm/AssistantPanel-M67P24GS.mjs.map +0 -7
  282. package/dist/lib/node-esm/AutomationPanel-MW42U6I4.mjs.map +0 -7
  283. package/dist/lib/node-esm/app-graph-builder-4SD4F3KN.mjs +0 -132
  284. package/dist/lib/node-esm/app-graph-builder-4SD4F3KN.mjs.map +0 -7
  285. package/dist/lib/node-esm/chunk-GIAYUM5I.mjs +0 -45
  286. package/dist/lib/node-esm/chunk-GIAYUM5I.mjs.map +0 -7
  287. package/dist/lib/node-esm/chunk-JFHI22MF.mjs.map +0 -7
  288. package/dist/lib/node-esm/chunk-XVIEXSEZ.mjs +0 -12
  289. package/dist/lib/node-esm/chunk-XVIEXSEZ.mjs.map +0 -7
  290. package/dist/lib/node-esm/react-surface-RWTR3TKT.mjs +0 -35
  291. package/dist/lib/node-esm/react-surface-RWTR3TKT.mjs.map +0 -7
  292. package/dist/types/src/components/AssistantPanel/AssistantPanel.d.ts +0 -8
  293. package/dist/types/src/components/AssistantPanel/AssistantPanel.d.ts.map +0 -1
  294. package/dist/types/src/components/AssistantPanel/index.d.ts +0 -3
  295. package/dist/types/src/components/AssistantPanel/index.d.ts.map +0 -1
  296. package/dist/types/src/components/AssistantPanel/system-instructions.d.ts +0 -6
  297. package/dist/types/src/components/AssistantPanel/system-instructions.d.ts.map +0 -1
  298. package/dist/types/src/testing/testing.d.ts.map +0 -1
  299. package/src/components/AssistantPanel/AssistantPanel.tsx +0 -227
  300. package/src/components/AssistantPanel/index.ts +0 -7
  301. package/src/components/AssistantPanel/system-instructions.ts +0 -166
@@ -0,0 +1,106 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import React, { type CSSProperties, useCallback, useMemo, useRef } from 'react';
6
+
7
+ import { useCapabilities } from '@dxos/app-framework';
8
+ import { type Message } from '@dxos/artifact';
9
+ import { TranscriptionCapabilities } from '@dxos/plugin-transcription';
10
+ import { type Space } from '@dxos/react-client/echo';
11
+ import { useIdentity } from '@dxos/react-client/halo';
12
+ import { type ThemedClassName } from '@dxos/react-ui';
13
+ import { ScrollContainer, type ScrollController } from '@dxos/react-ui-components';
14
+ import { mx } from '@dxos/react-ui-theme';
15
+ import { keyToFallback } from '@dxos/util';
16
+
17
+ import { ThreadMessage, type ThreadMessageProps } from './ThreadMessage';
18
+ import { messageReducer } from './reducer';
19
+ import { PromptBar, type PromptBarProps } from '../Prompt';
20
+
21
+ export type ThreadProps = ThemedClassName<{
22
+ space?: Space;
23
+ messages?: Message[];
24
+ collapse?: boolean;
25
+ }> &
26
+ Pick<PromptBarProps, 'processing' | 'error' | 'onSubmit' | 'onSuggest' | 'onCancel'> &
27
+ Pick<ThreadMessageProps, 'debug' | 'onPrompt' | 'onDelete'>;
28
+
29
+ /**
30
+ * Chat thread component.
31
+ */
32
+ export const Thread = ({
33
+ classNames,
34
+ space,
35
+ messages,
36
+ collapse = true,
37
+ processing,
38
+ error,
39
+ debug,
40
+ onSubmit,
41
+ onCancel,
42
+ onPrompt,
43
+ onDelete,
44
+ }: ThreadProps) => {
45
+ const hasTrascription = useCapabilities(TranscriptionCapabilities.Transcription).length > 0;
46
+
47
+ const scroller = useRef<ScrollController>(null);
48
+
49
+ const identity = useIdentity();
50
+ const fallbackValue = keyToFallback(identity!.identityKey);
51
+ const userHue = identity!.profile?.data?.hue || fallbackValue.hue;
52
+
53
+ const handleSubmit = useCallback<NonNullable<PromptBarProps['onSubmit']>>(
54
+ (value: string) => {
55
+ onSubmit?.(value);
56
+ scroller.current?.scrollToBottom();
57
+ return true;
58
+ },
59
+ [onSubmit],
60
+ );
61
+
62
+ // TODO(dmaretskyi): This needs to be a separate type: `id` is not a valid ObjectId, this needs to accommodate messageId for deletion.
63
+ const { messages: filteredMessages = [] } = useMemo(() => {
64
+ if (collapse) {
65
+ return (messages ?? []).reduce<{ messages: Message[]; current?: Message }>(messageReducer, {
66
+ messages: [],
67
+ });
68
+ } else {
69
+ return { messages: messages ?? [] };
70
+ }
71
+ }, [messages, collapse]);
72
+
73
+ return (
74
+ <>
75
+ <ScrollContainer ref={scroller} fade>
76
+ <div
77
+ role='none'
78
+ className={mx(filteredMessages.length > 0 && 'pbs-6 pbe-6', classNames)}
79
+ style={{ '--user-fill': `var(--dx-${userHue}Fill)` } as CSSProperties}
80
+ >
81
+ {filteredMessages.map((message) => (
82
+ <ThreadMessage
83
+ key={message.id}
84
+ classNames='px-4 pbe-4'
85
+ space={space}
86
+ message={message}
87
+ debug={debug}
88
+ onPrompt={onPrompt}
89
+ onDelete={onDelete}
90
+ />
91
+ ))}
92
+ </div>
93
+ </ScrollContainer>
94
+
95
+ {onSubmit && (
96
+ <PromptBar
97
+ microphone={hasTrascription}
98
+ processing={processing}
99
+ error={error}
100
+ onSubmit={handleSubmit}
101
+ onCancel={onCancel}
102
+ />
103
+ )}
104
+ </>
105
+ );
106
+ };
@@ -0,0 +1,245 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import '@dxos-theme';
6
+
7
+ import { type StoryObj, type Meta } from '@storybook/react';
8
+ import React, { useCallback, useEffect, useMemo, useState } from 'react';
9
+
10
+ import { Capabilities, IntentPlugin, Surface, useCapabilities, useIntentDispatcher } from '@dxos/app-framework';
11
+ import { withPluginManager } from '@dxos/app-framework/testing';
12
+ import { Message, type Tool } from '@dxos/artifact';
13
+ import { capabilities, genericTools, localServiceEndpoints, type IsObject } from '@dxos/artifact-testing';
14
+ import { AIServiceClientImpl } from '@dxos/assistant';
15
+ import { createStatic, ObjectId } from '@dxos/echo-schema';
16
+ import { EdgeHttpClient } from '@dxos/edge-client';
17
+ import { invariant } from '@dxos/invariant';
18
+ import { DXN, QueueSubspaceTags, SpaceId } from '@dxos/keys';
19
+ import { ChessPlugin } from '@dxos/plugin-chess';
20
+ import { ChessType } from '@dxos/plugin-chess/types';
21
+ import { ClientPlugin } from '@dxos/plugin-client';
22
+ import { MapPlugin } from '@dxos/plugin-map';
23
+ import { SpacePlugin } from '@dxos/plugin-space';
24
+ import { TablePlugin } from '@dxos/plugin-table';
25
+ import { useSpace } from '@dxos/react-client/echo';
26
+ import { useQueue } from '@dxos/react-edge-client';
27
+ import { IconButton, Input, Toolbar } from '@dxos/react-ui';
28
+ import { mx } from '@dxos/react-ui-theme';
29
+ import { withLayout, withSignals, withTheme } from '@dxos/storybook-utils';
30
+
31
+ import { Thread, type ThreadProps } from './Thread';
32
+ import { ChatProcessor } from '../../hooks';
33
+ import { createProcessorOptions } from '../../testing';
34
+ import translations from '../../translations';
35
+
36
+ const endpoints = localServiceEndpoints;
37
+
38
+ type RenderProps = {
39
+ items?: IsObject[];
40
+ prompts?: string[];
41
+ } & Pick<ThreadProps, 'debug'>;
42
+
43
+ // TODO(burdon): Use ChatContainer.
44
+ const Render = ({ items: _items, prompts = [], ...props }: RenderProps) => {
45
+ const space = useSpace();
46
+ const artifactDefinitions = useCapabilities(Capabilities.ArtifactDefinition);
47
+ const tools = useMemo<Tool[]>(
48
+ () => [...genericTools, ...artifactDefinitions.flatMap((definition) => definition.tools)],
49
+ [genericTools, artifactDefinitions],
50
+ );
51
+
52
+ const [aiClient] = useState(() => new AIServiceClientImpl({ endpoint: endpoints.ai }));
53
+ const [edgeClient] = useState(() => new EdgeHttpClient(endpoints.edge));
54
+ const { dispatchPromise: dispatch } = useIntentDispatcher();
55
+ const processor = useMemo<ChatProcessor | undefined>(() => {
56
+ if (!space) {
57
+ return;
58
+ }
59
+
60
+ return new ChatProcessor(
61
+ aiClient,
62
+ tools,
63
+ {
64
+ space,
65
+ dispatch,
66
+ },
67
+ createProcessorOptions(artifactDefinitions.map((definition) => definition.instructions)),
68
+ );
69
+ }, [aiClient, tools, space, dispatch, artifactDefinitions]);
70
+
71
+ // Queue.
72
+ const [queueDxn, setQueueDxn] = useState(() => randomQueueDxn());
73
+ const queue = useQueue<Message>(edgeClient, DXN.tryParse(queueDxn));
74
+
75
+ useEffect(() => {
76
+ if (queue?.items.length === 0 && !queue.isLoading && prompts.length > 0) {
77
+ queue.append([
78
+ createStatic(Message, {
79
+ role: 'assistant',
80
+ content: prompts.map(
81
+ (prompt) =>
82
+ ({
83
+ type: 'json',
84
+ disposition: 'suggest',
85
+ json: JSON.stringify({ text: prompt }),
86
+ }) as const,
87
+ ),
88
+ }),
89
+ ]);
90
+ }
91
+ }, [queueDxn, prompts, queue?.items.length, queue?.isLoading]);
92
+
93
+ // State.
94
+ const artifactItems: any[] = []; // TODO(burdon): Query from space.
95
+ const messages = [...(queue?.items ?? []), ...(processor?.messages.value ?? [])];
96
+
97
+ const handleSubmit = processor
98
+ ? (message: string) => {
99
+ requestAnimationFrame(async () => {
100
+ invariant(processor);
101
+ if (processor.streaming.value) {
102
+ await processor.cancel();
103
+ }
104
+
105
+ invariant(queue);
106
+ await processor.request(message, {
107
+ history: queue.items,
108
+ onComplete: (messages) => {
109
+ queue.append(messages);
110
+ },
111
+ });
112
+ });
113
+
114
+ return true;
115
+ }
116
+ : undefined;
117
+
118
+ const handlePrompt = useCallback(
119
+ (text: string) => {
120
+ void handleSubmit?.(text);
121
+ },
122
+ [handleSubmit],
123
+ );
124
+
125
+ const handleDelete = useCallback(
126
+ (id: string) => {
127
+ invariant(ObjectId.isValid(id), 'Invalid message id');
128
+ void queue?.delete([id]);
129
+ },
130
+ [queue],
131
+ );
132
+
133
+ return (
134
+ <div className='grid grid-cols-2 w-full h-full divide-x divide-separator overflow-hidden'>
135
+ {/* Thread */}
136
+ <div className='flex flex-col gap-4 overflow-hidden'>
137
+ <Toolbar.Root classNames='p-2'>
138
+ <Input.Root>
139
+ <Input.TextInput
140
+ spellCheck={false}
141
+ placeholder='Queue DXN'
142
+ value={queueDxn}
143
+ onChange={(ev) => setQueueDxn(ev.target.value)}
144
+ />
145
+ <IconButton
146
+ iconOnly
147
+ label='Copy DXN'
148
+ icon='ph--copy--regular'
149
+ onClick={() => navigator.clipboard.writeText(queueDxn)}
150
+ />
151
+ <IconButton
152
+ iconOnly
153
+ label='Clear history'
154
+ icon='ph--trash--regular'
155
+ onClick={() => setQueueDxn(randomQueueDxn())}
156
+ />
157
+ <IconButton iconOnly label='Stop' icon='ph--stop--regular' onClick={() => processor?.cancel()} />
158
+ </Input.Root>
159
+ </Toolbar.Root>
160
+
161
+ {/* TODO(burdon): Replace with ThreadContainer. */}
162
+ <Thread
163
+ messages={messages}
164
+ processing={processor?.streaming.value}
165
+ onSubmit={processor ? handleSubmit : undefined}
166
+ onPrompt={processor ? handlePrompt : undefined}
167
+ onDelete={processor ? handleDelete : undefined}
168
+ {...props}
169
+ />
170
+ </div>
171
+
172
+ {/* Artifacts Deck/Mosaic */}
173
+ <div className='overflow-hidden grid grid-rows-[2fr_1fr] divide-y divide-separator'>
174
+ {artifactItems.length > 0 && (
175
+ <div className={mx('flex grow overflow-hidden', artifactItems.length === 1 && 'row-span-2')}>
176
+ <Surface role='canvas-node' limit={1} data={artifactItems[0]} />
177
+ </div>
178
+ )}
179
+
180
+ {artifactItems.length > 1 && (
181
+ <div className='flex shrink-0 overflow-hidden divide-x divide-separator'>
182
+ <div className='flex flex-1 h-full'>
183
+ {artifactItems.slice(1, 3).map((item, idx) => (
184
+ <Surface key={idx} role='canvas-node' limit={1} data={item} />
185
+ ))}
186
+ </div>
187
+ </div>
188
+ )}
189
+ </div>
190
+ </div>
191
+ );
192
+ };
193
+
194
+ const randomQueueDxn = () =>
195
+ new DXN(DXN.kind.QUEUE, [QueueSubspaceTags.DATA, SpaceId.random(), ObjectId.random()]).toString();
196
+
197
+ const meta: Meta<typeof Render> = {
198
+ title: 'plugins/plugin-automation/ThreadContainer',
199
+ render: Render,
200
+ decorators: [
201
+ withSignals,
202
+ withPluginManager({
203
+ plugins: [
204
+ ClientPlugin({
205
+ onClientInitialized: async (_, client) => {
206
+ await client.halo.createIdentity();
207
+ },
208
+ }),
209
+ SpacePlugin({ observability: false }),
210
+ IntentPlugin(),
211
+ ChessPlugin(),
212
+ MapPlugin(),
213
+ TablePlugin(),
214
+ ],
215
+ capabilities,
216
+ }),
217
+ withTheme,
218
+ withLayout({ fullscreen: true, tooltips: true }),
219
+ ],
220
+ parameters: {
221
+ translations,
222
+ },
223
+ };
224
+
225
+ export default meta;
226
+
227
+ type Story = StoryObj<typeof Render>;
228
+
229
+ export const Default: Story = {
230
+ args: {
231
+ debug: true,
232
+ prompts: ['Ask me a question', 'Show me a chess puzzle'],
233
+ },
234
+ };
235
+
236
+ export const WithInitialItems: Story = {
237
+ args: {
238
+ debug: true,
239
+ items: [
240
+ createStatic(ChessType, {
241
+ fen: 'rnbqkbnr/pppppppp/8/8/8/8/PPPPPPPP/RNBQKBNR w KQkq - 0 1',
242
+ }),
243
+ ],
244
+ },
245
+ };
@@ -0,0 +1,67 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import React, { type FC, useCallback } from 'react';
6
+
7
+ import { invariant } from '@dxos/invariant';
8
+ import { log } from '@dxos/log';
9
+ import { getSpace } from '@dxos/react-client/echo';
10
+ import { type ThemedClassName } from '@dxos/react-ui';
11
+
12
+ import { Thread } from './Thread';
13
+ import { useChatProcessor, useMessageQueue } from '../../hooks';
14
+ import { type AIChatType } from '../../types';
15
+
16
+ // TODO(burdon): Since this only wraps Thread, just separate out hook?
17
+ export const ThreadContainer: FC<ThemedClassName<{ chat?: AIChatType; onOpenChange?: (open: boolean) => void }>> = ({
18
+ classNames,
19
+ chat,
20
+ onOpenChange,
21
+ }) => {
22
+ const space = getSpace(chat);
23
+ const processor = useChatProcessor(space);
24
+ const messageQueue = useMessageQueue(chat);
25
+ const messages = [...(messageQueue?.items ?? []), ...processor.messages.value];
26
+
27
+ const handleSubmit = useCallback(
28
+ (text: string) => {
29
+ // Don't accept input if still processing.
30
+ if (processor.streaming.value) {
31
+ log.warn('still processing');
32
+ return false;
33
+ }
34
+
35
+ onOpenChange?.(true);
36
+
37
+ invariant(messageQueue);
38
+ void processor.request(text, {
39
+ history: messageQueue.items,
40
+ onComplete: (messages) => {
41
+ messageQueue.append(messages);
42
+ },
43
+ });
44
+
45
+ return true;
46
+ },
47
+ [processor, messageQueue, onOpenChange],
48
+ );
49
+
50
+ const handleCancel = useCallback(() => {
51
+ if (processor.streaming.value) {
52
+ void processor.cancel();
53
+ }
54
+ }, [processor]);
55
+
56
+ return (
57
+ <Thread
58
+ classNames={classNames}
59
+ space={space}
60
+ messages={messages}
61
+ processing={processor.streaming.value}
62
+ error={processor.error.value}
63
+ onSubmit={handleSubmit}
64
+ onCancel={handleCancel}
65
+ />
66
+ );
67
+ };
@@ -0,0 +1,187 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import React, { type PropsWithChildren, type FC } from 'react';
6
+
7
+ import { type MessageContentBlock, type Message } from '@dxos/artifact';
8
+ import { invariant } from '@dxos/invariant';
9
+ import { type Space } from '@dxos/react-client/echo';
10
+ import { Button, ButtonGroup, Icon, IconButton, type ThemedClassName } from '@dxos/react-ui';
11
+ import {
12
+ MarkdownViewer,
13
+ ToggleContainer as NativeToggleContainer,
14
+ type ToggleContainerProps,
15
+ } from '@dxos/react-ui-components';
16
+ import { Json } from '@dxos/react-ui-syntax-highlighter';
17
+ import { mx } from '@dxos/react-ui-theme';
18
+ import { safeParseJson } from '@dxos/util';
19
+
20
+ import { ToolBlock, isToolMessage } from './ToolInvocations';
21
+ import { ToolboxContainer } from '../Toolbox';
22
+
23
+ const userClassNames = 'bg-[--user-fill]';
24
+ const panelClassNames = 'flex flex-col w-full bg-groupSurface rounded-md';
25
+
26
+ const ToggleContainer = (props: ToggleContainerProps) => {
27
+ return <NativeToggleContainer {...props} classNames={mx(panelClassNames, props.classNames)} />;
28
+ };
29
+
30
+ const MessageContainer = ({ children, classNames, user }: ThemedClassName<PropsWithChildren<{ user?: boolean }>>) => {
31
+ return (
32
+ <div role='list-item' className={mx('flex w-full', user && 'justify-end', classNames)}>
33
+ <div className={mx(user ? ['px-2 py-1 rounded-md', userClassNames] : 'w-full')}>{children}</div>
34
+ </div>
35
+ );
36
+ };
37
+
38
+ export type ThreadMessageProps = ThemedClassName<{
39
+ space?: Space;
40
+ message: Message;
41
+ debug?: boolean;
42
+ onPrompt?: (text: string) => void;
43
+ onDelete?: (id: string) => void;
44
+ }>;
45
+
46
+ export const ThreadMessage: FC<ThreadMessageProps> = ({ classNames, space, message, onPrompt }) => {
47
+ const { role, content = [] } = message;
48
+
49
+ // TODO(burdon): Restructure types to make check unnecessary.
50
+ if (isToolMessage(message)) {
51
+ return (
52
+ <MessageContainer classNames={classNames}>
53
+ <ToolBlock space={space} classNames={panelClassNames} message={message} />
54
+ </MessageContainer>
55
+ );
56
+ }
57
+
58
+ return (
59
+ <div role='none' className='flex flex-col gap-2'>
60
+ {content.map((block, idx) => (
61
+ <MessageContainer key={idx} classNames={classNames} user={block.type === 'text' && role === 'user'}>
62
+ <Block space={space} block={block} onPrompt={onPrompt} />
63
+ </MessageContainer>
64
+ ))}
65
+ </div>
66
+ );
67
+ };
68
+
69
+ const Block: FC<{ space?: Space; block: MessageContentBlock; onPrompt?: (text: string) => void }> = ({
70
+ space,
71
+ block,
72
+ onPrompt,
73
+ }) => {
74
+ const Component = components[block.type] ?? components.default;
75
+ return <Component space={space} block={block} onPrompt={onPrompt} />;
76
+ };
77
+
78
+ type BlockComponent = FC<{ space?: Space; block: MessageContentBlock; onPrompt?: (text: string) => void }>;
79
+
80
+ const components: Record<string, BlockComponent> = {
81
+ //
82
+ // Text
83
+ //
84
+ ['text' as const]: ({ block }) => {
85
+ invariant(block.type === 'text');
86
+ // const [open, setOpen] = useState(block.disposition === 'cot' && block.pending);
87
+ const title = block.disposition ? titles[block.disposition] : undefined;
88
+ if (!title) {
89
+ return <MarkdownViewer content={block.text} />;
90
+ }
91
+
92
+ // TOOD(burdon): Store last time user opened/closed COT.
93
+ // Autoclose when streaming ends.
94
+ // useEffect(() => {
95
+ // if (block.disposition === 'cot' && !block.pending) {
96
+ // setOpen(false);
97
+ // }
98
+ // }, [block.disposition, block.pending]);
99
+
100
+ return (
101
+ <ToggleContainer
102
+ // open={open}
103
+ defaultOpen={block.disposition === 'cot' && block.pending}
104
+ title={title}
105
+ icon={
106
+ block.pending ? (
107
+ <Icon icon={'ph--circle-notch--regular'} classNames='text-subdued ml-2 animate-spin' size={4} />
108
+ ) : undefined
109
+ }
110
+ >
111
+ <MarkdownViewer
112
+ content={block.text}
113
+ classNames={['pbe-2', block.disposition === 'cot' && 'text-sm text-subdued']}
114
+ />
115
+ </ToggleContainer>
116
+ );
117
+ },
118
+
119
+ //
120
+ // JSON
121
+ //
122
+ ['json' as const]: ({ space, block, onPrompt }) => {
123
+ invariant(block.type === 'json');
124
+
125
+ switch (block.disposition) {
126
+ case 'tool_list': {
127
+ return (
128
+ <ToggleContainer title={titles[block.disposition]} defaultOpen={true}>
129
+ <ToolboxContainer space={space} />
130
+ </ToggleContainer>
131
+ );
132
+ }
133
+
134
+ case 'suggest': {
135
+ const { text = '' }: { text: string } = safeParseJson(block.json ?? '{}') ?? ({} as any);
136
+ return <IconButton icon='ph--lightning--regular' label={text} onClick={() => onPrompt?.(text)} />;
137
+ }
138
+
139
+ case 'select': {
140
+ const { options = [] }: { options: string[] } = safeParseJson(block.json ?? '{}') ?? ({} as any);
141
+ return (
142
+ <ButtonGroup>
143
+ {options.map((option) => (
144
+ <Button key={option} onClick={() => onPrompt?.(option)}>
145
+ {option}
146
+ </Button>
147
+ ))}
148
+ </ButtonGroup>
149
+ );
150
+ }
151
+
152
+ default: {
153
+ const title = block.disposition ? titles[block.disposition] : undefined;
154
+ return (
155
+ <ToggleContainer title={title ?? 'JSON'}>
156
+ <Json data={safeParseJson(block.json ?? block)} classNames='!p-1 text-xs' />
157
+ </ToggleContainer>
158
+ );
159
+ }
160
+ }
161
+ },
162
+
163
+ //
164
+ // Default
165
+ //
166
+ default: ({ block }) => {
167
+ let title = titles[block.type];
168
+ if (block.type === 'tool_use') {
169
+ title = `Tool [${block.name}]`; // TODO(burdon): Get label from tool.
170
+ }
171
+
172
+ return (
173
+ <ToggleContainer title={title ?? 'JSON'}>
174
+ <Json data={block} classNames='!p-1 text-xs' />
175
+ </ToggleContainer>
176
+ );
177
+ },
178
+ };
179
+
180
+ // TODO(burdon): Translations.
181
+ const titles: Record<string, string> = {
182
+ ['cot' as const]: 'Chain of thought',
183
+ ['artifact' as const]: 'Artifact',
184
+ ['tool_use' as const]: 'Tool request',
185
+ ['tool_result' as const]: 'Tool result',
186
+ ['tool_list' as const]: 'Tools',
187
+ };
@@ -0,0 +1,86 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import React, { type FC, useEffect, useRef, useState } from 'react';
6
+
7
+ import { type Message, type MessageContentBlock } from '@dxos/artifact';
8
+ import { log } from '@dxos/log';
9
+ import { type ThemedClassName } from '@dxos/react-ui';
10
+ import { NumericTabs, StatusRoll, ToggleContainer } from '@dxos/react-ui-components';
11
+ import { Json } from '@dxos/react-ui-syntax-highlighter';
12
+
13
+ import { type ThreadMessageProps } from './ThreadMessage';
14
+
15
+ export const isToolMessage = (message: Message) => {
16
+ return message.content.some((block) => block.type === 'tool_use' || block.type === 'tool_result');
17
+ };
18
+
19
+ export const ToolBlock: FC<ThemedClassName<ThreadMessageProps>> = ({ classNames, message }) => {
20
+ const { content = [] } = message;
21
+
22
+ let request: (MessageContentBlock & { type: 'tool_use' }) | undefined;
23
+ const blocks = content.filter((block) => block.type === 'tool_use' || block.type === 'tool_result');
24
+ const items = blocks.map((block) => {
25
+ switch (block.type) {
26
+ case 'tool_use': {
27
+ request = block;
28
+ // TODO(burdon): Get plugin name.
29
+ return { title: `Calling ${block.name}...`, block };
30
+ }
31
+
32
+ case 'tool_result': {
33
+ if (!request) {
34
+ log.warn('unexpected message', { block });
35
+ return { title: 'Error', block };
36
+ }
37
+
38
+ return { title: `Processed ${request.name}`, block };
39
+ }
40
+
41
+ default: {
42
+ request = undefined;
43
+ return { title: 'Error', block };
44
+ }
45
+ }
46
+ });
47
+
48
+ return <ToolContainer classNames={classNames} items={items} />;
49
+ };
50
+
51
+ export const ToolContainer: FC<ThemedClassName<{ items: { title: string; block: any }[] }>> = ({
52
+ classNames,
53
+ items,
54
+ }) => {
55
+ const lines = items.map((item) => item.title);
56
+ const tabsRef = useRef<HTMLDivElement>(null);
57
+ const [selected, setSelected] = useState(0);
58
+ const [open, setOpen] = useState(false);
59
+ useEffect(() => {
60
+ if (open) {
61
+ tabsRef.current?.focus();
62
+ }
63
+ }, [open]);
64
+
65
+ const handleSelect = (index: number) => {
66
+ if (index === selected) {
67
+ setOpen(false);
68
+ } else {
69
+ setSelected(index);
70
+ }
71
+ };
72
+
73
+ return (
74
+ <ToggleContainer
75
+ classNames={['flex flex-col', classNames]}
76
+ title={<StatusRoll lines={lines} autoAdvance />}
77
+ open={open}
78
+ onChangeOpen={setOpen}
79
+ >
80
+ <div className='grid grid-cols-[32px_1fr]'>
81
+ <NumericTabs ref={tabsRef} length={items.length} selected={selected} onSelect={handleSelect} />
82
+ <Json data={items[selected].block} classNames='!p-1 text-xs' />
83
+ </div>
84
+ </ToggleContainer>
85
+ );
86
+ };
@@ -0,0 +1,6 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ export * from './Thread';
6
+ export * from './ThreadContainer';