@dxos/plugin-markdown 0.8.4-main.21d9917 → 0.8.4-main.2244d791bb

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 (202) hide show
  1. package/dist/lib/browser/MarkdownCard-Z6PWYEHJ.mjs +11 -0
  2. package/dist/lib/browser/MarkdownContainer-KGEX4KEN.mjs +11 -0
  3. package/dist/lib/browser/{anchor-sort-WQ3TL7ZI.mjs → anchor-sort-QCUZGON5.mjs} +5 -4
  4. package/dist/lib/browser/anchor-sort-QCUZGON5.mjs.map +7 -0
  5. package/dist/lib/browser/{app-graph-serializer-ULZUJKOD.mjs → app-graph-serializer-6NGPRTY6.mjs} +7 -6
  6. package/dist/lib/browser/app-graph-serializer-6NGPRTY6.mjs.map +7 -0
  7. package/dist/lib/browser/blueprint-definition-HYSNAH6Z.mjs +17 -0
  8. package/dist/lib/browser/blueprint-definition-HYSNAH6Z.mjs.map +7 -0
  9. package/dist/lib/browser/blueprints/index.mjs +4 -4
  10. package/dist/lib/browser/{chunk-BX73DASG.mjs → chunk-IKUJPKAE.mjs} +130 -36
  11. package/dist/lib/browser/chunk-IKUJPKAE.mjs.map +7 -0
  12. package/dist/lib/browser/{chunk-JOXYQLKH.mjs → chunk-JODBECJA.mjs} +5 -3
  13. package/dist/lib/browser/chunk-JODBECJA.mjs.map +7 -0
  14. package/dist/lib/browser/{chunk-CUGDX7KA.mjs → chunk-KW4FLYZS.mjs} +2 -2
  15. package/dist/lib/browser/{chunk-FWQQW6KU.mjs → chunk-LIZX2OAT.mjs} +72 -38
  16. package/dist/lib/browser/chunk-LIZX2OAT.mjs.map +7 -0
  17. package/dist/lib/browser/{chunk-VQ3WOAB6.mjs → chunk-LRRS3L7O.mjs} +7 -9
  18. package/dist/lib/browser/chunk-LRRS3L7O.mjs.map +7 -0
  19. package/dist/lib/browser/{chunk-LMO5UVKL.mjs → chunk-PV7GW7CI.mjs} +3 -4
  20. package/dist/lib/browser/{chunk-LMO5UVKL.mjs.map → chunk-PV7GW7CI.mjs.map} +3 -3
  21. package/dist/lib/browser/{chunk-KDKXFKDN.mjs → chunk-YZKNND37.mjs} +8 -7
  22. package/dist/lib/browser/chunk-YZKNND37.mjs.map +7 -0
  23. package/dist/lib/browser/cli/index.mjs +11 -10
  24. package/dist/lib/browser/cli/index.mjs.map +3 -3
  25. package/dist/lib/browser/index.mjs +45 -42
  26. package/dist/lib/browser/index.mjs.map +3 -3
  27. package/dist/lib/browser/meta.json +1 -1
  28. package/dist/lib/browser/{operation-resolver-EGCWOQKJ.mjs → operation-resolver-EHCL5OS7.mjs} +5 -5
  29. package/dist/lib/browser/operation-resolver-EHCL5OS7.mjs.map +7 -0
  30. package/dist/lib/browser/{react-surface-64FZ7T7F.mjs → react-surface-SQVRMKCQ.mjs} +27 -27
  31. package/dist/lib/browser/react-surface-SQVRMKCQ.mjs.map +7 -0
  32. package/dist/lib/browser/{settings-JY5JE7MI.mjs → settings-4DDC3NMP.mjs} +5 -4
  33. package/dist/lib/browser/settings-4DDC3NMP.mjs.map +7 -0
  34. package/dist/lib/browser/{state-QE7VSJWJ.mjs → state-UIZ3X7RA.mjs} +2 -2
  35. package/dist/lib/browser/types/index.mjs +1 -1
  36. package/dist/lib/node-esm/{MarkdownCard-GMNXUWWR.mjs → MarkdownCard-NUGJRT43.mjs} +4 -5
  37. package/dist/lib/node-esm/{MarkdownContainer-LUSMORP2.mjs → MarkdownContainer-6D7OHAZA.mjs} +4 -5
  38. package/dist/lib/node-esm/{anchor-sort-G7D5TAI6.mjs → anchor-sort-ABOJXSU6.mjs} +5 -4
  39. package/dist/lib/node-esm/anchor-sort-ABOJXSU6.mjs.map +7 -0
  40. package/dist/lib/node-esm/{app-graph-serializer-W5YMQP7P.mjs → app-graph-serializer-OYKI3LW2.mjs} +7 -6
  41. package/dist/lib/node-esm/app-graph-serializer-OYKI3LW2.mjs.map +7 -0
  42. package/dist/lib/node-esm/blueprint-definition-SOC744SQ.mjs +18 -0
  43. package/dist/lib/node-esm/blueprint-definition-SOC744SQ.mjs.map +7 -0
  44. package/dist/lib/node-esm/blueprints/index.mjs +4 -4
  45. package/dist/lib/node-esm/{chunk-KXEQCFMB.mjs → chunk-A5WB5WP7.mjs} +7 -9
  46. package/dist/lib/node-esm/chunk-A5WB5WP7.mjs.map +7 -0
  47. package/dist/lib/node-esm/{chunk-EYAFC4N7.mjs → chunk-BMKFZ5WK.mjs} +72 -38
  48. package/dist/lib/node-esm/chunk-BMKFZ5WK.mjs.map +7 -0
  49. package/dist/lib/node-esm/{chunk-W3IIKDV5.mjs → chunk-GXEX67UI.mjs} +2 -2
  50. package/dist/lib/node-esm/{chunk-PRV35A7Z.mjs → chunk-KG7JF7PJ.mjs} +3 -4
  51. package/dist/lib/node-esm/{chunk-PRV35A7Z.mjs.map → chunk-KG7JF7PJ.mjs.map} +3 -3
  52. package/dist/lib/node-esm/{chunk-44VJC3QF.mjs → chunk-PPGVKPFP.mjs} +5 -3
  53. package/dist/lib/node-esm/chunk-PPGVKPFP.mjs.map +7 -0
  54. package/dist/lib/node-esm/{chunk-3JYPCC7M.mjs → chunk-U4W25KQQ.mjs} +8 -7
  55. package/dist/lib/node-esm/chunk-U4W25KQQ.mjs.map +7 -0
  56. package/dist/lib/node-esm/{chunk-D4BCFPKK.mjs → chunk-WJRTAU55.mjs} +130 -36
  57. package/dist/lib/node-esm/chunk-WJRTAU55.mjs.map +7 -0
  58. package/dist/lib/node-esm/cli/index.mjs +11 -10
  59. package/dist/lib/node-esm/cli/index.mjs.map +3 -3
  60. package/dist/lib/node-esm/index.mjs +45 -42
  61. package/dist/lib/node-esm/index.mjs.map +3 -3
  62. package/dist/lib/node-esm/meta.json +1 -1
  63. package/dist/lib/node-esm/{operation-resolver-2HIS2AQZ.mjs → operation-resolver-ILDFG244.mjs} +5 -5
  64. package/dist/lib/node-esm/operation-resolver-ILDFG244.mjs.map +7 -0
  65. package/dist/lib/node-esm/{react-surface-PKEVHTJK.mjs → react-surface-5DVOPYF3.mjs} +27 -27
  66. package/dist/lib/node-esm/react-surface-5DVOPYF3.mjs.map +7 -0
  67. package/dist/lib/node-esm/{settings-4UGMPIRY.mjs → settings-MHB4FPJU.mjs} +5 -4
  68. package/dist/lib/node-esm/settings-MHB4FPJU.mjs.map +7 -0
  69. package/dist/lib/node-esm/{state-LXE5XIN4.mjs → state-B2JSHJKS.mjs} +2 -2
  70. package/dist/lib/node-esm/types/index.mjs +1 -1
  71. package/dist/types/src/MarkdownPlugin.d.ts.map +1 -1
  72. package/dist/types/src/blueprints/functions/create.d.ts.map +1 -1
  73. package/dist/types/src/blueprints/functions/index.d.ts +41 -3
  74. package/dist/types/src/blueprints/functions/index.d.ts.map +1 -1
  75. package/dist/types/src/blueprints/functions/open.d.ts +8 -1
  76. package/dist/types/src/blueprints/functions/open.d.ts.map +1 -1
  77. package/dist/types/src/blueprints/functions/update.d.ts +16 -3
  78. package/dist/types/src/blueprints/functions/update.d.ts.map +1 -1
  79. package/dist/types/src/blueprints/index.d.ts +1 -1
  80. package/dist/types/src/blueprints/index.d.ts.map +1 -1
  81. package/dist/types/src/blueprints/markdown-blueprint.d.ts +3 -20
  82. package/dist/types/src/blueprints/markdown-blueprint.d.ts.map +1 -1
  83. package/dist/types/src/capabilities/anchor-sort/anchor-sort.d.ts +3 -2
  84. package/dist/types/src/capabilities/anchor-sort/anchor-sort.d.ts.map +1 -1
  85. package/dist/types/src/capabilities/anchor-sort/index.d.ts +1 -1
  86. package/dist/types/src/capabilities/anchor-sort/index.d.ts.map +1 -1
  87. package/dist/types/src/capabilities/app-graph-serializer/app-graph-serializer.d.ts +2 -2
  88. package/dist/types/src/capabilities/app-graph-serializer/app-graph-serializer.d.ts.map +1 -1
  89. package/dist/types/src/capabilities/app-graph-serializer/index.d.ts +1 -1
  90. package/dist/types/src/capabilities/app-graph-serializer/index.d.ts.map +1 -1
  91. package/dist/types/src/capabilities/artifact-definition/artifact-definition.d.ts.map +1 -1
  92. package/dist/types/src/capabilities/blueprint-definition/blueprint-definition.d.ts +3 -7
  93. package/dist/types/src/capabilities/blueprint-definition/blueprint-definition.d.ts.map +1 -1
  94. package/dist/types/src/capabilities/blueprint-definition/index.d.ts +1 -1
  95. package/dist/types/src/capabilities/blueprint-definition/index.d.ts.map +1 -1
  96. package/dist/types/src/capabilities/operation-resolver/index.d.ts +1 -1
  97. package/dist/types/src/capabilities/operation-resolver/index.d.ts.map +1 -1
  98. package/dist/types/src/capabilities/operation-resolver/operation-resolver.d.ts +2 -2
  99. package/dist/types/src/capabilities/operation-resolver/operation-resolver.d.ts.map +1 -1
  100. package/dist/types/src/capabilities/react-surface/index.d.ts +1 -1
  101. package/dist/types/src/capabilities/react-surface/index.d.ts.map +1 -1
  102. package/dist/types/src/capabilities/react-surface/react-surface.d.ts +2 -2
  103. package/dist/types/src/capabilities/react-surface/react-surface.d.ts.map +1 -1
  104. package/dist/types/src/capabilities/settings/index.d.ts +1 -1
  105. package/dist/types/src/capabilities/settings/index.d.ts.map +1 -1
  106. package/dist/types/src/capabilities/settings/settings.d.ts +3 -2
  107. package/dist/types/src/capabilities/settings/settings.d.ts.map +1 -1
  108. package/dist/types/src/cli/plugin.d.ts.map +1 -1
  109. package/dist/types/src/components/MarkdownCard/MarkdownCard.d.ts.map +1 -1
  110. package/dist/types/src/components/MarkdownContainer.d.ts +1 -1
  111. package/dist/types/src/components/MarkdownContainer.d.ts.map +1 -1
  112. package/dist/types/src/components/MarkdownContainer.stories.d.ts +29 -0
  113. package/dist/types/src/components/MarkdownContainer.stories.d.ts.map +1 -1
  114. package/dist/types/src/components/MarkdownEditor/FileUpload.d.ts +2 -2
  115. package/dist/types/src/components/MarkdownEditor/FileUpload.d.ts.map +1 -1
  116. package/dist/types/src/components/MarkdownEditor/MarkdownEditor.d.ts.map +1 -1
  117. package/dist/types/src/components/MarkdownEditor/MarkdownEditorToolbar.d.ts +2 -2
  118. package/dist/types/src/components/MarkdownEditor/MarkdownEditorToolbar.d.ts.map +1 -1
  119. package/dist/types/src/hooks/useExtensions.d.ts +2 -2
  120. package/dist/types/src/hooks/useExtensions.d.ts.map +1 -1
  121. package/dist/types/src/index.d.ts +1 -0
  122. package/dist/types/src/index.d.ts.map +1 -1
  123. package/dist/types/src/translations.d.ts +29 -0
  124. package/dist/types/src/translations.d.ts.map +1 -1
  125. package/dist/types/src/types/MarkdownAction.d.ts +1 -1
  126. package/dist/types/src/types/events.d.ts.map +1 -1
  127. package/dist/types/tsconfig.tsbuildinfo +1 -1
  128. package/package.json +66 -59
  129. package/src/MarkdownPlugin.tsx +26 -25
  130. package/src/blueprints/functions/create.conversations.json +1 -1
  131. package/src/blueprints/functions/create.test.ts +23 -56
  132. package/src/blueprints/functions/create.ts +3 -2
  133. package/src/blueprints/functions/index.ts +11 -3
  134. package/src/blueprints/functions/open.ts +9 -9
  135. package/src/blueprints/functions/update.conversations.json +1 -1
  136. package/src/blueprints/functions/update.test.ts +132 -66
  137. package/src/blueprints/functions/update.ts +53 -12
  138. package/src/blueprints/index.ts +1 -1
  139. package/src/blueprints/markdown-blueprint.ts +14 -6
  140. package/src/capabilities/anchor-sort/anchor-sort.ts +3 -2
  141. package/src/capabilities/app-graph-serializer/app-graph-serializer.ts +4 -3
  142. package/src/capabilities/artifact-definition/artifact-definition.ts +2 -2
  143. package/src/capabilities/blueprint-definition/blueprint-definition.ts +6 -14
  144. package/src/capabilities/operation-resolver/operation-resolver.ts +3 -3
  145. package/src/capabilities/react-surface/react-surface.tsx +12 -10
  146. package/src/capabilities/settings/settings.ts +3 -2
  147. package/src/cli/plugin.ts +5 -4
  148. package/src/components/MarkdownCard/MarkdownCard.stories.tsx +1 -1
  149. package/src/components/MarkdownCard/MarkdownCard.tsx +7 -5
  150. package/src/components/MarkdownContainer.stories.tsx +5 -5
  151. package/src/components/MarkdownContainer.tsx +6 -4
  152. package/src/components/MarkdownEditor/FileUpload.tsx +2 -2
  153. package/src/components/MarkdownEditor/MarkdownEditor.stories.tsx +2 -2
  154. package/src/components/MarkdownEditor/MarkdownEditor.tsx +17 -11
  155. package/src/components/MarkdownEditor/MarkdownEditorToolbar.tsx +2 -2
  156. package/src/components/MarkdownSettings/MarkdownSettings.tsx +24 -24
  157. package/src/components/Suggestions.stories.tsx +3 -3
  158. package/src/hooks/useExtensions.tsx +5 -4
  159. package/src/hooks/useLinkQuery.ts +3 -3
  160. package/src/hooks/useSelectCurrentThread.tsx +3 -3
  161. package/src/index.ts +2 -0
  162. package/src/testing.ts +1 -1
  163. package/src/translations.ts +2 -0
  164. package/src/types/events.ts +3 -2
  165. package/dist/lib/browser/MarkdownCard-HFNNE4B4.mjs +0 -12
  166. package/dist/lib/browser/MarkdownContainer-V7EAADMF.mjs +0 -12
  167. package/dist/lib/browser/anchor-sort-WQ3TL7ZI.mjs.map +0 -7
  168. package/dist/lib/browser/app-graph-serializer-ULZUJKOD.mjs.map +0 -7
  169. package/dist/lib/browser/blueprint-definition-TLV4PNZW.mjs +0 -19
  170. package/dist/lib/browser/blueprint-definition-TLV4PNZW.mjs.map +0 -7
  171. package/dist/lib/browser/chunk-BX73DASG.mjs.map +0 -7
  172. package/dist/lib/browser/chunk-FWQQW6KU.mjs.map +0 -7
  173. package/dist/lib/browser/chunk-JOXYQLKH.mjs.map +0 -7
  174. package/dist/lib/browser/chunk-KDKXFKDN.mjs.map +0 -7
  175. package/dist/lib/browser/chunk-S45ULIOG.mjs +0 -101
  176. package/dist/lib/browser/chunk-S45ULIOG.mjs.map +0 -7
  177. package/dist/lib/browser/chunk-VQ3WOAB6.mjs.map +0 -7
  178. package/dist/lib/browser/operation-resolver-EGCWOQKJ.mjs.map +0 -7
  179. package/dist/lib/browser/react-surface-64FZ7T7F.mjs.map +0 -7
  180. package/dist/lib/browser/settings-JY5JE7MI.mjs.map +0 -7
  181. package/dist/lib/node-esm/anchor-sort-G7D5TAI6.mjs.map +0 -7
  182. package/dist/lib/node-esm/app-graph-serializer-W5YMQP7P.mjs.map +0 -7
  183. package/dist/lib/node-esm/blueprint-definition-HYFA7BKQ.mjs +0 -20
  184. package/dist/lib/node-esm/blueprint-definition-HYFA7BKQ.mjs.map +0 -7
  185. package/dist/lib/node-esm/chunk-3JYPCC7M.mjs.map +0 -7
  186. package/dist/lib/node-esm/chunk-44VJC3QF.mjs.map +0 -7
  187. package/dist/lib/node-esm/chunk-D4BCFPKK.mjs.map +0 -7
  188. package/dist/lib/node-esm/chunk-EYAFC4N7.mjs.map +0 -7
  189. package/dist/lib/node-esm/chunk-KXEQCFMB.mjs.map +0 -7
  190. package/dist/lib/node-esm/chunk-SDTYEGGL.mjs +0 -102
  191. package/dist/lib/node-esm/chunk-SDTYEGGL.mjs.map +0 -7
  192. package/dist/lib/node-esm/operation-resolver-2HIS2AQZ.mjs.map +0 -7
  193. package/dist/lib/node-esm/react-surface-PKEVHTJK.mjs.map +0 -7
  194. package/dist/lib/node-esm/settings-4UGMPIRY.mjs.map +0 -7
  195. /package/dist/lib/browser/{MarkdownCard-HFNNE4B4.mjs.map → MarkdownCard-Z6PWYEHJ.mjs.map} +0 -0
  196. /package/dist/lib/browser/{MarkdownContainer-V7EAADMF.mjs.map → MarkdownContainer-KGEX4KEN.mjs.map} +0 -0
  197. /package/dist/lib/browser/{chunk-CUGDX7KA.mjs.map → chunk-KW4FLYZS.mjs.map} +0 -0
  198. /package/dist/lib/browser/{state-QE7VSJWJ.mjs.map → state-UIZ3X7RA.mjs.map} +0 -0
  199. /package/dist/lib/node-esm/{MarkdownCard-GMNXUWWR.mjs.map → MarkdownCard-NUGJRT43.mjs.map} +0 -0
  200. /package/dist/lib/node-esm/{MarkdownContainer-LUSMORP2.mjs.map → MarkdownContainer-6D7OHAZA.mjs.map} +0 -0
  201. /package/dist/lib/node-esm/{chunk-W3IIKDV5.mjs.map → chunk-GXEX67UI.mjs.map} +0 -0
  202. /package/dist/lib/node-esm/{state-LXE5XIN4.mjs.map → state-B2JSHJKS.mjs.map} +0 -0
@@ -6,59 +6,33 @@ import { describe, expect, it } from '@effect/vitest';
6
6
  import * as Effect from 'effect/Effect';
7
7
  import * as Layer from 'effect/Layer';
8
8
 
9
- import { AiService, ConsolePrinter } from '@dxos/ai';
10
- import { MemoizedAiService, TestAiService } from '@dxos/ai/testing';
11
- import {
12
- AiConversation,
13
- type ContextBinding,
14
- GenerationObserver,
15
- makeToolExecutionServiceFromFunctions,
16
- makeToolResolverFromFunctions,
17
- } from '@dxos/assistant';
9
+ import { MemoizedAiService } from '@dxos/ai/testing';
10
+ import { AiContextService, AiConversationService } from '@dxos/assistant';
11
+ import { AssistantTestLayer } from '@dxos/assistant/testing';
18
12
  import { Blueprint } from '@dxos/blueprints';
19
13
  import { SpaceProperties } from '@dxos/client-protocol';
20
- import { Obj, Query, Ref } from '@dxos/echo';
21
- import { Database } from '@dxos/echo';
22
- import { acquireReleaseResource } from '@dxos/effect';
14
+ import { Database, Obj, Query, Ref } from '@dxos/echo';
23
15
  import { TestHelpers } from '@dxos/effect/testing';
24
- import { CredentialsService, FunctionInvocationService, QueueService, TracingService } from '@dxos/functions';
25
- import { FunctionInvocationServiceLayerTest, TestDatabaseLayer } from '@dxos/functions-runtime/testing';
16
+ import { FunctionInvocationService } from '@dxos/functions';
26
17
  import { invariant } from '@dxos/invariant';
27
18
  import { ObjectId } from '@dxos/keys';
28
19
  import { Markdown } from '@dxos/plugin-markdown/types';
29
20
  import { Collection } from '@dxos/schema';
30
- import { HasSubject, type Message } from '@dxos/types';
21
+ import { HasSubject } from '@dxos/types';
22
+ import { trim } from '@dxos/util';
31
23
 
32
- import { WithProperties, testToolkit } from '../../testing';
33
- import * as MarkdownBlueprint from '../markdown-blueprint';
24
+ import { WithProperties } from '../../testing';
25
+ import MarkdownBlueprint from '../markdown-blueprint';
34
26
 
35
27
  import update from './update';
36
28
 
37
29
  ObjectId.dangerouslyDisableRandomness();
38
30
 
39
- const TestLayer = Layer.mergeAll(
40
- AiService.model('@anthropic/claude-opus-4-0'),
41
- makeToolResolverFromFunctions(MarkdownBlueprint.functions, testToolkit),
42
- makeToolExecutionServiceFromFunctions(testToolkit, testToolkit.toLayer({}) as any),
43
- ).pipe(
44
- Layer.provideMerge(
45
- FunctionInvocationServiceLayerTest({
46
- functions: MarkdownBlueprint.functions,
47
- }),
48
- ),
49
- Layer.provideMerge(
50
- Layer.mergeAll(
51
- TestAiService(),
52
- TestDatabaseLayer({
53
- spaceKey: 'fixed',
54
- indexing: { vector: true },
55
- types: [SpaceProperties, Collection.Collection, Blueprint.Blueprint, Markdown.Document, HasSubject.HasSubject],
56
- }),
57
- CredentialsService.configuredLayer([]),
58
- TracingService.layerNoop,
59
- ),
60
- ),
61
- );
31
+ const TestLayer = AssistantTestLayer({
32
+ functions: [...MarkdownBlueprint.functions],
33
+ types: [SpaceProperties, Collection.Collection, Blueprint.Blueprint, Markdown.Document, HasSubject.HasSubject],
34
+ tracing: 'pretty',
35
+ });
62
36
 
63
37
  describe('update', () => {
64
38
  it.effect(
@@ -69,16 +43,16 @@ describe('update', () => {
69
43
  name: 'BlueYard',
70
44
  content: 'Founders and portfolio of BlueYard.',
71
45
  });
72
- yield* Database.Service.add(doc);
46
+ yield* Database.add(doc);
73
47
 
74
48
  yield* FunctionInvocationService.invokeFunction(update, {
75
- id: doc.id,
76
- diffs: ['- Founders', '+ # Founders'],
49
+ doc: Ref.make(doc),
50
+ edits: [{ oldString: 'Founders', newString: '# Founders' }],
77
51
  });
78
52
 
79
- const updatedDoc = yield* Database.Service.resolve(Obj.getDXN(doc), Markdown.Document);
53
+ const updatedDoc = yield* Database.resolve(Obj.getDXN(doc), Markdown.Document);
80
54
  expect(updatedDoc.name).toBe(doc.name);
81
- const text = yield* Database.Service.load(updatedDoc.content);
55
+ const text = yield* Database.load(updatedDoc.content);
82
56
  expect(text.content).toBe('# Founders and portfolio of BlueYard.');
83
57
  },
84
58
  WithProperties,
@@ -91,25 +65,16 @@ describe('update', () => {
91
65
  'create and update a markdown document',
92
66
  Effect.fnUntraced(
93
67
  function* (_) {
94
- const queue = yield* QueueService.createQueue<Message.Message | ContextBinding>();
95
- const conversation = yield* acquireReleaseResource(() => new AiConversation({ queue }));
96
-
97
- yield* Database.Service.flush({ indexes: true });
98
- const markdownBlueprint = yield* Database.Service.add(Obj.clone(MarkdownBlueprint.make()));
99
- yield* Effect.promise(() =>
100
- conversation.context.bind({
101
- blueprints: [Ref.make(markdownBlueprint)],
102
- }),
103
- );
104
-
105
- const observer = GenerationObserver.fromPrinter(new ConsolePrinter());
68
+ const markdownBlueprint = yield* Database.add(Obj.clone(MarkdownBlueprint.make()));
69
+ yield* AiContextService.bindContext({
70
+ blueprints: [Ref.make(markdownBlueprint)],
71
+ });
106
72
 
107
- yield* conversation.createRequest({
108
- observer,
73
+ yield* AiConversationService.run({
109
74
  prompt: `Create a document with a cookie recipe.`,
110
75
  });
111
76
  {
112
- const docs = yield* Database.Service.runQuery(Query.type(Markdown.Document));
77
+ const docs = yield* Database.runQuery(Query.type(Markdown.Document));
113
78
  if (docs.length !== 1) {
114
79
  throw new Error(`Expected 1 document; got ${docs.length}: ${docs.map((_) => _.name)}`);
115
80
  }
@@ -118,16 +83,15 @@ describe('update', () => {
118
83
  invariant(Obj.instanceOf(Markdown.Document, doc));
119
84
  console.log({
120
85
  name: doc.name,
121
- content: yield* Database.Service.load(doc.content).pipe(Effect.map((_) => _.content)),
86
+ content: yield* Database.load(doc.content).pipe(Effect.map((_) => _.content)),
122
87
  });
123
88
  }
124
89
 
125
- yield* conversation.createRequest({
126
- observer,
90
+ yield* AiConversationService.run({
127
91
  prompt: 'Add a section with a holiday-themed variation.',
128
92
  });
129
93
  {
130
- const docs = yield* Database.Service.runQuery(Query.type(Markdown.Document));
94
+ const docs = yield* Database.runQuery(Query.type(Markdown.Document));
131
95
  if (docs.length !== 1) {
132
96
  throw new Error(`Expected 1 document; got ${docs.length}: ${docs.map((_) => _.name)}`);
133
97
  }
@@ -136,12 +100,114 @@ describe('update', () => {
136
100
  invariant(Obj.instanceOf(Markdown.Document, doc));
137
101
  console.log({
138
102
  name: doc.name,
139
- content: yield* Database.Service.load(doc.content).pipe(Effect.map((_) => _.content)),
103
+ content: yield* Database.load(doc.content).pipe(Effect.map((_) => _.content)),
140
104
  });
141
105
  }
142
106
  },
143
107
  WithProperties,
144
- Effect.provide(TestLayer),
108
+ Effect.provide(AiConversationService.layerNewQueue().pipe(Layer.provideMerge(TestLayer))),
109
+ TestHelpers.provideTestContext,
110
+ ),
111
+ MemoizedAiService.isGenerationEnabled() ? 240_000 : 30_000,
112
+ );
113
+
114
+ it.scoped(
115
+ 'update existing document',
116
+ Effect.fnUntraced(
117
+ function* (_) {
118
+ const document = yield* Database.add(
119
+ Markdown.make({
120
+ name: 'Cookie Recipe',
121
+ content: trim`
122
+ Ingredients:
123
+ - 2 cups of ???
124
+ - 1 cup of sugar
125
+ - 1 cup of butter
126
+ - 1 cup of eggs
127
+ `,
128
+ }),
129
+ );
130
+ const markdownBlueprint = yield* Database.add(Obj.clone(MarkdownBlueprint.make()));
131
+ yield* AiContextService.bindContext({
132
+ blueprints: [Ref.make(markdownBlueprint)],
133
+ objects: [Ref.make(document)],
134
+ });
135
+
136
+ yield* AiConversationService.run({
137
+ prompt: 'Add the missing ingredient (its flour).',
138
+ });
139
+
140
+ {
141
+ const docs = yield* Database.runQuery(Query.type(Markdown.Document));
142
+ if (docs.length !== 1) {
143
+ throw new Error(`Expected 1 document; got ${docs.length}: ${docs.map((_) => _.name)}`);
144
+ }
145
+
146
+ const doc = docs[0];
147
+ invariant(Obj.instanceOf(Markdown.Document, doc));
148
+ const content = yield* Database.load(doc.content).pipe(Effect.map((_) => _.content));
149
+ console.log({
150
+ name: doc.name,
151
+ content: yield* Database.load(doc.content).pipe(Effect.map((_) => _.content)),
152
+ });
153
+ expect(content.toLowerCase()).toContain('flour');
154
+ }
155
+ },
156
+ WithProperties,
157
+ Effect.provide(AiConversationService.layerNewQueue().pipe(Layer.provideMerge(TestLayer))),
158
+ TestHelpers.provideTestContext,
159
+ ),
160
+ MemoizedAiService.isGenerationEnabled() ? 240_000 : 30_000,
161
+ );
162
+
163
+ it.scoped(
164
+ 'add lines to document one by one',
165
+ Effect.fnUntraced(
166
+ function* (_) {
167
+ const document = yield* Database.add(
168
+ Markdown.make({
169
+ name: 'Shopping list',
170
+ content: trim`
171
+ # Shopping list
172
+ `,
173
+ }),
174
+ );
175
+ const markdownBlueprint = yield* Database.add(Obj.clone(MarkdownBlueprint.make()));
176
+ yield* AiContextService.bindContext({
177
+ blueprints: [Ref.make(markdownBlueprint)],
178
+ objects: [Ref.make(document)],
179
+ });
180
+
181
+ yield* AiConversationService.run({
182
+ prompt: 'Add milk to the shopping list.',
183
+ });
184
+ yield* AiConversationService.run({
185
+ prompt: 'Add bread to the shopping list.',
186
+ });
187
+ yield* AiConversationService.run({
188
+ prompt: 'Add eggs to the shopping list.',
189
+ });
190
+
191
+ {
192
+ const docs = yield* Database.runQuery(Query.type(Markdown.Document));
193
+ if (docs.length !== 1) {
194
+ throw new Error(`Expected 1 document; got ${docs.length}: ${docs.map((_) => _.name)}`);
195
+ }
196
+
197
+ const doc = docs[0];
198
+ invariant(Obj.instanceOf(Markdown.Document, doc));
199
+ const content = yield* Database.load(doc.content).pipe(Effect.map((_) => _.content));
200
+ console.log({
201
+ name: doc.name,
202
+ content: yield* Database.load(doc.content).pipe(Effect.map((_) => _.content)),
203
+ });
204
+ expect(content.toLowerCase()).toContain('milk');
205
+ expect(content.toLowerCase()).toContain('bread');
206
+ expect(content.toLowerCase()).toContain('eggs');
207
+ }
208
+ },
209
+ WithProperties,
210
+ Effect.provide(AiConversationService.layerNewQueue().pipe(Layer.provideMerge(TestLayer))),
145
211
  TestHelpers.provideTestContext,
146
212
  ),
147
213
  MemoizedAiService.isGenerationEnabled() ? 240_000 : 30_000,
@@ -2,37 +2,78 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
+ import { next as A, type Doc } from '@automerge/automerge';
5
6
  import * as Effect from 'effect/Effect';
6
7
  import * as Schema from 'effect/Schema';
7
8
 
8
- import { ArtifactId, applyDiffs } from '@dxos/assistant';
9
- import { Database } from '@dxos/echo';
10
- import { createDocAccessor } from '@dxos/echo-db';
9
+ import { Database, Type } from '@dxos/echo';
10
+ import { DocAccessor, createDocAccessor } from '@dxos/echo-db';
11
11
  import { defineFunction } from '@dxos/functions';
12
12
  import { trim } from '@dxos/util';
13
13
 
14
14
  import { Markdown } from '../../types';
15
15
 
16
+ const Edit = Schema.Struct({
17
+ oldString: Schema.String.annotations({
18
+ description: 'The text to find in the document.',
19
+ }),
20
+ newString: Schema.String.annotations({
21
+ description: 'The text to replace it with.',
22
+ }),
23
+ replaceAll: Schema.optional(Schema.Boolean).annotations({
24
+ description: 'If true, replaces all occurrences. Defaults to false (first occurrence only).',
25
+ }),
26
+ });
27
+
16
28
  // TODO(wittjosiah): Reconcile with ThreadAction.AddProposal.
29
+ // Based on opencode's MutilEdit tool: https://deepwiki.com/search/how-does-text-edit-tool-work_a159bc76-5401-424e-b29f-f087d1ea4f78
17
30
  export default defineFunction({
18
31
  key: 'dxos.org/function/markdown/update',
19
32
  name: 'Update',
20
33
  description: trim`
21
- Applies a set of diffs to the markdown document.
34
+ Applies a set of edits to the markdown document.
22
35
  `,
23
36
  inputSchema: Schema.Struct({
24
- id: ArtifactId.annotations({
37
+ doc: Type.Ref(Markdown.Document).annotations({
25
38
  description: 'The ID of the markdown document.',
26
39
  }),
27
- diffs: Schema.Array(Schema.String).annotations({
28
- description: 'The diffs to apply to the document.',
40
+ edits: Schema.Array(Edit).annotations({
41
+ description: 'The edits to apply to the document. Each edit finds oldString and replaces it with newString.',
29
42
  }),
30
43
  }),
31
- outputSchema: Schema.Void,
32
- handler: Effect.fn(function* ({ data: { id, diffs } }) {
33
- const object = yield* Database.Service.resolve(ArtifactId.toDXN(id), Markdown.Document);
34
- const content = yield* Effect.promise(() => object.content.load());
44
+ outputSchema: Schema.Struct({
45
+ newContent: Schema.String,
46
+ }),
47
+ handler: Effect.fn(function* ({ data: { doc, edits } }) {
48
+ const content = yield* doc.pipe(
49
+ Database.load,
50
+ Effect.map((_) => _.content),
51
+ Effect.flatMap(Database.load),
52
+ );
35
53
  const accessor = createDocAccessor(content, ['content']);
36
- applyDiffs(accessor, diffs);
54
+
55
+ for (const edit of edits) {
56
+ accessor.handle.change((doc: Doc<typeof content>) => {
57
+ const text = DocAccessor.getValue<string>(accessor);
58
+ if (edit.replaceAll) {
59
+ let idx = text.indexOf(edit.oldString);
60
+ while (idx !== -1) {
61
+ A.splice(doc, accessor.path as A.Prop[], idx, edit.oldString.length, edit.newString);
62
+ const updated = DocAccessor.getValue<string>(accessor);
63
+ idx = updated.indexOf(edit.oldString, idx + edit.newString.length);
64
+ }
65
+ } else {
66
+ const idx = text.indexOf(edit.oldString);
67
+ if (idx === -1) {
68
+ throw new Error(`Edit not found: ${JSON.stringify(edit.oldString)}`);
69
+ }
70
+ A.splice(doc, accessor.path as A.Prop[], idx, edit.oldString.length, edit.newString);
71
+ }
72
+ });
73
+ }
74
+
75
+ return {
76
+ newContent: DocAccessor.getValue<string>(accessor),
77
+ };
37
78
  }),
38
79
  });
@@ -2,4 +2,4 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- export * as MarkdownBlueprint from './markdown-blueprint';
5
+ export { default as MarkdownBlueprint } from './markdown-blueprint';
@@ -2,19 +2,19 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
+ import { type AppCapabilities } from '@dxos/app-toolkit';
5
6
  import { Blueprint, Template } from '@dxos/blueprints';
6
- import { type FunctionDefinition } from '@dxos/functions';
7
7
  import { trim } from '@dxos/util';
8
8
 
9
- import { create, open, update } from './functions';
9
+ import { MarkdownFunctions } from './functions';
10
10
 
11
- export const functions: FunctionDefinition[] = [create, open, update];
11
+ const BLUEPRINT_KEY = 'dxos.org/blueprint/markdown';
12
12
 
13
- export const Key = 'dxos.org/blueprint/markdown';
13
+ const functions = Object.values(MarkdownFunctions);
14
14
 
15
- export const make = () =>
15
+ const make = () =>
16
16
  Blueprint.make({
17
- key: Key,
17
+ key: BLUEPRINT_KEY,
18
18
  name: 'Markdown',
19
19
  tools: Blueprint.toolDefinitions({ functions }),
20
20
  instructions: Template.make({
@@ -36,3 +36,11 @@ export const make = () =>
36
36
  `,
37
37
  }),
38
38
  });
39
+
40
+ const blueprint: AppCapabilities.BlueprintDefinition = {
41
+ key: BLUEPRINT_KEY,
42
+ functions,
43
+ make,
44
+ };
45
+
46
+ export default blueprint;
@@ -4,7 +4,8 @@
4
4
 
5
5
  import * as Effect from 'effect/Effect';
6
6
 
7
- import { Capability, Common } from '@dxos/app-framework';
7
+ import { Capability } from '@dxos/app-framework';
8
+ import { AppCapabilities } from '@dxos/app-toolkit';
8
9
  import { Relation } from '@dxos/echo';
9
10
  import { createDocAccessor, getRangeFromCursor } from '@dxos/echo-db';
10
11
  import { type AnchoredTo } from '@dxos/types';
@@ -13,7 +14,7 @@ import { Markdown } from '../../types';
13
14
 
14
15
  export default Capability.makeModule(() =>
15
16
  Effect.succeed(
16
- Capability.contributes(Common.Capability.AnchorSort, {
17
+ Capability.contributes(AppCapabilities.AnchorSort, {
17
18
  key: Markdown.Document.typename,
18
19
  sort: (anchorA: AnchoredTo.AnchoredTo, anchorB: AnchoredTo.AnchoredTo) => {
19
20
  const doc = Relation.getTarget(anchorA) as Markdown.Document;
@@ -4,7 +4,8 @@
4
4
 
5
5
  import * as Effect from 'effect/Effect';
6
6
 
7
- import { Capability, Common } from '@dxos/app-framework';
7
+ import { Capabilities, Capability } from '@dxos/app-framework';
8
+ import { AppCapabilities } from '@dxos/app-toolkit';
8
9
  import { Obj } from '@dxos/echo';
9
10
  import { SpaceOperation } from '@dxos/plugin-space/types';
10
11
  import { isSpace } from '@dxos/react-client/echo';
@@ -18,7 +19,7 @@ export default Capability.makeModule(
18
19
  // Get context for lazy capability access in callbacks.
19
20
  const capabilities = yield* Capability.Service;
20
21
 
21
- return Capability.contributes(Common.Capability.AppGraphSerializer, [
22
+ return Capability.contributes(AppCapabilities.AppGraphSerializer, [
22
23
  {
23
24
  inputType: Markdown.Document.typename,
24
25
  outputType: 'text/markdown',
@@ -44,7 +45,7 @@ export default Capability.makeModule(
44
45
  return;
45
46
  }
46
47
 
47
- const { invokePromise } = capabilities.get(Common.Capability.OperationInvoker);
48
+ const { invokePromise } = capabilities.get(Capabilities.OperationInvoker);
48
49
  const createResult = await invokePromise(MarkdownOperation.Create, { name: data.name, content: data.data });
49
50
  if (!createResult.data?.object) {
50
51
  return undefined;
@@ -9,7 +9,7 @@ import * as Effect from 'effect/Effect';
9
9
  import * as Schema from 'effect/Schema';
10
10
 
11
11
  import { ToolResult, createTool } from '@dxos/ai';
12
- import { Capability, Common } from '@dxos/app-framework';
12
+ import { Capabilities, Capability } from '@dxos/app-framework';
13
13
  import { ArtifactId, createArtifactElement } from '@dxos/assistant';
14
14
  import { defineArtifact } from '@dxos/blueprints';
15
15
  import { Obj } from '@dxos/echo';
@@ -105,6 +105,6 @@ export default Capability.makeModule(() =>
105
105
  ],
106
106
  });
107
107
 
108
- return Capability.contributes(Common.Capability.ArtifactDefinition, definition);
108
+ return Capability.contributes(Capabilities.ArtifactDefinition, definition);
109
109
  }),
110
110
  );
@@ -4,22 +4,14 @@
4
4
 
5
5
  import * as Effect from 'effect/Effect';
6
6
 
7
- import { Capability, Common } from '@dxos/app-framework';
7
+ import { Capability } from '@dxos/app-framework';
8
+ import { AppCapabilities } from '@dxos/app-toolkit';
8
9
 
9
10
  import { MarkdownBlueprint } from '../../blueprints';
10
11
 
11
- export const functions = MarkdownBlueprint.functions;
12
-
13
- export type BlueprintCapabilities = [
14
- Capability.Capability<typeof Common.Capability.Functions>,
15
- Capability.Capability<typeof Common.Capability.BlueprintDefinition>,
16
- ];
17
-
18
- const blueprintDefinition = Capability.makeModule<[], BlueprintCapabilities>(() =>
19
- Effect.succeed([
20
- Capability.contributes(Common.Capability.Functions, functions),
21
- Capability.contributes(Common.Capability.BlueprintDefinition, MarkdownBlueprint.make()),
22
- ]),
23
- );
12
+ const blueprintDefinition = Capability.makeModule<
13
+ [],
14
+ Capability.Capability<typeof AppCapabilities.BlueprintDefinition>[]
15
+ >(() => Effect.succeed([Capability.contributes(AppCapabilities.BlueprintDefinition, MarkdownBlueprint)]));
24
16
 
25
17
  export default blueprintDefinition;
@@ -4,7 +4,7 @@
4
4
 
5
5
  import * as Effect from 'effect/Effect';
6
6
 
7
- import { Capability, Common } from '@dxos/app-framework';
7
+ import { Capabilities, Capability } from '@dxos/app-framework';
8
8
  import { Obj, Ref, Type } from '@dxos/echo';
9
9
  import { OperationResolver } from '@dxos/operation';
10
10
  import { Collection } from '@dxos/schema';
@@ -13,7 +13,7 @@ import { Markdown, MarkdownCapabilities, MarkdownOperation } from '../../types';
13
13
 
14
14
  export default Capability.makeModule(
15
15
  Effect.fnUntraced(function* () {
16
- return Capability.contributes(Common.Capability.OperationResolver, [
16
+ return Capability.contributes(Capabilities.OperationResolver, [
17
17
  OperationResolver.make({
18
18
  operation: MarkdownOperation.OnCreateSpace,
19
19
  handler: Effect.fnUntraced(function* ({ rootCollection }) {
@@ -34,7 +34,7 @@ export default Capability.makeModule(
34
34
  OperationResolver.make({
35
35
  operation: MarkdownOperation.SetViewMode,
36
36
  handler: Effect.fnUntraced(function* ({ id, viewMode }) {
37
- yield* Common.Capability.updateAtomValue(MarkdownCapabilities.State, (current) => ({
37
+ yield* Capabilities.updateAtomValue(MarkdownCapabilities.State, (current) => ({
38
38
  ...current,
39
39
  viewMode: { ...current.viewMode, [id]: viewMode },
40
40
  }));
@@ -5,14 +5,16 @@
5
5
  import * as Effect from 'effect/Effect';
6
6
  import React, { forwardRef, useCallback, useMemo } from 'react';
7
7
 
8
- import { Capability, Common } from '@dxos/app-framework';
8
+ import { Capabilities, Capability } from '@dxos/app-framework';
9
9
  import {
10
- type SurfaceComponentProps,
10
+ Surface,
11
11
  useAtomCapability,
12
12
  useAtomCapabilityState,
13
13
  useCapability,
14
14
  useSettingsState,
15
- } from '@dxos/app-framework/react';
15
+ } from '@dxos/app-framework/ui';
16
+ import { AppCapabilities } from '@dxos/app-toolkit';
17
+ import { type SurfaceComponentProps } from '@dxos/app-toolkit/ui';
16
18
  import { Obj } from '@dxos/echo';
17
19
  import { AttentionCapabilities } from '@dxos/plugin-attention';
18
20
  import { Text } from '@dxos/schema';
@@ -24,8 +26,8 @@ import { Markdown, MarkdownCapabilities } from '../../types';
24
26
 
25
27
  export default Capability.makeModule(() =>
26
28
  Effect.succeed(
27
- Capability.contributes(Common.Capability.ReactSurface, [
28
- Common.createSurface({
29
+ Capability.contributes(Capabilities.ReactSurface, [
30
+ Surface.create({
29
31
  id: `${meta.id}/surface/document`,
30
32
  role: ['article', 'section', 'tabpanel'],
31
33
  filter: (data): data is { subject: Markdown.Document; variant: undefined } =>
@@ -34,7 +36,7 @@ export default Capability.makeModule(() =>
34
36
  return <Container id={Obj.getDXN(data.subject).toString()} subject={data.subject} role={role} ref={ref} />;
35
37
  },
36
38
  }),
37
- Common.createSurface({
39
+ Surface.create({
38
40
  id: `${meta.id}/surface/text`,
39
41
  role: ['article', 'section', 'tabpanel'],
40
42
  filter: (data): data is { id: string; subject: Text.Text } =>
@@ -43,17 +45,17 @@ export default Capability.makeModule(() =>
43
45
  return <Container id={data.id} subject={data.subject} role={role} />;
44
46
  },
45
47
  }),
46
- Common.createSurface({
48
+ Surface.create({
47
49
  id: `${meta.id}/surface/plugin-settings`,
48
50
  role: 'article',
49
- filter: (data): data is { subject: Common.Capability.Settings } =>
50
- Common.Capability.isSettings(data.subject) && data.subject.prefix === meta.id,
51
+ filter: (data): data is { subject: AppCapabilities.Settings } =>
52
+ AppCapabilities.isSettings(data.subject) && data.subject.prefix === meta.id,
51
53
  component: ({ data: { subject } }) => {
52
54
  const { settings, updateSettings } = useSettingsState<Markdown.Settings>(subject.atom);
53
55
  return <MarkdownSettings settings={settings} onSettingsChange={updateSettings} />;
54
56
  },
55
57
  }),
56
- Common.createSurface({
58
+ Surface.create({
57
59
  id: `${meta.id}/surface/preview`,
58
60
  role: 'card--content',
59
61
  filter: (data): data is { subject: Markdown.Document | Text.Text } =>
@@ -4,7 +4,8 @@
4
4
 
5
5
  import * as Effect from 'effect/Effect';
6
6
 
7
- import { Capability, Common } from '@dxos/app-framework';
7
+ import { Capability } from '@dxos/app-framework';
8
+ import { AppCapabilities } from '@dxos/app-toolkit';
8
9
  import { createKvsStore } from '@dxos/effect';
9
10
 
10
11
  import { meta } from '../../meta';
@@ -26,7 +27,7 @@ export default Capability.makeModule(() =>
26
27
 
27
28
  return [
28
29
  Capability.contributes(MarkdownCapabilities.Settings, settingsAtom),
29
- Capability.contributes(Common.Capability.Settings, {
30
+ Capability.contributes(AppCapabilities.Settings, {
30
31
  prefix: meta.id,
31
32
  schema: Markdown.Settings,
32
33
  atom: settingsAtom,
package/src/cli/plugin.ts CHANGED
@@ -4,7 +4,8 @@
4
4
 
5
5
  import * as Effect from 'effect/Effect';
6
6
 
7
- import { Common, Plugin } from '@dxos/app-framework';
7
+ import { Plugin } from '@dxos/app-framework';
8
+ import { AppPlugin } from '@dxos/app-toolkit';
8
9
  import { type CreateObject } from '@dxos/plugin-space/types';
9
10
  import { Text } from '@dxos/schema';
10
11
 
@@ -14,8 +15,7 @@ import { meta } from '../meta';
14
15
  import { Markdown } from '../types';
15
16
 
16
17
  export const MarkdownPlugin = Plugin.define(meta).pipe(
17
- Common.Plugin.addSchemaModule({ schema: [Markdown.Document, Text.Text] }),
18
- Common.Plugin.addMetadataModule({
18
+ AppPlugin.addMetadataModule({
19
19
  metadata: {
20
20
  id: Markdown.Document.typename,
21
21
  metadata: {
@@ -24,6 +24,7 @@ export const MarkdownPlugin = Plugin.define(meta).pipe(
24
24
  },
25
25
  },
26
26
  }),
27
- Common.Plugin.addOperationResolverModule({ activate: OperationResolver }),
27
+ AppPlugin.addOperationResolverModule({ activate: OperationResolver }),
28
+ AppPlugin.addSchemaModule({ schema: [Markdown.Document, Text.Text] }),
28
29
  Plugin.make,
29
30
  );
@@ -39,7 +39,7 @@ const meta: Meta<typeof MarkdownCardStory> = {
39
39
  title: 'plugins/plugin-markdown/Card',
40
40
  component: MarkdownCardStory,
41
41
  decorators: [
42
- withTheme,
42
+ withTheme(),
43
43
  withPluginManager({
44
44
  plugins: [OperationPlugin()],
45
45
  }),
@@ -24,15 +24,17 @@ export const MarkdownCard = ({ subject }: MarkdownCardProps) => {
24
24
  return (
25
25
  <Card.Content>
26
26
  {snippet && (
27
- <div className='max-h-[300px] overflow-hidden'>
27
+ <Card.Row className='max-h-[300px] overflow-hidden'>
28
28
  <MarkdownEditor.Root id={subject.id} viewMode='readonly'>
29
29
  <MarkdownEditor.Content initialValue={snippet} slots={{}} classNames='!bg-transparent' />
30
30
  </MarkdownEditor.Root>
31
- </div>
31
+ </Card.Row>
32
32
  )}
33
- <Card.Text classNames='text-xs text-description'>
34
- {info.words} {t('words label', { count: info.words })}
35
- </Card.Text>
33
+ <Card.Row>
34
+ <Card.Text classNames='text-xs text-description'>
35
+ {info.words} {t('words label', { count: info.words })}
36
+ </Card.Text>
37
+ </Card.Row>
36
38
  </Card.Content>
37
39
  );
38
40
  };