@dxos/plugin-presenter 0.8.4-main.72ec0f3 → 0.8.4-main.74a063c4e0

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 (144) hide show
  1. package/dist/lib/browser/PresenterSettings-DZO2DC53.mjs +29 -0
  2. package/dist/lib/browser/PresenterSettings-DZO2DC53.mjs.map +7 -0
  3. package/dist/lib/browser/chunk-J5LGTIGS.mjs +10 -0
  4. package/dist/lib/browser/chunk-J5LGTIGS.mjs.map +7 -0
  5. package/dist/lib/browser/index.mjs +533 -46
  6. package/dist/lib/browser/index.mjs.map +4 -4
  7. package/dist/lib/browser/meta.json +1 -1
  8. package/dist/lib/browser/operations/index.mjs +30 -0
  9. package/dist/lib/browser/operations/index.mjs.map +7 -0
  10. package/dist/types/src/PresenterPlugin.d.ts +2 -1
  11. package/dist/types/src/PresenterPlugin.d.ts.map +1 -1
  12. package/dist/types/src/capabilities/app-graph-builder.d.ts +4 -2
  13. package/dist/types/src/capabilities/app-graph-builder.d.ts.map +1 -1
  14. package/dist/types/src/capabilities/index.d.ts +8 -3
  15. package/dist/types/src/capabilities/index.d.ts.map +1 -1
  16. package/dist/types/src/capabilities/react-surface.d.ts +3 -2
  17. package/dist/types/src/capabilities/react-surface.d.ts.map +1 -1
  18. package/dist/types/src/capabilities/settings.d.ts +8 -2
  19. package/dist/types/src/capabilities/settings.d.ts.map +1 -1
  20. package/dist/types/src/components/Markdown/Panel.d.ts +8 -0
  21. package/dist/types/src/components/Markdown/Panel.d.ts.map +1 -0
  22. package/dist/types/src/components/Markdown/{Container.stories.d.ts → Panel.stories.d.ts} +1 -1
  23. package/dist/types/src/components/Markdown/Panel.stories.d.ts.map +1 -0
  24. package/dist/types/src/components/Markdown/Slide.d.ts +1 -1
  25. package/dist/types/src/components/Markdown/Slide.d.ts.map +1 -1
  26. package/dist/types/src/components/Markdown/index.d.ts +1 -1
  27. package/dist/types/src/components/Markdown/index.d.ts.map +1 -1
  28. package/dist/types/src/components/Presenter/Layout.d.ts +9 -5
  29. package/dist/types/src/components/Presenter/Layout.d.ts.map +1 -1
  30. package/dist/types/src/components/Presenter/Pager.d.ts +1 -1
  31. package/dist/types/src/components/Presenter/Pager.d.ts.map +1 -1
  32. package/dist/types/src/components/PresenterSettings/PresenterSettings.d.ts +6 -0
  33. package/dist/types/src/components/PresenterSettings/PresenterSettings.d.ts.map +1 -0
  34. package/dist/types/src/components/PresenterSettings/PresenterSettings.stories.d.ts +25 -0
  35. package/dist/types/src/components/PresenterSettings/PresenterSettings.stories.d.ts.map +1 -0
  36. package/dist/types/src/components/PresenterSettings/index.d.ts +2 -0
  37. package/dist/types/src/components/PresenterSettings/index.d.ts.map +1 -0
  38. package/dist/types/src/components/RevealPlayer/RevealPlayer.d.ts +9 -4
  39. package/dist/types/src/components/RevealPlayer/RevealPlayer.d.ts.map +1 -1
  40. package/dist/types/src/components/RevealPlayer/RevealPlayer.stories.d.ts +12 -5
  41. package/dist/types/src/components/RevealPlayer/RevealPlayer.stories.d.ts.map +1 -1
  42. package/dist/types/src/components/index.d.ts +5 -27
  43. package/dist/types/src/components/index.d.ts.map +1 -1
  44. package/dist/types/src/containers/CollectionPresenterContainer/CollectionPresenterContainer.d.ts +6 -0
  45. package/dist/types/src/containers/CollectionPresenterContainer/CollectionPresenterContainer.d.ts.map +1 -0
  46. package/dist/types/src/containers/CollectionPresenterContainer/index.d.ts +2 -0
  47. package/dist/types/src/containers/CollectionPresenterContainer/index.d.ts.map +1 -0
  48. package/dist/types/src/{components/MarkdownSlide.d.ts → containers/DocumentPresenterContainer/DocumentPresenterContainer.d.ts} +2 -3
  49. package/dist/types/src/containers/DocumentPresenterContainer/DocumentPresenterContainer.d.ts.map +1 -0
  50. package/dist/types/src/containers/DocumentPresenterContainer/index.d.ts +2 -0
  51. package/dist/types/src/containers/DocumentPresenterContainer/index.d.ts.map +1 -0
  52. package/dist/types/src/containers/MarkdownSlide/MarkdownSlide.d.ts +8 -0
  53. package/dist/types/src/containers/MarkdownSlide/MarkdownSlide.d.ts.map +1 -0
  54. package/dist/types/src/containers/MarkdownSlide/index.d.ts +2 -0
  55. package/dist/types/src/containers/MarkdownSlide/index.d.ts.map +1 -0
  56. package/dist/types/src/containers/index.d.ts +5 -0
  57. package/dist/types/src/containers/index.d.ts.map +1 -0
  58. package/dist/types/src/meta.d.ts +2 -2
  59. package/dist/types/src/meta.d.ts.map +1 -1
  60. package/dist/types/src/operations/definitions.d.ts +17 -0
  61. package/dist/types/src/operations/definitions.d.ts.map +1 -0
  62. package/dist/types/src/operations/index.d.ts +2 -0
  63. package/dist/types/src/operations/index.d.ts.map +1 -0
  64. package/dist/types/src/translations.d.ts +6 -5
  65. package/dist/types/src/translations.d.ts.map +1 -1
  66. package/dist/types/src/types/Settings.d.ts +6 -0
  67. package/dist/types/src/types/Settings.d.ts.map +1 -0
  68. package/dist/types/src/types/index.d.ts +18 -0
  69. package/dist/types/src/types/index.d.ts.map +1 -0
  70. package/dist/types/src/useExitPresenter.d.ts +3 -4
  71. package/dist/types/src/useExitPresenter.d.ts.map +1 -1
  72. package/dist/types/tsconfig.tsbuildinfo +1 -1
  73. package/package.json +62 -41
  74. package/src/PresenterPlugin.tsx +12 -25
  75. package/src/capabilities/app-graph-builder.ts +91 -104
  76. package/src/capabilities/index.ts +4 -4
  77. package/src/capabilities/react-surface.tsx +56 -53
  78. package/src/capabilities/settings.ts +24 -12
  79. package/src/components/Markdown/{Container.stories.tsx → Panel.stories.tsx} +7 -7
  80. package/src/components/Markdown/{Container.tsx → Panel.tsx} +6 -6
  81. package/src/components/Markdown/Slide.stories.tsx +4 -4
  82. package/src/components/Markdown/Slide.tsx +2 -2
  83. package/src/components/Markdown/index.ts +1 -1
  84. package/src/components/Markdown/theme.ts +6 -6
  85. package/src/components/Presenter/Layout.tsx +24 -23
  86. package/src/components/Presenter/Pager.stories.tsx +2 -2
  87. package/src/components/Presenter/Pager.tsx +2 -2
  88. package/src/components/PresenterSettings/PresenterSettings.stories.tsx +31 -0
  89. package/src/components/PresenterSettings/PresenterSettings.tsx +32 -0
  90. package/src/components/PresenterSettings/index.ts +5 -0
  91. package/src/components/RevealPlayer/RevealPlayer.stories.tsx +3 -4
  92. package/src/components/RevealPlayer/RevealPlayer.tsx +104 -102
  93. package/src/components/index.ts +5 -5
  94. package/src/containers/CollectionPresenterContainer/CollectionPresenterContainer.tsx +44 -0
  95. package/src/containers/CollectionPresenterContainer/index.ts +5 -0
  96. package/src/containers/DocumentPresenterContainer/DocumentPresenterContainer.tsx +24 -0
  97. package/src/containers/DocumentPresenterContainer/index.ts +5 -0
  98. package/src/{components → containers/MarkdownSlide}/MarkdownSlide.tsx +9 -7
  99. package/src/containers/MarkdownSlide/index.ts +5 -0
  100. package/src/containers/index.ts +9 -0
  101. package/src/meta.ts +3 -3
  102. package/src/operations/definitions.ts +23 -0
  103. package/src/operations/index.ts +5 -0
  104. package/src/testing.ts +6 -6
  105. package/src/translations.ts +6 -5
  106. package/src/types/Settings.ts +15 -0
  107. package/src/types/index.ts +30 -0
  108. package/src/useExitPresenter.ts +25 -25
  109. package/dist/lib/browser/CollectionPresenterContainer-4JFW3EKV.mjs +0 -199
  110. package/dist/lib/browser/CollectionPresenterContainer-4JFW3EKV.mjs.map +0 -7
  111. package/dist/lib/browser/DocumentPresenterContainer-SX32NAEH.mjs +0 -183
  112. package/dist/lib/browser/DocumentPresenterContainer-SX32NAEH.mjs.map +0 -7
  113. package/dist/lib/browser/MarkdownSlide-MEXR54L5.mjs +0 -223
  114. package/dist/lib/browser/MarkdownSlide-MEXR54L5.mjs.map +0 -7
  115. package/dist/lib/browser/app-graph-builder-UJEKUKAP.mjs +0 -106
  116. package/dist/lib/browser/app-graph-builder-UJEKUKAP.mjs.map +0 -7
  117. package/dist/lib/browser/chunk-BJWK5GQV.mjs +0 -38
  118. package/dist/lib/browser/chunk-BJWK5GQV.mjs.map +0 -7
  119. package/dist/lib/browser/chunk-Q3H4KEFB.mjs +0 -41
  120. package/dist/lib/browser/chunk-Q3H4KEFB.mjs.map +0 -7
  121. package/dist/lib/browser/chunk-TMTUZQCK.mjs +0 -18
  122. package/dist/lib/browser/chunk-TMTUZQCK.mjs.map +0 -7
  123. package/dist/lib/browser/chunk-VRRPVTKA.mjs +0 -39
  124. package/dist/lib/browser/chunk-VRRPVTKA.mjs.map +0 -7
  125. package/dist/lib/browser/react-surface-S4T6X72R.mjs +0 -57
  126. package/dist/lib/browser/react-surface-S4T6X72R.mjs.map +0 -7
  127. package/dist/lib/browser/settings-57TUVRW3.mjs +0 -22
  128. package/dist/lib/browser/settings-57TUVRW3.mjs.map +0 -7
  129. package/dist/types/src/components/CollectionPresenterContainer.d.ts +0 -7
  130. package/dist/types/src/components/CollectionPresenterContainer.d.ts.map +0 -1
  131. package/dist/types/src/components/DocumentPresenterContainer.d.ts +0 -7
  132. package/dist/types/src/components/DocumentPresenterContainer.d.ts.map +0 -1
  133. package/dist/types/src/components/Markdown/Container.d.ts +0 -8
  134. package/dist/types/src/components/Markdown/Container.d.ts.map +0 -1
  135. package/dist/types/src/components/Markdown/Container.stories.d.ts.map +0 -1
  136. package/dist/types/src/components/MarkdownSlide.d.ts.map +0 -1
  137. package/dist/types/src/components/PresenterSettings.d.ts +0 -6
  138. package/dist/types/src/components/PresenterSettings.d.ts.map +0 -1
  139. package/dist/types/src/types.d.ts +0 -40
  140. package/dist/types/src/types.d.ts.map +0 -1
  141. package/src/components/CollectionPresenterContainer.tsx +0 -44
  142. package/src/components/DocumentPresenterContainer.tsx +0 -23
  143. package/src/components/PresenterSettings.tsx +0 -30
  144. package/src/types.ts +0 -46
package/package.json CHANGED
@@ -1,18 +1,36 @@
1
1
  {
2
2
  "name": "@dxos/plugin-presenter",
3
- "version": "0.8.4-main.72ec0f3",
3
+ "version": "0.8.4-main.74a063c4e0",
4
4
  "description": "Braneframe presenter plugin",
5
5
  "homepage": "https://dxos.org",
6
6
  "bugs": "https://github.com/dxos/dxos/issues",
7
+ "repository": {
8
+ "type": "git",
9
+ "url": "https://github.com/dxos/dxos"
10
+ },
7
11
  "license": "MIT",
8
12
  "author": "DXOS.org",
9
13
  "sideEffects": true,
10
14
  "type": "module",
15
+ "imports": {
16
+ "#capabilities": "./src/capabilities/index.ts",
17
+ "#components": "./src/components/index.ts",
18
+ "#containers": "./src/containers/index.ts",
19
+ "#meta": "./src/meta.ts",
20
+ "#operations": "./src/operations/index.ts",
21
+ "#testing": "./src/testing.ts",
22
+ "#types": "./src/types/index.ts"
23
+ },
11
24
  "exports": {
12
25
  ".": {
13
26
  "source": "./src/index.ts",
14
- "types": "./dist/types/src/index.d.ts",
15
- "browser": "./dist/lib/browser/index.mjs"
27
+ "browser": "./dist/lib/browser/index.mjs",
28
+ "types": "./dist/types/src/index.d.ts"
29
+ },
30
+ "./operations": {
31
+ "source": "./src/operations/index.ts",
32
+ "types": "./dist/types/src/operations/index.d.ts",
33
+ "browser": "./dist/lib/browser/operations/index.mjs"
16
34
  }
17
35
  },
18
36
  "types": "dist/types/src/index.d.ts",
@@ -24,8 +42,8 @@
24
42
  "src"
25
43
  ],
26
44
  "dependencies": {
27
- "@preact-signals/safe-react": "^0.9.0",
28
- "@preact/signals-core": "^1.12.1",
45
+ "@effect-atom/atom": "^0.5.1",
46
+ "@effect-atom/atom-react": "^0.5.0",
29
47
  "hastscript": "^7.1.0",
30
48
  "highlight.js": "^11.9.0",
31
49
  "marked": "^12.0.2",
@@ -36,47 +54,50 @@
36
54
  "remark-frontmatter": "^5.0.0",
37
55
  "remark-parse-frontmatter": "^1.0.3",
38
56
  "reveal.js": "^5.1.0",
39
- "@dxos/app-framework": "0.8.4-main.72ec0f3",
40
- "@dxos/async": "0.8.4-main.72ec0f3",
41
- "@dxos/echo": "0.8.4-main.72ec0f3",
42
- "@dxos/log": "0.8.4-main.72ec0f3",
43
- "@dxos/live-object": "0.8.4-main.72ec0f3",
44
- "@dxos/local-storage": "0.8.4-main.72ec0f3",
45
- "@dxos/plugin-deck": "0.8.4-main.72ec0f3",
46
- "@dxos/plugin-client": "0.8.4-main.72ec0f3",
47
- "@dxos/plugin-graph": "0.8.4-main.72ec0f3",
48
- "@dxos/plugin-markdown": "0.8.4-main.72ec0f3",
49
- "@dxos/plugin-stack": "0.8.4-main.72ec0f3",
50
- "@dxos/react-client": "0.8.4-main.72ec0f3",
51
- "@dxos/react-ui-form": "0.8.4-main.72ec0f3",
52
- "@dxos/types": "0.8.4-main.72ec0f3",
53
- "@dxos/util": "0.8.4-main.72ec0f3",
54
- "@dxos/react-ui-stack": "0.8.4-main.72ec0f3",
55
- "@dxos/schema": "0.8.4-main.72ec0f3"
57
+ "@dxos/app-framework": "0.8.4-main.74a063c4e0",
58
+ "@dxos/app-toolkit": "0.8.4-main.74a063c4e0",
59
+ "@dxos/async": "0.8.4-main.74a063c4e0",
60
+ "@dxos/echo": "0.8.4-main.74a063c4e0",
61
+ "@dxos/effect": "0.8.4-main.74a063c4e0",
62
+ "@dxos/log": "0.8.4-main.74a063c4e0",
63
+ "@dxos/operation": "0.8.4-main.74a063c4e0",
64
+ "@dxos/plugin-client": "0.8.4-main.74a063c4e0",
65
+ "@dxos/plugin-graph": "0.8.4-main.74a063c4e0",
66
+ "@dxos/plugin-deck": "0.8.4-main.74a063c4e0",
67
+ "@dxos/plugin-stack": "0.8.4-main.74a063c4e0",
68
+ "@dxos/react-client": "0.8.4-main.74a063c4e0",
69
+ "@dxos/plugin-markdown": "0.8.4-main.74a063c4e0",
70
+ "@dxos/react-ui-attention": "0.8.4-main.74a063c4e0",
71
+ "@dxos/react-ui-form": "0.8.4-main.74a063c4e0",
72
+ "@dxos/react-ui-mosaic": "0.8.4-main.74a063c4e0",
73
+ "@dxos/react-ui-stack": "0.8.4-main.74a063c4e0",
74
+ "@dxos/types": "0.8.4-main.74a063c4e0",
75
+ "@dxos/schema": "0.8.4-main.74a063c4e0",
76
+ "@dxos/util": "0.8.4-main.74a063c4e0"
56
77
  },
57
78
  "devDependencies": {
58
- "@effect-atom/atom-react": "^0.3.4",
59
- "@effect/platform": "0.92.1",
60
- "@types/react": "~19.2.2",
61
- "@types/react-dom": "~19.2.2",
79
+ "@effect-atom/atom-react": "^0.5.0",
80
+ "@effect/platform": "0.94.4",
81
+ "@types/react": "~19.2.7",
82
+ "@types/react-dom": "~19.2.3",
62
83
  "@types/reveal.js": "^5.0.3",
63
- "effect": "3.18.3",
64
- "react": "~19.2.0",
65
- "react-dom": "~19.2.0",
66
- "vite": "7.1.9",
67
- "@dxos/random": "0.8.4-main.72ec0f3",
68
- "@dxos/react-ui": "0.8.4-main.72ec0f3",
69
- "@dxos/storybook-utils": "0.8.4-main.72ec0f3",
70
- "@dxos/react-ui-theme": "0.8.4-main.72ec0f3"
84
+ "effect": "3.20.0",
85
+ "react": "~19.2.3",
86
+ "react-dom": "~19.2.3",
87
+ "vite": "^7.1.11",
88
+ "@dxos/random": "0.8.4-main.74a063c4e0",
89
+ "@dxos/react-ui": "0.8.4-main.74a063c4e0",
90
+ "@dxos/ui-theme": "0.8.4-main.74a063c4e0",
91
+ "@dxos/storybook-utils": "0.8.4-main.74a063c4e0"
71
92
  },
72
93
  "peerDependencies": {
73
- "@effect-atom/atom-react": "^0.3.4",
74
- "@effect/platform": "^0.80.12",
75
- "effect": "^3.13.3",
76
- "react": "^19.0.0",
77
- "react-dom": "^19.0.0",
78
- "@dxos/react-ui": "0.8.4-main.72ec0f3",
79
- "@dxos/react-ui-theme": "0.8.4-main.72ec0f3"
94
+ "@effect-atom/atom-react": "^0.5.0",
95
+ "@effect/platform": "0.94.4",
96
+ "effect": "3.20.0",
97
+ "react": "~19.2.3",
98
+ "react-dom": "~19.2.3",
99
+ "@dxos/react-ui": "0.8.4-main.74a063c4e0",
100
+ "@dxos/ui-theme": "0.8.4-main.74a063c4e0"
80
101
  },
81
102
  "publishConfig": {
82
103
  "access": "public"
@@ -2,34 +2,21 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import { Capabilities, Events, contributes, defineModule, definePlugin } from '@dxos/app-framework';
5
+ import { Plugin } from '@dxos/app-framework';
6
+ import { AppPlugin } from '@dxos/app-toolkit';
7
+
8
+ import { AppGraphBuilder, PresenterSettings, ReactSurface } from '#capabilities';
9
+ import { meta } from '#meta';
6
10
 
7
- import { AppGraphBuilder, PresenterSettings, ReactSurface } from './capabilities';
8
- import { meta } from './meta';
9
11
  import { translations } from './translations';
10
12
 
11
13
  // TODO(burdon): Only scale markdown content.
12
14
  // TODO(burdon): Map stack content; Slide content type (e.g., markdown, sketch, IPFS image, table, etc.)
13
15
 
14
- export const PresenterPlugin = definePlugin(meta, () => [
15
- defineModule({
16
- id: `${meta.id}/module/settings`,
17
- activatesOn: Events.SetupSettings,
18
- activate: PresenterSettings,
19
- }),
20
- defineModule({
21
- id: `${meta.id}/module/translations`,
22
- activatesOn: Events.SetupTranslations,
23
- activate: () => contributes(Capabilities.Translations, translations),
24
- }),
25
- defineModule({
26
- id: `${meta.id}/module/react-surface`,
27
- activatesOn: Events.SetupReactSurface,
28
- activate: ReactSurface,
29
- }),
30
- defineModule({
31
- id: `${meta.id}/module/app-graph-builder`,
32
- activatesOn: Events.SetupAppGraph,
33
- activate: AppGraphBuilder,
34
- }),
35
- ]);
16
+ export const PresenterPlugin = Plugin.define(meta).pipe(
17
+ AppPlugin.addAppGraphModule({ activate: AppGraphBuilder }),
18
+ AppPlugin.addSettingsModule({ activate: PresenterSettings }),
19
+ AppPlugin.addSurfaceModule({ activate: ReactSurface }),
20
+ AppPlugin.addTranslationsModule({ translations }),
21
+ Plugin.make,
22
+ );
@@ -2,116 +2,103 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import { Atom } from '@effect-atom/atom-react';
6
- import * as Function from 'effect/Function';
5
+ import * as Effect from 'effect/Effect';
7
6
  import * as Option from 'effect/Option';
8
7
 
9
- import { Capabilities, LayoutAction, type PluginContext, contributes, createIntent } from '@dxos/app-framework';
8
+ import { Capabilities, Capability } from '@dxos/app-framework';
9
+ import { AppCapabilities, AppNode, LayoutOperation, getObjectPathFromObject, getSpacePath } from '@dxos/app-toolkit';
10
10
  import { Obj } from '@dxos/echo';
11
+ import { Collection } from '@dxos/echo';
12
+ import { Operation } from '@dxos/operation';
11
13
  import { DeckCapabilities } from '@dxos/plugin-deck';
12
- import { ATTENDABLE_PATH_SEPARATOR, DeckAction } from '@dxos/plugin-deck/types';
13
- import { atomFromSignal, createExtension } from '@dxos/plugin-graph';
14
+ import { DeckOperation } from '@dxos/plugin-deck/operations';
15
+ import { GraphBuilder, type Node, NodeMatcher } from '@dxos/plugin-graph';
14
16
  import { Markdown } from '@dxos/plugin-markdown/types';
15
- import { getSpace } from '@dxos/react-client/echo';
16
- import { Collection } from '@dxos/schema';
17
+ import { linkedSegment } from '@dxos/react-ui-attention';
17
18
 
18
- import { meta } from '../meta';
19
- import { PresenterAction, type PresenterSettingsProps } from '../types';
19
+ import { meta } from '#meta';
20
+ import { PresenterOperation } from '#operations';
21
+ import { PresenterCapabilities } from '#types';
20
22
 
21
- export default (context: PluginContext) =>
22
- contributes(
23
- Capabilities.AppGraphBuilder,
24
- createExtension({
25
- id: `${meta.id}/root`,
23
+ /** Match nodes that can be presented (Collection or Document). */
24
+ const whenPresentable = (node: Node.Node) =>
25
+ Option.orElse(NodeMatcher.whenEchoType(Collection.Collection)(node), () =>
26
+ NodeMatcher.whenEchoType(Markdown.Document)(node),
27
+ );
28
+
29
+ export default Capability.makeModule(
30
+ Effect.fnUntraced(function* () {
31
+ const capabilities = yield* Capability.Service;
32
+
33
+ const extensions = yield* GraphBuilder.createExtension({
34
+ id: 'root',
26
35
  // TODO(wittjosiah): This is a hack to work around presenter previously relying on "variant". Remove.
27
- connector: (node) =>
28
- Atom.make((get) =>
29
- Function.pipe(
30
- get(node),
31
- Option.flatMap((node) => {
32
- const [settingsStore] = get(context.capabilities(Capabilities.SettingsStore));
33
- const settings = get(
34
- atomFromSignal(() => settingsStore?.getStore<PresenterSettingsProps>(meta.id)?.value),
35
- );
36
- const isPresentable = settings?.presentCollections
37
- ? Obj.instanceOf(Collection.Collection, node.data) || Obj.instanceOf(Markdown.Document, node.data)
38
- : Obj.instanceOf(Markdown.Document, node.data);
39
- return isPresentable ? Option.some(node.data) : Option.none();
40
- }),
41
- Option.map((object) => {
42
- const id = Obj.getDXN(object).toString();
43
- return [
44
- {
45
- id: [id, 'presenter'].join(ATTENDABLE_PATH_SEPARATOR),
46
- data: { type: meta.id, object },
47
- type: meta.id,
48
- properties: {
49
- label: 'Presenter',
50
- icon: 'ph--presentation--regular',
51
- disposition: 'hidden',
52
- },
53
- },
54
- ];
55
- }),
56
- Option.getOrElse(() => []),
57
- ),
58
- ),
59
- actions: (node) =>
60
- Atom.make((get) =>
61
- Function.pipe(
62
- get(node),
63
- Option.flatMap((node) => {
64
- const [settingsStore] = get(context.capabilities(Capabilities.SettingsStore));
65
- const settings = get(
66
- atomFromSignal(() => settingsStore?.getStore<PresenterSettingsProps>(meta.id)?.value),
67
- );
68
- const isPresentable = settings?.presentCollections
69
- ? Obj.instanceOf(Collection.Collection, node.data) || Obj.instanceOf(Markdown.Document, node.data)
70
- : Obj.instanceOf(Markdown.Document, node.data);
71
- return isPresentable ? Option.some(node.data) : Option.none();
72
- }),
73
- Option.map((object) => {
74
- const id = Obj.getDXN(object).toString();
75
- const spaceId = getSpace(object)?.id;
76
- return [
77
- {
78
- id: `${PresenterAction.TogglePresentation._tag}/${id}`,
79
- // TODO(burdon): Allow function so can generate state when activated.
80
- // So can set explicit fullscreen state coordinated with current presenter state.
81
- data: async () => {
82
- const { dispatchPromise: dispatch } = context.getCapability(Capabilities.IntentDispatcher);
83
- const layout = context.getCapability(DeckCapabilities.MutableDeckState);
84
- const presenterId = [id, 'presenter'].join(ATTENDABLE_PATH_SEPARATOR);
85
- if (!layout.deck.fullscreen) {
86
- void dispatch(
87
- createIntent(DeckAction.Adjust, {
88
- type: 'solo--fullscreen',
89
- id: presenterId,
90
- }),
91
- );
92
- }
93
- await dispatch(
94
- createIntent(LayoutAction.Open, {
95
- part: 'main',
96
- subject: [presenterId],
97
- options: { workspace: spaceId },
98
- }),
99
- );
100
- },
101
- properties: {
102
- label: ['toggle presentation label', { ns: meta.id }],
103
- icon: 'ph--presentation--regular',
104
- disposition: 'list-item',
105
- keyBinding: {
106
- macos: 'shift+meta+p',
107
- windows: 'shift+alt+p',
108
- },
109
- },
110
- },
111
- ];
36
+ match: whenPresentable,
37
+ connector: (object, get) => {
38
+ const settingsAtom = capabilities.get(PresenterCapabilities.Settings);
39
+ const settings = get(settingsAtom);
40
+ const isPresentable = settings?.presentCollections
41
+ ? Obj.instanceOf(Collection.Collection, object) || Obj.instanceOf(Markdown.Document, object)
42
+ : Obj.instanceOf(Markdown.Document, object);
43
+ if (!isPresentable) {
44
+ return Effect.succeed([]);
45
+ }
46
+
47
+ return Effect.succeed([
48
+ AppNode.makeCompanion({
49
+ id: linkedSegment('presenter'),
50
+ label: 'Presenter',
51
+ icon: 'ph--presentation--regular',
52
+ data: { type: meta.id, object },
53
+ }),
54
+ ]);
55
+ },
56
+ actions: (object, get) => {
57
+ const settingsAtom = capabilities.get(PresenterCapabilities.Settings);
58
+ const settings = get(settingsAtom);
59
+ const isPresentable = settings?.presentCollections
60
+ ? Obj.instanceOf(Collection.Collection, object) || Obj.instanceOf(Markdown.Document, object)
61
+ : Obj.instanceOf(Markdown.Document, object);
62
+ const db = Obj.getDatabase(object);
63
+ if (!isPresentable || !db) {
64
+ return Effect.succeed([]);
65
+ }
66
+ const objectPath = getObjectPathFromObject(object);
67
+
68
+ return Effect.succeed([
69
+ {
70
+ id: PresenterOperation.TogglePresentation.meta.key,
71
+ // TODO(burdon): Allow function so can generate state when activated.
72
+ // So can set explicit fullscreen state coordinated with current presenter state.
73
+ data: Effect.fnUntraced(function* () {
74
+ const deckState = yield* Capabilities.getAtomValue(DeckCapabilities.State);
75
+ const deck = deckState.decks[deckState.activeDeck];
76
+ const presenterId = `${objectPath}/${linkedSegment('presenter')}`;
77
+ if (!deck?.fullscreen) {
78
+ yield* Operation.invoke(DeckOperation.Adjust, {
79
+ type: 'solo--fullscreen' as const,
80
+ id: presenterId,
81
+ });
82
+ }
83
+ yield* Operation.invoke(LayoutOperation.Open, {
84
+ subject: [presenterId],
85
+ workspace: getSpacePath(db.spaceId),
86
+ });
112
87
  }),
113
- Option.getOrElse(() => []),
114
- ),
115
- ),
116
- }),
117
- );
88
+ properties: {
89
+ label: ['toggle-presentation.label', { ns: meta.id }],
90
+ icon: 'ph--presentation--regular',
91
+ disposition: 'list-item',
92
+ keyBinding: {
93
+ macos: 'shift+meta+p',
94
+ windows: 'shift+alt+p',
95
+ },
96
+ },
97
+ },
98
+ ]);
99
+ },
100
+ });
101
+
102
+ return Capability.contributes(AppCapabilities.AppGraphBuilder, extensions);
103
+ }),
104
+ );
@@ -2,8 +2,8 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import { lazy } from '@dxos/app-framework';
5
+ import { Capability } from '@dxos/app-framework';
6
6
 
7
- export const AppGraphBuilder = lazy(() => import('./app-graph-builder'));
8
- export const ReactSurface = lazy(() => import('./react-surface'));
9
- export const PresenterSettings = lazy(() => import('./settings'));
7
+ export const AppGraphBuilder = Capability.lazy('AppGraphBuilder', () => import('./app-graph-builder'));
8
+ export const ReactSurface = Capability.lazy('ReactSurface', () => import('./react-surface'));
9
+ export const PresenterSettings = Capability.lazy('PresenterSettings', () => import('./settings'));
@@ -2,62 +2,65 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
+ import * as Effect from 'effect/Effect';
5
6
  import React from 'react';
6
7
 
7
- import { Capabilities, contributes, createSurface } from '@dxos/app-framework';
8
+ import { Capabilities, Capability } from '@dxos/app-framework';
9
+ import { Surface, useSettingsState } from '@dxos/app-framework/ui';
10
+ import { AppSurface } from '@dxos/app-toolkit/ui';
8
11
  import { Obj } from '@dxos/echo';
9
- import { SettingsStore } from '@dxos/local-storage';
12
+ import { Collection } from '@dxos/echo';
10
13
  import { Markdown } from '@dxos/plugin-markdown/types';
11
- import { Collection } from '@dxos/schema';
12
14
 
13
- import {
14
- CollectionPresenterContainer,
15
- DocumentPresenterContainer,
16
- MarkdownSlide,
17
- PresenterSettings,
18
- } from '../components';
19
- import { meta } from '../meta';
20
- import { type PresenterSettingsProps } from '../types';
15
+ import { PresenterSettings } from '#components';
16
+ import { CollectionPresenterContainer, DocumentPresenterContainer, MarkdownSlide } from '#containers';
17
+ import { meta } from '#meta';
18
+ import { type Settings } from '#types';
21
19
 
22
- export default () =>
23
- contributes(Capabilities.ReactSurface, [
24
- createSurface({
25
- id: `${meta.id}/document`,
26
- role: 'article',
27
- position: 'hoist',
28
- filter: (data): data is { subject: { type: typeof meta.id; object: Markdown.Document } } =>
29
- !!data.subject &&
30
- typeof data.subject === 'object' &&
31
- 'type' in data.subject &&
32
- 'object' in data.subject &&
33
- data.subject.type === meta.id &&
34
- Obj.instanceOf(Markdown.Document, data.subject.object),
35
- component: ({ data }) => <DocumentPresenterContainer document={data.subject.object} />,
36
- }),
37
- createSurface({
38
- id: `${meta.id}/collection`,
39
- role: 'article',
40
- position: 'hoist',
41
- filter: (data): data is { subject: { type: typeof meta.id; object: Collection.Collection } } =>
42
- !!data.subject &&
43
- typeof data.subject === 'object' &&
44
- 'type' in data.subject &&
45
- 'object' in data.subject &&
46
- data.subject.type === meta.id &&
47
- Obj.instanceOf(Collection.Collection, data.subject.object),
48
- component: ({ data }) => <CollectionPresenterContainer collection={data.subject.object} />,
49
- }),
50
- createSurface({
51
- id: `${meta.id}/slide`,
52
- role: 'slide',
53
- filter: (data): data is { subject: Markdown.Document } => Obj.instanceOf(Markdown.Document, data.subject),
54
- component: ({ data }) => <MarkdownSlide document={data.subject} />,
55
- }),
56
- createSurface({
57
- id: `${meta.id}/plugin-settings`,
58
- role: 'article',
59
- filter: (data): data is { subject: SettingsStore<PresenterSettingsProps> } =>
60
- data.subject instanceof SettingsStore && data.subject.prefix === meta.id,
61
- component: ({ data: { subject } }) => <PresenterSettings settings={subject.value} />,
62
- }),
63
- ]);
20
+ export default Capability.makeModule(() =>
21
+ Effect.succeed(
22
+ Capability.contributes(Capabilities.ReactSurface, [
23
+ Surface.create({
24
+ id: 'document',
25
+ role: 'article',
26
+ position: 'hoist',
27
+ filter: (data): data is { subject: { type: typeof meta.id; object: Markdown.Document } } =>
28
+ !!data.subject &&
29
+ typeof data.subject === 'object' &&
30
+ 'type' in data.subject &&
31
+ 'object' in data.subject &&
32
+ data.subject.type === meta.id &&
33
+ Obj.instanceOf(Markdown.Document, data.subject.object),
34
+ component: ({ data }) => <DocumentPresenterContainer document={data.subject.object} />,
35
+ }),
36
+ Surface.create({
37
+ id: 'collection',
38
+ role: 'article',
39
+ position: 'hoist',
40
+ filter: (data): data is { subject: { type: typeof meta.id; object: Collection.Collection } } =>
41
+ !!data.subject &&
42
+ typeof data.subject === 'object' &&
43
+ 'type' in data.subject &&
44
+ 'object' in data.subject &&
45
+ data.subject.type === meta.id &&
46
+ Obj.instanceOf(Collection.Collection, data.subject.object),
47
+ component: ({ role, data }) => <CollectionPresenterContainer role={role} subject={data.subject.object} />,
48
+ }),
49
+ Surface.create({
50
+ id: 'slide',
51
+ role: 'slide',
52
+ filter: AppSurface.objectSection(Markdown.Document),
53
+ component: ({ data }) => <MarkdownSlide document={data.subject} />,
54
+ }),
55
+ Surface.create({
56
+ id: 'plugin-settings',
57
+ role: 'article',
58
+ filter: AppSurface.settingsArticle(meta.id),
59
+ component: ({ data: { subject } }) => {
60
+ const { settings, updateSettings } = useSettingsState<Settings.Settings>(subject.atom);
61
+ return <PresenterSettings settings={settings} onSettingsChange={updateSettings} />;
62
+ },
63
+ }),
64
+ ]),
65
+ ),
66
+ );
@@ -2,18 +2,30 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import { Capabilities, contributes } from '@dxos/app-framework';
6
- import { live } from '@dxos/live-object';
5
+ import * as Effect from 'effect/Effect';
7
6
 
8
- import { meta } from '../meta';
9
- import { type PresenterSettingsProps, PresenterSettingsSchema } from '../types';
7
+ import { Capability } from '@dxos/app-framework';
8
+ import { AppCapabilities } from '@dxos/app-toolkit';
9
+ import { createKvsStore } from '@dxos/effect';
10
10
 
11
- export default () => {
12
- const settings = live<PresenterSettingsProps>({});
11
+ import { meta } from '#meta';
12
+ import { PresenterCapabilities, Settings } from '#types';
13
13
 
14
- return contributes(Capabilities.Settings, {
15
- prefix: meta.id,
16
- schema: PresenterSettingsSchema,
17
- value: settings,
18
- });
19
- };
14
+ export default Capability.makeModule(() =>
15
+ Effect.sync(() => {
16
+ const settingsAtom = createKvsStore({
17
+ key: meta.id,
18
+ schema: Settings.Settings,
19
+ defaultValue: () => ({}),
20
+ });
21
+
22
+ return [
23
+ Capability.contributes(PresenterCapabilities.Settings, settingsAtom),
24
+ Capability.contributes(AppCapabilities.Settings, {
25
+ prefix: meta.id,
26
+ schema: Settings.Settings,
27
+ atom: settingsAtom,
28
+ }),
29
+ ];
30
+ }),
31
+ );
@@ -5,25 +5,25 @@
5
5
  import { type Meta, type StoryObj } from '@storybook/react-vite';
6
6
  import React from 'react';
7
7
 
8
- import { withTheme } from '@dxos/react-ui/testing';
8
+ import { withLayout, withTheme } from '@dxos/react-ui/testing';
9
9
 
10
- import { createSlide } from '../../testing';
10
+ import { createSlide } from '#testing';
11
11
 
12
- import { Container } from './Container';
12
+ import { Panel } from './Panel';
13
13
  import { Slide, type SlideProps } from './Slide';
14
14
 
15
15
  const DefaultStory = ({ content = '' }: SlideProps) => {
16
16
  return (
17
- <Container classNames='bg-neutral-200'>
17
+ <Panel classNames='bg-neutral-200'>
18
18
  <Slide content={content} />
19
- </Container>
19
+ </Panel>
20
20
  );
21
21
  };
22
22
 
23
23
  const meta = {
24
- title: 'plugins/plugin-presenter/Container',
24
+ title: 'plugins/plugin-presenter/components/Panel',
25
25
  render: DefaultStory,
26
- decorators: [withTheme],
26
+ decorators: [withTheme(), withLayout({ layout: 'fullscreen' })],
27
27
  parameters: {
28
28
  layout: 'fullscreen',
29
29
  },
@@ -6,14 +6,14 @@ import React, { type PropsWithChildren, useState } from 'react';
6
6
  import { useResizeDetector } from 'react-resize-detector';
7
7
 
8
8
  import { type ThemedClassName } from '@dxos/react-ui';
9
- import { mx } from '@dxos/react-ui-theme';
9
+ import { mx } from '@dxos/ui-theme';
10
10
 
11
- export type ContainerProps = ThemedClassName<PropsWithChildren<{}>>;
11
+ export type PanelProps = ThemedClassName<PropsWithChildren<{}>>;
12
12
 
13
13
  /**
14
- * Scaled markdown container.
14
+ * Scaled markdown panel.
15
15
  */
16
- export const Container = ({ children, classNames }: ContainerProps) => {
16
+ export const Panel = ({ children, classNames }: PanelProps) => {
17
17
  const [props, setProps] = useState({});
18
18
  const {
19
19
  ref: containerRef,
@@ -35,8 +35,8 @@ export const Container = ({ children, classNames }: ContainerProps) => {
35
35
  // TODO(burdon): Reconcile highlight colors with markdown editor.
36
36
  // https://www.npmjs.com/package/react-markdown
37
37
  return (
38
- <div ref={containerRef} className={mx('flex grow relative overflow-hidden bg-attention', classNames)}>
39
- <div className={mx('flex is-full bs-full overflow-hidden absolute')} style={props}>
38
+ <div ref={containerRef} className={mx('flex grow relative overflow-hidden bg-attention-surface', classNames)}>
39
+ <div className={mx('dx-container absolute')} style={props}>
40
40
  {width && height && children}
41
41
  </div>
42
42
  </div>