@dxos/plugin-assistant 0.8.1-staging.391c573 → 0.8.1-staging.5be625a

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 (207) hide show
  1. package/dist/lib/browser/{AssistantDialog-CM33SOCT.mjs → AssistantDialog-EIUUSXDS.mjs} +11 -10
  2. package/dist/lib/browser/AssistantDialog-EIUUSXDS.mjs.map +7 -0
  3. package/dist/lib/browser/{ChatContainer-VW73VNUY.mjs → ChatContainer-NHHA6CSP.mjs} +5 -5
  4. package/dist/lib/browser/TemplateContainer-7IQ6V5AD.mjs +78 -0
  5. package/dist/lib/browser/TemplateContainer-7IQ6V5AD.mjs.map +7 -0
  6. package/dist/lib/browser/ai-client-BAPVMSNX.mjs +35 -0
  7. package/dist/lib/browser/ai-client-BAPVMSNX.mjs.map +7 -0
  8. package/dist/lib/browser/{app-graph-builder-4DIOTGOG.mjs → app-graph-builder-DTCUWBKB.mjs} +57 -6
  9. package/dist/lib/browser/app-graph-builder-DTCUWBKB.mjs.map +7 -0
  10. package/dist/lib/browser/{chunk-M2SADJVS.mjs → chunk-25APJ3ZK.mjs} +34 -12
  11. package/dist/lib/browser/chunk-25APJ3ZK.mjs.map +7 -0
  12. package/dist/lib/browser/{chunk-EUMPBC4T.mjs → chunk-6FTPLBSC.mjs} +2 -2
  13. package/dist/lib/browser/{chunk-NV7SVHMV.mjs → chunk-AF7VQAKS.mjs} +1 -1
  14. package/dist/lib/browser/{chunk-NV7SVHMV.mjs.map → chunk-AF7VQAKS.mjs.map} +2 -2
  15. package/dist/lib/browser/{chunk-WVME6K4X.mjs → chunk-KGIACFAX.mjs} +333 -205
  16. package/dist/lib/browser/chunk-KGIACFAX.mjs.map +7 -0
  17. package/dist/lib/browser/{chunk-NTLTGYYS.mjs → chunk-SVUCJXGN.mjs} +2 -2
  18. package/dist/lib/browser/chunk-SVUCJXGN.mjs.map +7 -0
  19. package/dist/lib/browser/{chunk-FPXC3LKK.mjs → chunk-X6ALDUA5.mjs} +25 -3
  20. package/dist/lib/browser/{chunk-FPXC3LKK.mjs.map → chunk-X6ALDUA5.mjs.map} +3 -3
  21. package/dist/lib/browser/index.mjs +25 -27
  22. package/dist/lib/browser/index.mjs.map +3 -3
  23. package/dist/lib/browser/{intent-resolver-FQRN6747.mjs → intent-resolver-U57FXP3I.mjs} +6 -3
  24. package/dist/lib/browser/{intent-resolver-FQRN6747.mjs.map → intent-resolver-U57FXP3I.mjs.map} +3 -3
  25. package/dist/lib/browser/meta.json +1 -1
  26. package/dist/lib/browser/{react-surface-ZT6EQ7FJ.mjs → react-surface-YNN5NYJW.mjs} +10 -9
  27. package/dist/lib/browser/react-surface-YNN5NYJW.mjs.map +7 -0
  28. package/dist/lib/browser/{settings-VVQUGG56.mjs → settings-VAW6UWFL.mjs} +3 -3
  29. package/dist/lib/browser/types/index.mjs +6 -2
  30. package/dist/lib/node/{AssistantDialog-5PVOCPE2.cjs → AssistantDialog-GT7R7MKH.cjs} +17 -16
  31. package/dist/lib/node/AssistantDialog-GT7R7MKH.cjs.map +7 -0
  32. package/dist/lib/node/{ChatContainer-I45I3ICI.cjs → ChatContainer-JUZOEZZ7.cjs} +10 -10
  33. package/dist/lib/node/TemplateContainer-VPAZRFQA.cjs +104 -0
  34. package/dist/lib/node/TemplateContainer-VPAZRFQA.cjs.map +7 -0
  35. package/dist/lib/node/{ai-client-6CY35ZIR.cjs → ai-client-5ESLYXAV.cjs} +19 -8
  36. package/dist/lib/node/ai-client-5ESLYXAV.cjs.map +7 -0
  37. package/dist/lib/node/{app-graph-builder-7O4U4RXH.cjs → app-graph-builder-3P6WSON2.cjs} +63 -13
  38. package/dist/lib/node/app-graph-builder-3P6WSON2.cjs.map +7 -0
  39. package/dist/lib/node/{chunk-HFOUAFOW.cjs → chunk-G2HY3UJ4.cjs} +40 -18
  40. package/dist/lib/node/chunk-G2HY3UJ4.cjs.map +7 -0
  41. package/dist/lib/node/{chunk-JWUHUQ56.cjs → chunk-JAB6KLPP.cjs} +334 -210
  42. package/dist/lib/node/chunk-JAB6KLPP.cjs.map +7 -0
  43. package/dist/lib/node/{chunk-34WE2FD2.cjs → chunk-KLSNCP34.cjs} +31 -7
  44. package/dist/lib/node/chunk-KLSNCP34.cjs.map +7 -0
  45. package/dist/lib/node/{chunk-NV4TQQSU.cjs → chunk-N3SW6DJ6.cjs} +6 -6
  46. package/dist/lib/node/{chunk-GNPXCHFT.cjs → chunk-U6J2GO7I.cjs} +4 -4
  47. package/dist/lib/node/{chunk-GNPXCHFT.cjs.map → chunk-U6J2GO7I.cjs.map} +2 -2
  48. package/dist/lib/node/{chunk-3WXG6WA6.cjs → chunk-VRXFIS4X.cjs} +6 -6
  49. package/dist/lib/node/chunk-VRXFIS4X.cjs.map +7 -0
  50. package/dist/lib/node/index.cjs +75 -77
  51. package/dist/lib/node/index.cjs.map +3 -3
  52. package/dist/lib/node/{intent-resolver-Z37RNNMC.cjs → intent-resolver-YIFAMM3B.cjs} +13 -10
  53. package/dist/lib/node/{intent-resolver-Z37RNNMC.cjs.map → intent-resolver-YIFAMM3B.cjs.map} +3 -3
  54. package/dist/lib/node/meta.json +1 -1
  55. package/dist/lib/node/{react-surface-KFXP3GGK.cjs → react-surface-IDGIN55C.cjs} +25 -24
  56. package/dist/lib/node/react-surface-IDGIN55C.cjs.map +7 -0
  57. package/dist/lib/node/{settings-TJHHVI6B.cjs → settings-2FEYGLYU.cjs} +8 -8
  58. package/dist/lib/node/types/index.cjs +15 -11
  59. package/dist/lib/node/types/index.cjs.map +2 -2
  60. package/dist/lib/node-esm/{AssistantDialog-Q2ZBOHOV.mjs → AssistantDialog-JMBFM6QH.mjs} +11 -10
  61. package/dist/lib/node-esm/AssistantDialog-JMBFM6QH.mjs.map +7 -0
  62. package/dist/lib/node-esm/{ChatContainer-5ULBF3GW.mjs → ChatContainer-ZNN5CMVL.mjs} +5 -5
  63. package/dist/lib/node-esm/TemplateContainer-YLA6BJY6.mjs +79 -0
  64. package/dist/lib/node-esm/TemplateContainer-YLA6BJY6.mjs.map +7 -0
  65. package/dist/lib/node-esm/ai-client-XYZ5N7CR.mjs +36 -0
  66. package/dist/lib/node-esm/ai-client-XYZ5N7CR.mjs.map +7 -0
  67. package/dist/lib/node-esm/{app-graph-builder-SX3SREXF.mjs → app-graph-builder-PMAQLTTN.mjs} +57 -6
  68. package/dist/lib/node-esm/app-graph-builder-PMAQLTTN.mjs.map +7 -0
  69. package/dist/lib/node-esm/{chunk-ICQN3TDS.mjs → chunk-2CIYX3SD.mjs} +2 -2
  70. package/dist/lib/node-esm/chunk-2CIYX3SD.mjs.map +7 -0
  71. package/dist/lib/node-esm/{chunk-LBQGJE5T.mjs → chunk-J63VQFQO.mjs} +2 -2
  72. package/dist/lib/node-esm/{chunk-7SV6X6XU.mjs → chunk-N6BVC2C2.mjs} +1 -1
  73. package/dist/lib/node-esm/{chunk-7SV6X6XU.mjs.map → chunk-N6BVC2C2.mjs.map} +2 -2
  74. package/dist/lib/node-esm/{chunk-LELXJPGJ.mjs → chunk-NMMRHHAR.mjs} +25 -3
  75. package/dist/lib/node-esm/{chunk-LELXJPGJ.mjs.map → chunk-NMMRHHAR.mjs.map} +3 -3
  76. package/dist/lib/node-esm/{chunk-ASMI7PQ2.mjs → chunk-PK5JCOYB.mjs} +333 -205
  77. package/dist/lib/node-esm/chunk-PK5JCOYB.mjs.map +7 -0
  78. package/dist/lib/node-esm/{chunk-2OUUYMBT.mjs → chunk-SMUINDXQ.mjs} +34 -12
  79. package/dist/lib/node-esm/chunk-SMUINDXQ.mjs.map +7 -0
  80. package/dist/lib/node-esm/index.mjs +25 -27
  81. package/dist/lib/node-esm/index.mjs.map +3 -3
  82. package/dist/lib/node-esm/{intent-resolver-LOTXWV4J.mjs → intent-resolver-SQ4HLL5L.mjs} +6 -3
  83. package/dist/lib/node-esm/{intent-resolver-LOTXWV4J.mjs.map → intent-resolver-SQ4HLL5L.mjs.map} +3 -3
  84. package/dist/lib/node-esm/meta.json +1 -1
  85. package/dist/lib/node-esm/{react-surface-NZNNP2QI.mjs → react-surface-424JZTZ4.mjs} +10 -9
  86. package/dist/lib/node-esm/react-surface-424JZTZ4.mjs.map +7 -0
  87. package/dist/lib/node-esm/{settings-BVWR244C.mjs → settings-VHR5KT4J.mjs} +3 -3
  88. package/dist/lib/node-esm/types/index.mjs +6 -2
  89. package/dist/types/src/AssistantPlugin.d.ts.map +1 -1
  90. package/dist/types/src/capabilities/ai-client.d.ts +2 -1
  91. package/dist/types/src/capabilities/ai-client.d.ts.map +1 -1
  92. package/dist/types/src/capabilities/app-graph-builder.d.ts.map +1 -1
  93. package/dist/types/src/capabilities/capabilities.d.ts +2 -1
  94. package/dist/types/src/capabilities/capabilities.d.ts.map +1 -1
  95. package/dist/types/src/capabilities/index.d.ts +1 -1
  96. package/dist/types/src/capabilities/index.d.ts.map +1 -1
  97. package/dist/types/src/capabilities/react-surface.d.ts.map +1 -1
  98. package/dist/types/src/components/AmbientDialog/AmbientDialog.d.ts +4 -3
  99. package/dist/types/src/components/AmbientDialog/AmbientDialog.d.ts.map +1 -1
  100. package/dist/types/src/components/AssistantDialog.d.ts.map +1 -1
  101. package/dist/types/src/components/AssistantSettings/AssistantSettings.d.ts.map +1 -1
  102. package/dist/types/src/components/Prompt/Prompt.d.ts +3 -0
  103. package/dist/types/src/components/Prompt/Prompt.d.ts.map +1 -1
  104. package/dist/types/src/components/Prompt/Prompt.stories.d.ts +1 -0
  105. package/dist/types/src/components/Prompt/Prompt.stories.d.ts.map +1 -1
  106. package/dist/types/src/components/Prompt/PromptBar.d.ts +2 -2
  107. package/dist/types/src/components/Prompt/PromptBar.d.ts.map +1 -1
  108. package/dist/types/src/components/Prompt/references.d.ts +30 -0
  109. package/dist/types/src/components/Prompt/references.d.ts.map +1 -0
  110. package/dist/types/src/components/TemplateContainer.d.ts.map +1 -1
  111. package/dist/types/src/components/Thread/Thread.d.ts +11 -1
  112. package/dist/types/src/components/Thread/Thread.d.ts.map +1 -1
  113. package/dist/types/src/components/Thread/ThreadContainer.d.ts +1 -0
  114. package/dist/types/src/components/Thread/ThreadContainer.d.ts.map +1 -1
  115. package/dist/types/src/components/Thread/ThreadContainer.stories.d.ts.map +1 -1
  116. package/dist/types/src/components/Thread/ThreadMessage.d.ts.map +1 -1
  117. package/dist/types/src/hooks/email.d.ts.map +1 -1
  118. package/dist/types/src/hooks/index.d.ts +1 -0
  119. package/dist/types/src/hooks/index.d.ts.map +1 -1
  120. package/dist/types/src/hooks/processor.d.ts +7 -15
  121. package/dist/types/src/hooks/processor.d.ts.map +1 -1
  122. package/dist/types/src/hooks/useChatProcessor.d.ts +9 -2
  123. package/dist/types/src/hooks/useChatProcessor.d.ts.map +1 -1
  124. package/dist/types/src/hooks/useContextProvider.d.ts +17 -0
  125. package/dist/types/src/hooks/useContextProvider.d.ts.map +1 -0
  126. package/dist/types/src/hooks/useMessageQueue.d.ts +4 -4
  127. package/dist/types/src/hooks/useMessageQueue.d.ts.map +1 -1
  128. package/dist/types/src/meta.d.ts +2 -8
  129. package/dist/types/src/meta.d.ts.map +1 -1
  130. package/dist/types/src/testing/test-functions.d.ts.map +1 -1
  131. package/dist/types/src/translations.d.ts +23 -12
  132. package/dist/types/src/translations.d.ts.map +1 -1
  133. package/dist/types/src/types/template.d.ts +36 -0
  134. package/dist/types/src/types/template.d.ts.map +1 -1
  135. package/dist/types/src/types/types.d.ts +3 -1
  136. package/dist/types/src/types/types.d.ts.map +1 -1
  137. package/package.json +56 -55
  138. package/src/AssistantPlugin.tsx +11 -4
  139. package/src/capabilities/ai-client.ts +23 -5
  140. package/src/capabilities/app-graph-builder.ts +48 -2
  141. package/src/capabilities/capabilities.ts +3 -1
  142. package/src/capabilities/intent-resolver.ts +1 -1
  143. package/src/capabilities/react-surface.tsx +3 -2
  144. package/src/components/AmbientDialog/AmbientDialog.tsx +11 -11
  145. package/src/components/AssistantDialog.tsx +7 -1
  146. package/src/components/AssistantSettings/AssistantSettings.tsx +39 -5
  147. package/src/components/Prompt/Prompt.stories.tsx +34 -0
  148. package/src/components/Prompt/Prompt.tsx +28 -18
  149. package/src/components/Prompt/PromptBar.tsx +3 -1
  150. package/src/components/Prompt/references.ts +180 -0
  151. package/src/components/TemplateContainer.tsx +79 -4
  152. package/src/components/TemplateEditor/TemplateEditor.stories.tsx +1 -1
  153. package/src/components/TemplateEditor/TemplateForm.stories.tsx +1 -1
  154. package/src/components/Thread/Thread.tsx +21 -0
  155. package/src/components/Thread/ThreadContainer.stories.tsx +5 -10
  156. package/src/components/Thread/ThreadContainer.tsx +7 -4
  157. package/src/components/Thread/ThreadMessage.tsx +17 -9
  158. package/src/hooks/email.ts +3 -3
  159. package/src/hooks/index.ts +1 -0
  160. package/src/hooks/processor.ts +57 -116
  161. package/src/hooks/useChatProcessor.tsx +24 -16
  162. package/src/hooks/useContextProvider.ts +55 -0
  163. package/src/hooks/useMessageQueue.ts +2 -4
  164. package/src/meta.ts +2 -2
  165. package/src/testing/test-functions.ts +2 -2
  166. package/src/translations.ts +10 -5
  167. package/src/types/template.ts +22 -0
  168. package/src/types/types.ts +3 -1
  169. package/dist/lib/browser/AssistantDialog-CM33SOCT.mjs.map +0 -7
  170. package/dist/lib/browser/TemplateContainer-B7MQNUPY.mjs +0 -23
  171. package/dist/lib/browser/TemplateContainer-B7MQNUPY.mjs.map +0 -7
  172. package/dist/lib/browser/ai-client-HZPB7E6M.mjs +0 -22
  173. package/dist/lib/browser/ai-client-HZPB7E6M.mjs.map +0 -7
  174. package/dist/lib/browser/app-graph-builder-4DIOTGOG.mjs.map +0 -7
  175. package/dist/lib/browser/chunk-M2SADJVS.mjs.map +0 -7
  176. package/dist/lib/browser/chunk-NTLTGYYS.mjs.map +0 -7
  177. package/dist/lib/browser/chunk-WVME6K4X.mjs.map +0 -7
  178. package/dist/lib/browser/react-surface-ZT6EQ7FJ.mjs.map +0 -7
  179. package/dist/lib/node/AssistantDialog-5PVOCPE2.cjs.map +0 -7
  180. package/dist/lib/node/TemplateContainer-R4BZZP3E.cjs +0 -53
  181. package/dist/lib/node/TemplateContainer-R4BZZP3E.cjs.map +0 -7
  182. package/dist/lib/node/ai-client-6CY35ZIR.cjs.map +0 -7
  183. package/dist/lib/node/app-graph-builder-7O4U4RXH.cjs.map +0 -7
  184. package/dist/lib/node/chunk-34WE2FD2.cjs.map +0 -7
  185. package/dist/lib/node/chunk-3WXG6WA6.cjs.map +0 -7
  186. package/dist/lib/node/chunk-HFOUAFOW.cjs.map +0 -7
  187. package/dist/lib/node/chunk-JWUHUQ56.cjs.map +0 -7
  188. package/dist/lib/node/react-surface-KFXP3GGK.cjs.map +0 -7
  189. package/dist/lib/node-esm/AssistantDialog-Q2ZBOHOV.mjs.map +0 -7
  190. package/dist/lib/node-esm/TemplateContainer-WSHTZBB5.mjs +0 -24
  191. package/dist/lib/node-esm/TemplateContainer-WSHTZBB5.mjs.map +0 -7
  192. package/dist/lib/node-esm/ai-client-GSA67EIB.mjs +0 -23
  193. package/dist/lib/node-esm/ai-client-GSA67EIB.mjs.map +0 -7
  194. package/dist/lib/node-esm/app-graph-builder-SX3SREXF.mjs.map +0 -7
  195. package/dist/lib/node-esm/chunk-2OUUYMBT.mjs.map +0 -7
  196. package/dist/lib/node-esm/chunk-ASMI7PQ2.mjs.map +0 -7
  197. package/dist/lib/node-esm/chunk-ICQN3TDS.mjs.map +0 -7
  198. package/dist/lib/node-esm/react-surface-NZNNP2QI.mjs.map +0 -7
  199. /package/dist/lib/browser/{ChatContainer-VW73VNUY.mjs.map → ChatContainer-NHHA6CSP.mjs.map} +0 -0
  200. /package/dist/lib/browser/{chunk-EUMPBC4T.mjs.map → chunk-6FTPLBSC.mjs.map} +0 -0
  201. /package/dist/lib/browser/{settings-VVQUGG56.mjs.map → settings-VAW6UWFL.mjs.map} +0 -0
  202. /package/dist/lib/node/{ChatContainer-I45I3ICI.cjs.map → ChatContainer-JUZOEZZ7.cjs.map} +0 -0
  203. /package/dist/lib/node/{chunk-NV4TQQSU.cjs.map → chunk-N3SW6DJ6.cjs.map} +0 -0
  204. /package/dist/lib/node/{settings-TJHHVI6B.cjs.map → settings-2FEYGLYU.cjs.map} +0 -0
  205. /package/dist/lib/node-esm/{ChatContainer-5ULBF3GW.mjs.map → ChatContainer-ZNN5CMVL.mjs.map} +0 -0
  206. /package/dist/lib/node-esm/{chunk-LBQGJE5T.mjs.map → chunk-J63VQFQO.mjs.map} +0 -0
  207. /package/dist/lib/node-esm/{settings-BVWR244C.mjs.map → settings-VHR5KT4J.mjs.map} +0 -0
@@ -4,13 +4,16 @@
4
4
 
5
5
  import React from 'react';
6
6
 
7
- import { DEFAULT_LLM_MODELS } from '@dxos/assistant';
7
+ import { DEFAULT_EDGE_MODELS, DEFAULT_OLLAMA_MODELS } from '@dxos/assistant';
8
8
  import { Input, Select, useTranslation } from '@dxos/react-ui';
9
9
  import { DeprecatedFormContainer, DeprecatedFormInput } from '@dxos/react-ui-form';
10
10
 
11
11
  import { ASSISTANT_PLUGIN } from '../../meta';
12
12
  import { type AssistantSettingsProps } from '../../types';
13
13
 
14
+ // TODO(burdon): Factor out.
15
+ const DEFAULT_VALUE = '__default';
16
+
14
17
  export const AssistantSettings = ({ settings }: { settings: AssistantSettingsProps }) => {
15
18
  const { t } = useTranslation(ASSISTANT_PLUGIN);
16
19
 
@@ -23,18 +26,49 @@ export const AssistantSettings = ({ settings }: { settings: AssistantSettingsPro
23
26
  />
24
27
  </DeprecatedFormInput>
25
28
 
26
- <DeprecatedFormInput label={t('settings llm model label')}>
29
+ <DeprecatedFormInput label={t('settings llm provider label')}>
30
+ <Input.Switch
31
+ checked={settings.llmProvider === 'ollama'}
32
+ onCheckedChange={(checked) => (settings.llmProvider = checked ? 'ollama' : 'edge')}
33
+ />
34
+ </DeprecatedFormInput>
35
+
36
+ <DeprecatedFormInput label={t('settings edge llm model label')}>
37
+ <Select.Root
38
+ value={settings.edgeModel ?? DEFAULT_VALUE}
39
+ onValueChange={(value) => {
40
+ settings.edgeModel = value === DEFAULT_VALUE ? undefined : value;
41
+ }}
42
+ >
43
+ <Select.TriggerButton placeholder={t('settings default llm model label')} />
44
+ <Select.Portal>
45
+ <Select.Content>
46
+ <Select.Viewport>
47
+ <Select.Option value={DEFAULT_VALUE}>{t('settings default label')}</Select.Option>
48
+ {DEFAULT_EDGE_MODELS.map((model) => (
49
+ <Select.Option key={model} value={model}>
50
+ {model}
51
+ </Select.Option>
52
+ ))}
53
+ </Select.Viewport>
54
+ </Select.Content>
55
+ </Select.Portal>
56
+ </Select.Root>
57
+ </DeprecatedFormInput>
58
+
59
+ <DeprecatedFormInput label={t('settings ollama llm model label')}>
27
60
  <Select.Root
28
- value={settings.llmModel ?? 'default'}
61
+ value={settings.ollamaModel ?? DEFAULT_VALUE}
29
62
  onValueChange={(value) => {
30
- settings.llmModel = value;
63
+ settings.ollamaModel = value === DEFAULT_VALUE ? undefined : value;
31
64
  }}
32
65
  >
33
66
  <Select.TriggerButton placeholder={t('settings default llm model label')} />
34
67
  <Select.Portal>
35
68
  <Select.Content>
36
69
  <Select.Viewport>
37
- {DEFAULT_LLM_MODELS.map((model) => (
70
+ <Select.Option value={DEFAULT_VALUE}>{t('settings default label')}</Select.Option>
71
+ {DEFAULT_OLLAMA_MODELS.map((model) => (
38
72
  <Select.Option key={model} value={model}>
39
73
  {model}
40
74
  </Select.Option>
@@ -11,6 +11,7 @@ import { withTheme, withLayout } from '@dxos/storybook-utils';
11
11
 
12
12
  import { Prompt } from './Prompt';
13
13
  import { PromptBar } from './PromptBar';
14
+ import type { ReferenceData } from './references';
14
15
  import translations from '../../translations';
15
16
 
16
17
  const meta: Meta<typeof Prompt> = {
@@ -77,3 +78,36 @@ export const Toolbar: Story = {
77
78
  );
78
79
  },
79
80
  };
81
+
82
+ export const Includes: Story = {
83
+ args: {
84
+ classNames: 'w-96 p-4 rounded outline outline-gray-200',
85
+ references: {
86
+ getReferences: async ({ query }) => {
87
+ const res = references.filter((i) => i.label.toLowerCase().startsWith(query.toLowerCase()));
88
+ console.log('getReferences', { query, res });
89
+ return res;
90
+ },
91
+ resolveReference: async ({ uri }) => {
92
+ const res = references.find((i) => i.uri === uri);
93
+ console.log('resolveReference', { uri, res });
94
+ return res ?? null;
95
+ },
96
+ },
97
+ },
98
+ };
99
+
100
+ const references: ReferenceData[] = [
101
+ {
102
+ uri: 'dxn:echo:@:AAAAAAAA',
103
+ label: 'Meeting Notes',
104
+ },
105
+ {
106
+ uri: 'dxn:echo:@:BBBBBBBB',
107
+ label: 'Project Plan',
108
+ },
109
+ {
110
+ uri: 'dxn:echo:@:CCCCCCCC',
111
+ label: 'Meeting Plan',
112
+ },
113
+ ];
@@ -2,6 +2,7 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
+ import { Prec } from '@codemirror/state';
5
6
  import React, { forwardRef, useImperativeHandle } from 'react';
6
7
 
7
8
  import { type ThemedClassName, useThemeContext } from '@dxos/react-ui';
@@ -16,6 +17,7 @@ import {
16
17
  import { mx } from '@dxos/react-ui-theme';
17
18
 
18
19
  import { createAutocompleteExtension, type AutocompleteOptions } from './autocomplete';
20
+ import { promptReferences, type ReferencesProvider } from './references';
19
21
 
20
22
  // TODO(burdon): Handle object references.
21
23
 
@@ -27,16 +29,21 @@ export interface PromptController {
27
29
  export type PromptProps = ThemedClassName<
28
30
  {
29
31
  onOpenChange?: (open: boolean) => void;
32
+ references?: ReferencesProvider;
30
33
  } & AutocompleteOptions &
31
34
  Pick<UseTextEditorProps, 'autoFocus'> &
32
35
  Pick<BasicExtensionsOptions, 'lineWrapping' | 'placeholder'>
33
36
  >;
34
37
 
35
38
  export const Prompt = forwardRef<PromptController, PromptProps>(
36
- ({ classNames, autoFocus, lineWrapping = false, placeholder, onSubmit, onSuggest, onOpenChange }, forwardRef) => {
39
+ (
40
+ { classNames, autoFocus, lineWrapping = false, placeholder, onSubmit, onSuggest, onOpenChange, references },
41
+ forwardRef,
42
+ ) => {
37
43
  const { themeMode } = useThemeContext();
38
44
  const { parentRef, view } = useTextEditor(
39
45
  {
46
+ debug: true,
40
47
  autoFocus,
41
48
  extensions: [
42
49
  createBasicExtensions({
@@ -45,25 +52,28 @@ export const Prompt = forwardRef<PromptController, PromptProps>(
45
52
  placeholder,
46
53
  }),
47
54
  createThemeExtensions({ themeMode }),
55
+ references ? promptReferences({ provider: references }) : [],
48
56
  createAutocompleteExtension({ onSubmit, onSuggest }),
49
- keymap.of([
50
- {
51
- key: 'Alt-ArrowUp',
52
- preventDefault: true,
53
- run: (view) => {
54
- onOpenChange?.(true);
55
- return true;
57
+ Prec.highest(
58
+ keymap.of([
59
+ {
60
+ key: 'cmd-ArrowUp',
61
+ preventDefault: true,
62
+ run: (view) => {
63
+ onOpenChange?.(true);
64
+ return true;
65
+ },
56
66
  },
57
- },
58
- {
59
- key: 'Alt-ArrowDown',
60
- preventDefault: true,
61
- run: (view) => {
62
- onOpenChange?.(false);
63
- return true;
67
+ {
68
+ key: 'cmd-ArrowDown',
69
+ preventDefault: true,
70
+ run: (view) => {
71
+ onOpenChange?.(false);
72
+ return true;
73
+ },
64
74
  },
65
- },
66
- ]),
75
+ ]),
76
+ ),
67
77
  ],
68
78
  },
69
79
  [themeMode, onSubmit, onSuggest],
@@ -95,6 +105,6 @@ export const Prompt = forwardRef<PromptController, PromptProps>(
95
105
  [view, onSubmit],
96
106
  );
97
107
 
98
- return <div ref={parentRef} className={mx('w-full overflow-hidden', classNames)} />;
108
+ return <div ref={parentRef} className={mx('w-full', classNames)} />;
99
109
  },
100
110
  );
@@ -13,7 +13,7 @@ import { Prompt, type PromptController, type PromptProps } from './Prompt';
13
13
  import { ASSISTANT_PLUGIN } from '../../meta';
14
14
 
15
15
  export type PromptBarProps = ThemedClassName<
16
- Pick<PromptProps, 'placeholder' | 'lineWrapping' | 'onSubmit' | 'onSuggest' | 'onOpenChange'> & {
16
+ Pick<PromptProps, 'placeholder' | 'lineWrapping' | 'onSubmit' | 'onSuggest' | 'onOpenChange' | 'references'> & {
17
17
  processing?: boolean;
18
18
  error?: Error;
19
19
  microphone?: boolean;
@@ -28,6 +28,7 @@ export const PromptBar = ({
28
28
  error,
29
29
  microphone,
30
30
  onCancel,
31
+ references,
31
32
  ...props
32
33
  }: PromptBarProps) => {
33
34
  const { t } = useTranslation(ASSISTANT_PLUGIN);
@@ -72,6 +73,7 @@ export const PromptBar = ({
72
73
  classNames='pbs-2'
73
74
  lineWrapping={true}
74
75
  placeholder={placeholder ?? t('prompt placeholder')}
76
+ references={references}
75
77
  {...props}
76
78
  />
77
79
  {(onCancel || microphone) && (
@@ -0,0 +1,180 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { autocompletion, completionKeymap, type CompletionResult } from '@codemirror/autocomplete';
6
+ import { type Extension, RangeSet } from '@codemirror/state';
7
+ import {
8
+ Decoration,
9
+ EditorView,
10
+ keymap,
11
+ ViewPlugin,
12
+ WidgetType,
13
+ type DecorationSet,
14
+ type ViewUpdate,
15
+ } from '@codemirror/view';
16
+
17
+ import { Mutex } from '@dxos/async';
18
+
19
+ export type ReferenceData = {
20
+ uri: string;
21
+ label: string;
22
+ // TODO(dmaretskyi): Consider adding details renderer for when you hover over the reference.
23
+ };
24
+
25
+ export interface ReferencesProvider {
26
+ getReferences({ query }: { query: string }): Promise<ReferenceData[]>;
27
+
28
+ resolveReference({ uri }: { uri: string }): Promise<ReferenceData | null>;
29
+ }
30
+
31
+ export type PromptReferencesOptions = {
32
+ provider: ReferencesProvider;
33
+ /**
34
+ * Will prevent the autocomplete from closing when the user blurs the editor.
35
+ * @default false
36
+ */
37
+ debug?: boolean;
38
+ /**
39
+ * @default '@'
40
+ */
41
+ triggerCharacter?: string;
42
+ };
43
+
44
+ /**
45
+ * Include references into text.
46
+ */
47
+ export const promptReferences = ({
48
+ provider,
49
+ debug = false,
50
+ triggerCharacter = '@',
51
+ }: PromptReferencesOptions): Extension => {
52
+ if (triggerCharacter.length !== 1) {
53
+ throw new Error('triggerCharacter must be a single character');
54
+ }
55
+
56
+ const decorationField = ViewPlugin.fromClass(
57
+ class ReferenceView {
58
+ private _mutex = new Mutex();
59
+
60
+ decorations: DecorationSet = Decoration.set([]);
61
+
62
+ constructor(view: EditorView) {
63
+ queueMicrotask(async () => {
64
+ const guard = await this._mutex.acquire();
65
+ try {
66
+ this.decorations = await this._computeDecorations(view);
67
+ } finally {
68
+ guard.release();
69
+ }
70
+ });
71
+ }
72
+
73
+ update(update: ViewUpdate) {
74
+ if (update.docChanged) {
75
+ queueMicrotask(async () => {
76
+ const guard = await this._mutex.acquire();
77
+ try {
78
+ this.decorations = await this._computeDecorations(update.view);
79
+ } finally {
80
+ guard.release();
81
+ }
82
+ });
83
+ }
84
+ }
85
+
86
+ private async _computeDecorations(view: EditorView): Promise<DecorationSet> {
87
+ const text = view.state.doc.toString();
88
+ const references = text.matchAll(new RegExp(`${triggerCharacter}[a-zA-Z0-9@:]+\\s`, 'g'));
89
+
90
+ const decorations = [];
91
+ for (const match of references) {
92
+ const reference = match[0];
93
+ const uri = reference.slice(1, -1);
94
+ const data = await provider.resolveReference({ uri });
95
+ if (data) {
96
+ decorations.push(
97
+ Decoration.replace({
98
+ widget: new ReferenceWidget(data),
99
+ }).range(match.index!, match.index! + reference.length),
100
+ );
101
+ }
102
+ }
103
+
104
+ return Decoration.set(decorations);
105
+ }
106
+ },
107
+ {
108
+ decorations: (v) => v.decorations,
109
+ provide: (plugin) => [
110
+ EditorView.atomicRanges.of(
111
+ (view): DecorationSet => view.plugin(decorationField)?.decorations ?? RangeSet.empty,
112
+ ),
113
+ ],
114
+ },
115
+ );
116
+
117
+ return [
118
+ decorationField,
119
+
120
+ EditorView.theme({
121
+ '.cm-reference-pill': {
122
+ borderRadius: '0.25rem',
123
+ borderWidth: '1px',
124
+ marginRight: '0.25rem',
125
+ marginLeft: '0.25rem',
126
+ },
127
+ }),
128
+
129
+ autocompletion({
130
+ activateOnTyping: true,
131
+ override: [
132
+ async (context): Promise<CompletionResult | null> => {
133
+ const match = context.matchBefore(new RegExp(`${triggerCharacter}[a-zA-Z0-9]+`));
134
+
135
+ if (!match || match?.to === match?.from) {
136
+ return null;
137
+ }
138
+
139
+ const query = match.text.slice(1);
140
+ const references = await provider.getReferences({ query });
141
+
142
+ return {
143
+ from: match.from,
144
+ filter: false,
145
+ options: references.map((reference) => ({
146
+ label: reference.label,
147
+ apply: `${triggerCharacter}${reference.uri} `,
148
+ })),
149
+ };
150
+ },
151
+ ],
152
+ closeOnBlur: !debug,
153
+ tooltipClass: () => 'shadow rounded',
154
+ aboveCursor: true,
155
+ }),
156
+
157
+ keymap.of(completionKeymap),
158
+ ];
159
+ };
160
+
161
+ class ReferenceWidget extends WidgetType {
162
+ constructor(private data: ReferenceData) {
163
+ super();
164
+ }
165
+
166
+ override toDOM() {
167
+ const span = document.createElement('span');
168
+ span.textContent = `@ ${this.data.label}`;
169
+ span.className = 'cm-reference-pill';
170
+ return span;
171
+ }
172
+
173
+ override eq(other: ReferenceWidget) {
174
+ return other.data.uri === this.data.uri;
175
+ }
176
+
177
+ override ignoreEvent() {
178
+ return true;
179
+ }
180
+ }
@@ -2,17 +2,92 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import React from 'react';
5
+ import { Match, type Schema as S } from 'effect';
6
+ import React, { type ChangeEvent, useCallback } from 'react';
6
7
 
8
+ import { debounce } from '@dxos/async';
9
+ import { Input, Select, Toolbar, useTranslation } from '@dxos/react-ui';
7
10
  import { StackItem } from '@dxos/react-ui-stack';
8
11
 
9
12
  import { TemplateEditor } from './TemplateEditor';
10
- import { type TemplateType } from '../types';
13
+ import { ASSISTANT_PLUGIN } from '../meta';
14
+ import { TemplateKinds, type TemplateKindSchema, type TemplateType } from '../types';
11
15
 
12
- // TODO(burdon): Attention.
13
16
  export const TemplateContainer = ({ template, role }: { template: TemplateType; role: string }) => {
17
+ const { t } = useTranslation(ASSISTANT_PLUGIN);
18
+
19
+ const handleKindChange = useCallback(
20
+ (value: string) => {
21
+ const kind = Match.type<string>().pipe(
22
+ Match.withReturnType<S.Schema.Type<typeof TemplateKindSchema>>(),
23
+ Match.when('always', () => ({ include: 'always' })),
24
+ Match.when('schema-matching', () => ({ include: 'schema-matching', typename: '' })),
25
+ Match.when('automatically', () => ({ include: 'automatically', description: '' })),
26
+ Match.orElse(() => ({ include: 'manual' })),
27
+ )(value);
28
+
29
+ template.kind = kind;
30
+ },
31
+ [template],
32
+ );
33
+
34
+ const handleTypenameChange = useCallback(
35
+ debounce((event: ChangeEvent<HTMLInputElement>) => {
36
+ if (template.kind.include === 'schema-matching') {
37
+ template.kind.typename = event.target.value;
38
+ }
39
+ }, 300),
40
+ [template.kind.include],
41
+ );
42
+
43
+ const handleDescriptionChange = useCallback(
44
+ debounce((event: ChangeEvent<HTMLInputElement>) => {
45
+ if (template.kind.include === 'automatically') {
46
+ template.kind.description = event.target.value;
47
+ }
48
+ }, 300),
49
+ [template.kind.include],
50
+ );
51
+
14
52
  return (
15
- <StackItem.Content toolbar={false} role={role} classNames='mli-auto w-full max-w-[50rem]'>
53
+ <StackItem.Content toolbar role={role} classNames='mli-auto w-full max-w-[50rem]'>
54
+ {/* TODO(wittjosiah): Move this toolbar into c11y sidebar. */}
55
+ <Toolbar.Root>
56
+ <Select.Root value={template.kind.include} onValueChange={handleKindChange}>
57
+ <Toolbar.Button asChild>
58
+ <Select.TriggerButton />
59
+ </Toolbar.Button>
60
+ <Select.Portal>
61
+ <Select.Content>
62
+ <Select.Viewport>
63
+ {TemplateKinds.map((kind) => (
64
+ <Select.Option key={kind} value={kind}>
65
+ {kind}
66
+ </Select.Option>
67
+ ))}
68
+ </Select.Viewport>
69
+ </Select.Content>
70
+ </Select.Portal>
71
+ </Select.Root>
72
+ {template.kind.include === 'schema-matching' && (
73
+ <Input.Root>
74
+ <Input.TextInput
75
+ placeholder={t('typename placeholder')}
76
+ defaultValue={template.kind.typename}
77
+ onChange={handleTypenameChange}
78
+ />
79
+ </Input.Root>
80
+ )}
81
+ {template.kind.include === 'automatically' && (
82
+ <Input.Root>
83
+ <Input.TextInput
84
+ placeholder={t('description placeholder')}
85
+ defaultValue={template.kind.description}
86
+ onChange={handleDescriptionChange}
87
+ />
88
+ </Input.Root>
89
+ )}
90
+ </Toolbar.Root>
16
91
  <TemplateEditor template={template} />
17
92
  </StackItem.Content>
18
93
  );
@@ -38,7 +38,7 @@ const Render = ({ text }: TemplateEditorProps & { text: string }) => {
38
38
  const client = useClient();
39
39
  const [template] = useState(() => {
40
40
  const space = client.spaces.default;
41
- return space.db.add(create(TemplateType, { source: text }));
41
+ return space.db.add(create(TemplateType, { source: text, kind: { include: 'manual' } }));
42
42
  });
43
43
 
44
44
  return (
@@ -30,7 +30,7 @@ const Render = () => {
30
30
  const client = useClient();
31
31
  const [template] = useState(() => {
32
32
  const space = client.spaces.default;
33
- return space.db.add(create(TemplateType, { source: TEMPLATE }));
33
+ return space.db.add(create(TemplateType, { source: TEMPLATE, kind: { include: 'manual' } }));
34
34
  });
35
35
 
36
36
  return (
@@ -15,6 +15,13 @@ import { keyToFallback } from '@dxos/util';
15
15
  import { ThreadMessage, type ThreadMessageProps } from './ThreadMessage';
16
16
  import { messageReducer } from './reducer';
17
17
  import { PromptBar, type PromptBarProps } from '../Prompt';
18
+ import type { ReferenceData, ReferencesProvider } from '../Prompt/references';
19
+
20
+ export interface ContextProvider {
21
+ query({ query }: { query: string }): Promise<ReferenceData[]>;
22
+
23
+ resolveMetadata({ uri }: { uri: string }): Promise<ReferenceData | null>;
24
+ }
18
25
 
19
26
  export type ThreadProps = ThemedClassName<{
20
27
  space?: Space;
@@ -22,6 +29,7 @@ export type ThreadProps = ThemedClassName<{
22
29
  collapse?: boolean;
23
30
  transcription?: boolean;
24
31
  onOpenChange?: (open: boolean) => void;
32
+ contextProvider?: ContextProvider;
25
33
  }> &
26
34
  Pick<PromptBarProps, 'processing' | 'error' | 'onSubmit' | 'onSuggest' | 'onCancel'> &
27
35
  Pick<ThreadMessageProps, 'debug' | 'tools' | 'onPrompt' | 'onDelete'>;
@@ -40,6 +48,7 @@ export const Thread = ({
40
48
  onSubmit,
41
49
  onCancel,
42
50
  onOpenChange,
51
+ contextProvider,
43
52
  ...props
44
53
  }: ThreadProps) => {
45
54
  const scroller = useRef<ScrollController>(null);
@@ -68,6 +77,17 @@ export const Thread = ({
68
77
  }
69
78
  }, [messages, collapse]);
70
79
 
80
+ const references = useMemo<ReferencesProvider | undefined>(() => {
81
+ if (!contextProvider) {
82
+ return undefined;
83
+ }
84
+
85
+ return {
86
+ getReferences: async ({ query }: { query: string }) => contextProvider.query({ query }),
87
+ resolveReference: async ({ uri }: { uri: string }) => contextProvider.resolveMetadata({ uri }),
88
+ };
89
+ }, [contextProvider]);
90
+
71
91
  return (
72
92
  <div role='none' className={mx('flex flex-col grow overflow-hidden', classNames)}>
73
93
  <ScrollContainer ref={scroller} fade>
@@ -90,6 +110,7 @@ export const Thread = ({
90
110
  onSubmit={handleSubmit}
91
111
  onCancel={onCancel}
92
112
  onOpenChange={onOpenChange}
113
+ references={references}
93
114
  />
94
115
  )}
95
116
  </div>
@@ -21,7 +21,6 @@ import { Message, type Tool } from '@dxos/artifact';
21
21
  import { genericTools, localServiceEndpoints, type IsObject } from '@dxos/artifact-testing';
22
22
  import { AIServiceEdgeClient } from '@dxos/assistant';
23
23
  import { createStatic, ObjectId } from '@dxos/echo-schema';
24
- import { EdgeHttpClient } from '@dxos/edge-client';
25
24
  import { invariant } from '@dxos/invariant';
26
25
  import { DXN, QueueSubspaceTags, SpaceId } from '@dxos/keys';
27
26
  import { ChessPlugin } from '@dxos/plugin-chess';
@@ -31,9 +30,8 @@ import { InboxPlugin } from '@dxos/plugin-inbox';
31
30
  import { MapPlugin } from '@dxos/plugin-map';
32
31
  import { SpacePlugin } from '@dxos/plugin-space';
33
32
  import { TablePlugin } from '@dxos/plugin-table';
34
- import { useSpace } from '@dxos/react-client/echo';
33
+ import { useQueue, useSpace } from '@dxos/react-client/echo';
35
34
  import { withClientProvider } from '@dxos/react-client/testing';
36
- import { useQueue } from '@dxos/react-edge-client';
37
35
  import { IconButton, Input, Toolbar } from '@dxos/react-ui';
38
36
  import { mx } from '@dxos/react-ui-theme';
39
37
  import { withLayout, withSignals, withTheme } from '@dxos/storybook-utils';
@@ -54,13 +52,9 @@ type RenderProps = {
54
52
  const Render = ({ items: _items, prompts = [], ...props }: RenderProps) => {
55
53
  const space = useSpace();
56
54
  const artifactDefinitions = useCapabilities(Capabilities.ArtifactDefinition);
57
- const tools = useMemo<Tool[]>(
58
- () => [...genericTools, ...artifactDefinitions.flatMap((definition) => definition.tools)],
59
- [genericTools, artifactDefinitions],
60
- );
55
+ const tools = useMemo<Tool[]>(() => [...genericTools], []);
61
56
 
62
57
  const [aiClient] = useState(() => new AIServiceEdgeClient({ endpoint: endpoints.ai }));
63
- const [edgeClient] = useState(() => new EdgeHttpClient(endpoints.edge));
64
58
  const { dispatchPromise: dispatch } = useIntentDispatcher();
65
59
 
66
60
  // TODO(burdon): Replace with useChatProcessor.
@@ -73,6 +67,7 @@ const Render = ({ items: _items, prompts = [], ...props }: RenderProps) => {
73
67
  return new ChatProcessor(
74
68
  aiClient,
75
69
  tools,
70
+ artifactDefinitions,
76
71
  {
77
72
  space,
78
73
  dispatch,
@@ -82,8 +77,8 @@ const Render = ({ items: _items, prompts = [], ...props }: RenderProps) => {
82
77
  }, [aiClient, tools, space, dispatch, artifactDefinitions]);
83
78
 
84
79
  // Queue.
85
- const [queueDxn, setQueueDxn] = useState(() => randomQueueDxn());
86
- const queue = useQueue<Message>(edgeClient, DXN.tryParse(queueDxn));
80
+ const [queueDxn, setQueueDxn] = useState<string>(() => randomQueueDxn());
81
+ const queue = useQueue<Message>(DXN.tryParse(queueDxn));
87
82
 
88
83
  useEffect(() => {
89
84
  if (queue?.items.length === 0 && !queue.isLoading && prompts.length > 0) {
@@ -2,7 +2,7 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import React, { type FC, useCallback } from 'react';
5
+ import React, { useCallback, type FC } from 'react';
6
6
 
7
7
  import { invariant } from '@dxos/invariant';
8
8
  import { log } from '@dxos/log';
@@ -10,12 +10,13 @@ import { getSpace } from '@dxos/react-client/echo';
10
10
  import { type ThemedClassName } from '@dxos/react-ui';
11
11
 
12
12
  import { Thread, type ThreadProps } from './Thread';
13
- import { useChatProcessor, useMessageQueue } from '../../hooks';
13
+ import { useChatProcessor, useContextProvider, useMessageQueue } from '../../hooks';
14
14
  import { type AIChatType, type AssistantSettingsProps } from '../../types';
15
15
 
16
16
  export type ThreadContainerProps = {
17
17
  chat?: AIChatType;
18
18
  settings?: AssistantSettingsProps;
19
+ part?: 'deck' | 'dialog';
19
20
  } & Pick<ThreadProps, 'debug' | 'transcription' | 'onOpenChange'>;
20
21
 
21
22
  // TODO(burdon): Since this only wraps Thread, just separate out hook?
@@ -23,12 +24,13 @@ export const ThreadContainer: FC<ThemedClassName<ThreadContainerProps>> = ({
23
24
  classNames,
24
25
  chat,
25
26
  settings,
27
+ part,
26
28
  onOpenChange,
27
29
  ...props
28
30
  }) => {
29
- // Push up capabilities hooks out of components.
30
31
  const space = getSpace(chat);
31
- const processor = useChatProcessor(space, settings);
32
+ const contextProvider = useContextProvider(space);
33
+ const processor = useChatProcessor({ chat, space, settings, part });
32
34
  const messageQueue = useMessageQueue(chat);
33
35
  const messages = [...(messageQueue?.items ?? []), ...processor.messages.value];
34
36
 
@@ -73,6 +75,7 @@ export const ThreadContainer: FC<ThemedClassName<ThreadContainerProps>> = ({
73
75
  onCancel={handleCancel}
74
76
  onPrompt={handleSubmit}
75
77
  onOpenChange={onOpenChange}
78
+ contextProvider={contextProvider}
76
79
  {...props}
77
80
  />
78
81
  );