@dreamboard-games/ui-sdk 0.0.41
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.
- package/LICENSE +89 -0
- package/NOTICE +1 -0
- package/README.md +154 -0
- package/dist/components/ActionButton.d.ts +13 -0
- package/dist/components/ActionButton.d.ts.map +1 -0
- package/dist/components/ActionButton.js +14 -0
- package/dist/components/ActionPanel.d.ts +33 -0
- package/dist/components/ActionPanel.d.ts.map +1 -0
- package/dist/components/ActionPanel.js +148 -0
- package/dist/components/Card.d.ts +29 -0
- package/dist/components/Card.d.ts.map +1 -0
- package/dist/components/Card.js +220 -0
- package/dist/components/ChromeSuppressionContext.d.ts +7 -0
- package/dist/components/ChromeSuppressionContext.d.ts.map +1 -0
- package/dist/components/ChromeSuppressionContext.js +34 -0
- package/dist/components/CostDisplay.d.ts +22 -0
- package/dist/components/CostDisplay.d.ts.map +1 -0
- package/dist/components/CostDisplay.js +41 -0
- package/dist/components/DiceRoller.d.ts +30 -0
- package/dist/components/DiceRoller.d.ts.map +1 -0
- package/dist/components/DiceRoller.js +319 -0
- package/dist/components/Drawer.d.ts +19 -0
- package/dist/components/Drawer.d.ts.map +1 -0
- package/dist/components/Drawer.js +55 -0
- package/dist/components/ErrorBoundary.d.ts +24 -0
- package/dist/components/ErrorBoundary.d.ts.map +1 -0
- package/dist/components/ErrorBoundary.js +37 -0
- package/dist/components/GameEndDisplay.d.ts +27 -0
- package/dist/components/GameEndDisplay.d.ts.map +1 -0
- package/dist/components/GameEndDisplay.js +185 -0
- package/dist/components/GameSkeleton.d.ts +12 -0
- package/dist/components/GameSkeleton.d.ts.map +1 -0
- package/dist/components/GameSkeleton.js +54 -0
- package/dist/components/Hand.d.ts +99 -0
- package/dist/components/Hand.d.ts.map +1 -0
- package/dist/components/Hand.js +162 -0
- package/dist/components/HandDock.d.ts +35 -0
- package/dist/components/HandDock.d.ts.map +1 -0
- package/dist/components/HandDock.js +124 -0
- package/dist/components/InteractionForm.d.ts +50 -0
- package/dist/components/InteractionForm.d.ts.map +1 -0
- package/dist/components/InteractionForm.js +402 -0
- package/dist/components/MoreActions.d.ts +49 -0
- package/dist/components/MoreActions.d.ts.map +1 -0
- package/dist/components/MoreActions.js +64 -0
- package/dist/components/PhaseIndicator.d.ts +35 -0
- package/dist/components/PhaseIndicator.d.ts.map +1 -0
- package/dist/components/PhaseIndicator.js +212 -0
- package/dist/components/PlayArea.d.ts +28 -0
- package/dist/components/PlayArea.d.ts.map +1 -0
- package/dist/components/PlayArea.js +48 -0
- package/dist/components/PluginRuntime.d.ts +37 -0
- package/dist/components/PluginRuntime.d.ts.map +1 -0
- package/dist/components/PluginRuntime.js +47 -0
- package/dist/components/PrimaryActionButton.d.ts +98 -0
- package/dist/components/PrimaryActionButton.d.ts.map +1 -0
- package/dist/components/PrimaryActionButton.js +183 -0
- package/dist/components/PrimaryButton.d.ts +20 -0
- package/dist/components/PrimaryButton.d.ts.map +1 -0
- package/dist/components/PrimaryButton.js +5 -0
- package/dist/components/PromptDialogHost.d.ts +15 -0
- package/dist/components/PromptDialogHost.d.ts.map +1 -0
- package/dist/components/PromptDialogHost.js +22 -0
- package/dist/components/ResourceCounter.d.ts +38 -0
- package/dist/components/ResourceCounter.d.ts.map +1 -0
- package/dist/components/ResourceCounter.js +118 -0
- package/dist/components/ThemedButton.d.ts +12 -0
- package/dist/components/ThemedButton.d.ts.map +1 -0
- package/dist/components/ThemedButton.js +38 -0
- package/dist/components/Toast.d.ts +35 -0
- package/dist/components/Toast.d.ts.map +1 -0
- package/dist/components/Toast.js +116 -0
- package/dist/components/board/HexGrid.d.ts +344 -0
- package/dist/components/board/HexGrid.d.ts.map +1 -0
- package/dist/components/board/HexGrid.js +340 -0
- package/dist/components/board/NetworkGraph.d.ts +100 -0
- package/dist/components/board/NetworkGraph.d.ts.map +1 -0
- package/dist/components/board/NetworkGraph.js +123 -0
- package/dist/components/board/SlotSystem.d.ts +71 -0
- package/dist/components/board/SlotSystem.d.ts.map +1 -0
- package/dist/components/board/SlotSystem.js +87 -0
- package/dist/components/board/SquareGrid.d.ts +188 -0
- package/dist/components/board/SquareGrid.d.ts.map +1 -0
- package/dist/components/board/SquareGrid.js +328 -0
- package/dist/components/board/TrackBoard.d.ts +113 -0
- package/dist/components/board/TrackBoard.d.ts.map +1 -0
- package/dist/components/board/TrackBoard.js +135 -0
- package/dist/components/board/ZoneMap.d.ts +88 -0
- package/dist/components/board/ZoneMap.d.ts.map +1 -0
- package/dist/components/board/ZoneMap.js +133 -0
- package/dist/components/board/hex-board-view.d.ts +69 -0
- package/dist/components/board/hex-board-view.d.ts.map +1 -0
- package/dist/components/board/hex-board-view.js +60 -0
- package/dist/components/board/index.d.ts +23 -0
- package/dist/components/board/index.d.ts.map +1 -0
- package/dist/components/board/index.js +40 -0
- package/dist/components/board/interaction-accessibility.d.ts +5 -0
- package/dist/components/board/interaction-accessibility.d.ts.map +1 -0
- package/dist/components/board/interaction-accessibility.js +13 -0
- package/dist/components/board/target-layer.d.ts +13 -0
- package/dist/components/board/target-layer.d.ts.map +1 -0
- package/dist/components/board/target-layer.js +10 -0
- package/dist/components/card-render-content.type-test.d.ts +2 -0
- package/dist/components/card-render-content.type-test.d.ts.map +1 -0
- package/dist/components/card-render-content.type-test.js +1 -0
- package/dist/components/index.d.ts +34 -0
- package/dist/components/index.d.ts.map +1 -0
- package/dist/components/index.js +35 -0
- package/dist/components/interaction-dialog-behavior.d.ts +15 -0
- package/dist/components/interaction-dialog-behavior.d.ts.map +1 -0
- package/dist/components/interaction-dialog-behavior.js +9 -0
- package/dist/components/surfaces/BlockerSurface.d.ts +27 -0
- package/dist/components/surfaces/BlockerSurface.d.ts.map +1 -0
- package/dist/components/surfaces/BlockerSurface.js +38 -0
- package/dist/components/surfaces/BoardSurface.d.ts +77 -0
- package/dist/components/surfaces/BoardSurface.d.ts.map +1 -0
- package/dist/components/surfaces/BoardSurface.js +180 -0
- package/dist/components/surfaces/ChromeSurface.d.ts +29 -0
- package/dist/components/surfaces/ChromeSurface.d.ts.map +1 -0
- package/dist/components/surfaces/ChromeSurface.js +34 -0
- package/dist/components/surfaces/ExhaustivenessAudit.d.ts +32 -0
- package/dist/components/surfaces/ExhaustivenessAudit.d.ts.map +1 -0
- package/dist/components/surfaces/ExhaustivenessAudit.js +65 -0
- package/dist/components/surfaces/InboxSurface.d.ts +40 -0
- package/dist/components/surfaces/InboxSurface.d.ts.map +1 -0
- package/dist/components/surfaces/InboxSurface.js +99 -0
- package/dist/components/surfaces/MarketSurface.d.ts +62 -0
- package/dist/components/surfaces/MarketSurface.d.ts.map +1 -0
- package/dist/components/surfaces/MarketSurface.js +242 -0
- package/dist/components/surfaces/PanelSurface.d.ts +111 -0
- package/dist/components/surfaces/PanelSurface.d.ts.map +1 -0
- package/dist/components/surfaces/PanelSurface.js +180 -0
- package/dist/components/surfaces/PlayerCardsSurface.d.ts +104 -0
- package/dist/components/surfaces/PlayerCardsSurface.d.ts.map +1 -0
- package/dist/components/surfaces/PlayerCardsSurface.js +178 -0
- package/dist/components/surfaces/internal/CardZoneFollowUpForm.d.ts +7 -0
- package/dist/components/surfaces/internal/CardZoneFollowUpForm.d.ts.map +1 -0
- package/dist/components/surfaces/internal/CardZoneFollowUpForm.js +9 -0
- package/dist/components/surfaces/internal/DefaultInteractionButton.d.ts +71 -0
- package/dist/components/surfaces/internal/DefaultInteractionButton.d.ts.map +1 -0
- package/dist/components/surfaces/internal/DefaultInteractionButton.js +82 -0
- package/dist/components/surfaces/internal/useCardZoneInteractions.d.ts +21 -0
- package/dist/components/surfaces/internal/useCardZoneInteractions.d.ts.map +1 -0
- package/dist/components/surfaces/internal/useCardZoneInteractions.js +202 -0
- package/dist/components/surfaces/types.d.ts +59 -0
- package/dist/components/surfaces/types.d.ts.map +1 -0
- package/dist/components/surfaces/types.js +1 -0
- package/dist/context/ClientParamSchemaContext.d.ts +21 -0
- package/dist/context/ClientParamSchemaContext.d.ts.map +1 -0
- package/dist/context/ClientParamSchemaContext.js +12 -0
- package/dist/context/InteractionDraftContext.d.ts +69 -0
- package/dist/context/InteractionDraftContext.d.ts.map +1 -0
- package/dist/context/InteractionDraftContext.js +145 -0
- package/dist/context/PluginSessionContext.d.ts +33 -0
- package/dist/context/PluginSessionContext.d.ts.map +1 -0
- package/dist/context/PluginSessionContext.js +38 -0
- package/dist/context/PluginStateContext.d.ts +116 -0
- package/dist/context/PluginStateContext.d.ts.map +1 -0
- package/dist/context/PluginStateContext.js +186 -0
- package/dist/context/RuntimeContext.d.ts +49 -0
- package/dist/context/RuntimeContext.d.ts.map +1 -0
- package/dist/context/RuntimeContext.js +67 -0
- package/dist/defaults/components.d.ts +52 -0
- package/dist/defaults/components.d.ts.map +1 -0
- package/dist/defaults/components.js +159 -0
- package/dist/defaults/index.d.ts +2 -0
- package/dist/defaults/index.d.ts.map +1 -0
- package/dist/defaults/index.js +1 -0
- package/dist/errors/ValidationError.d.ts +10 -0
- package/dist/errors/ValidationError.d.ts.map +1 -0
- package/dist/errors/ValidationError.js +23 -0
- package/dist/helpers/cards.d.ts +3 -0
- package/dist/helpers/cards.d.ts.map +1 -0
- package/dist/helpers/cards.js +11 -0
- package/dist/helpers/track-board.d.ts +79 -0
- package/dist/helpers/track-board.d.ts.map +1 -0
- package/dist/helpers/track-board.js +56 -0
- package/dist/hooks/useActivePlayers.d.ts +16 -0
- package/dist/hooks/useActivePlayers.d.ts.map +1 -0
- package/dist/hooks/useActivePlayers.js +17 -0
- package/dist/hooks/useBoardInteractions.d.ts +110 -0
- package/dist/hooks/useBoardInteractions.d.ts.map +1 -0
- package/dist/hooks/useBoardInteractions.js +248 -0
- package/dist/hooks/useBoardTopology.d.ts +23 -0
- package/dist/hooks/useBoardTopology.d.ts.map +1 -0
- package/dist/hooks/useBoardTopology.js +128 -0
- package/dist/hooks/useCards.d.ts +3 -0
- package/dist/hooks/useCards.d.ts.map +1 -0
- package/dist/hooks/useCards.js +5 -0
- package/dist/hooks/useGameSelector.d.ts +13 -0
- package/dist/hooks/useGameSelector.d.ts.map +1 -0
- package/dist/hooks/useGameSelector.js +67 -0
- package/dist/hooks/useGameView.d.ts +6 -0
- package/dist/hooks/useGameView.d.ts.map +1 -0
- package/dist/hooks/useGameView.js +7 -0
- package/dist/hooks/useHandLayout.d.ts +120 -0
- package/dist/hooks/useHandLayout.d.ts.map +1 -0
- package/dist/hooks/useHandLayout.js +235 -0
- package/dist/hooks/useHexBoard.d.ts +19 -0
- package/dist/hooks/useHexBoard.d.ts.map +1 -0
- package/dist/hooks/useHexBoard.js +28 -0
- package/dist/hooks/useHexGrid.d.ts +56 -0
- package/dist/hooks/useHexGrid.d.ts.map +1 -0
- package/dist/hooks/useHexGrid.js +112 -0
- package/dist/hooks/useInteractionByKey.d.ts +29 -0
- package/dist/hooks/useInteractionByKey.d.ts.map +1 -0
- package/dist/hooks/useInteractionByKey.js +263 -0
- package/dist/hooks/useInteractionHandle.d.ts +103 -0
- package/dist/hooks/useInteractionHandle.d.ts.map +1 -0
- package/dist/hooks/useInteractionHandle.js +254 -0
- package/dist/hooks/useIsMobile.d.ts +7 -0
- package/dist/hooks/useIsMobile.d.ts.map +1 -0
- package/dist/hooks/useIsMobile.js +29 -0
- package/dist/hooks/useIsMyTurn.d.ts +6 -0
- package/dist/hooks/useIsMyTurn.d.ts.map +1 -0
- package/dist/hooks/useIsMyTurn.js +11 -0
- package/dist/hooks/useLobby.d.ts +28 -0
- package/dist/hooks/useLobby.d.ts.map +1 -0
- package/dist/hooks/useLobby.js +60 -0
- package/dist/hooks/useMe.d.ts +11 -0
- package/dist/hooks/useMe.d.ts.map +1 -0
- package/dist/hooks/useMe.js +32 -0
- package/dist/hooks/usePanZoom.d.ts +113 -0
- package/dist/hooks/usePanZoom.d.ts.map +1 -0
- package/dist/hooks/usePanZoom.js +165 -0
- package/dist/hooks/usePlayerInfo.d.ts +4 -0
- package/dist/hooks/usePlayerInfo.d.ts.map +1 -0
- package/dist/hooks/usePlayerInfo.js +21 -0
- package/dist/hooks/usePlayerTurnOrder.d.ts +15 -0
- package/dist/hooks/usePlayerTurnOrder.d.ts.map +1 -0
- package/dist/hooks/usePlayerTurnOrder.js +22 -0
- package/dist/hooks/usePluginRuntime.d.ts +45 -0
- package/dist/hooks/usePluginRuntime.d.ts.map +1 -0
- package/dist/hooks/usePluginRuntime.js +92 -0
- package/dist/hooks/useSeatInbox.d.ts +22 -0
- package/dist/hooks/useSeatInbox.d.ts.map +1 -0
- package/dist/hooks/useSeatInbox.js +43 -0
- package/dist/hooks/useSimultaneousPhase.d.ts +7 -0
- package/dist/hooks/useSimultaneousPhase.d.ts.map +1 -0
- package/dist/hooks/useSimultaneousPhase.js +8 -0
- package/dist/hooks/useSquareBoard.d.ts +21 -0
- package/dist/hooks/useSquareBoard.d.ts.map +1 -0
- package/dist/hooks/useSquareBoard.js +67 -0
- package/dist/hooks/useSquareGrid.d.ts +96 -0
- package/dist/hooks/useSquareGrid.d.ts.map +1 -0
- package/dist/hooks/useSquareGrid.js +152 -0
- package/dist/index.d.ts +30 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +20 -0
- package/dist/internal/ui/alert.d.ts +8 -0
- package/dist/internal/ui/alert.d.ts.map +1 -0
- package/dist/internal/ui/alert.js +11 -0
- package/dist/internal/ui/button.d.ts +10 -0
- package/dist/internal/ui/button.d.ts.map +1 -0
- package/dist/internal/ui/button.js +21 -0
- package/dist/internal/ui/dialog.d.ts +16 -0
- package/dist/internal/ui/dialog.d.ts.map +1 -0
- package/dist/internal/ui/dialog.js +35 -0
- package/dist/internal/ui/input.d.ts +3 -0
- package/dist/internal/ui/input.d.ts.map +1 -0
- package/dist/internal/ui/input.js +5 -0
- package/dist/internal/ui/label.d.ts +4 -0
- package/dist/internal/ui/label.d.ts.map +1 -0
- package/dist/internal/ui/label.js +7 -0
- package/dist/internal/ui/select.d.ts +9 -0
- package/dist/internal/ui/select.d.ts.map +1 -0
- package/dist/internal/ui/select.js +23 -0
- package/dist/internal/ui/tooltip.d.ts +7 -0
- package/dist/internal/ui/tooltip.d.ts.map +1 -0
- package/dist/internal/ui/tooltip.js +16 -0
- package/dist/internal/ui/utils.d.ts +3 -0
- package/dist/internal/ui/utils.d.ts.map +1 -0
- package/dist/internal/ui/utils.js +4 -0
- package/dist/internal.d.ts +7 -0
- package/dist/internal.d.ts.map +1 -0
- package/dist/internal.js +4 -0
- package/dist/plugin-styles.css +246 -0
- package/dist/primitives/board.d.ts +29 -0
- package/dist/primitives/board.d.ts.map +1 -0
- package/dist/primitives/board.js +163 -0
- package/dist/primitives/game-ui-provider.d.ts +12 -0
- package/dist/primitives/game-ui-provider.d.ts.map +1 -0
- package/dist/primitives/game-ui-provider.js +7 -0
- package/dist/primitives/index.d.ts +8 -0
- package/dist/primitives/index.d.ts.map +1 -0
- package/dist/primitives/index.js +7 -0
- package/dist/primitives/interaction.d.ts +52 -0
- package/dist/primitives/interaction.d.ts.map +1 -0
- package/dist/primitives/interaction.js +250 -0
- package/dist/primitives/phase.d.ts +15 -0
- package/dist/primitives/phase.d.ts.map +1 -0
- package/dist/primitives/phase.js +18 -0
- package/dist/primitives/player-roster.d.ts +64 -0
- package/dist/primitives/player-roster.d.ts.map +1 -0
- package/dist/primitives/player-roster.js +149 -0
- package/dist/primitives/primitive-props.d.ts +15 -0
- package/dist/primitives/primitive-props.d.ts.map +1 -0
- package/dist/primitives/primitive-props.js +39 -0
- package/dist/primitives/prompt.d.ts +44 -0
- package/dist/primitives/prompt.d.ts.map +1 -0
- package/dist/primitives/prompt.js +101 -0
- package/dist/primitives/zone.d.ts +31 -0
- package/dist/primitives/zone.d.ts.map +1 -0
- package/dist/primitives/zone.js +58 -0
- package/dist/reducer.d.ts +21 -0
- package/dist/reducer.d.ts.map +1 -0
- package/dist/reducer.js +14 -0
- package/dist/runtime/createPluginRuntimeAPI.d.ts +67 -0
- package/dist/runtime/createPluginRuntimeAPI.d.ts.map +1 -0
- package/dist/runtime/createPluginRuntimeAPI.js +419 -0
- package/dist/theme/ThemeProvider.d.ts +98 -0
- package/dist/theme/ThemeProvider.d.ts.map +1 -0
- package/dist/theme/ThemeProvider.js +148 -0
- package/dist/theme/board.d.ts +42 -0
- package/dist/theme/board.d.ts.map +1 -0
- package/dist/theme/board.js +34 -0
- package/dist/theme/css-vars.d.ts +31 -0
- package/dist/theme/css-vars.d.ts.map +1 -0
- package/dist/theme/css-vars.js +88 -0
- package/dist/theme/derive.d.ts +66 -0
- package/dist/theme/derive.d.ts.map +1 -0
- package/dist/theme/derive.js +161 -0
- package/dist/theme/index.d.ts +22 -0
- package/dist/theme/index.d.ts.map +1 -0
- package/dist/theme/index.js +20 -0
- package/dist/theme/presets/arcade.d.ts +10 -0
- package/dist/theme/presets/arcade.d.ts.map +1 -0
- package/dist/theme/presets/arcade.js +257 -0
- package/dist/theme/presets/studio.d.ts +10 -0
- package/dist/theme/presets/studio.d.ts.map +1 -0
- package/dist/theme/presets/studio.js +257 -0
- package/dist/theme/presets/tabletop.d.ts +15 -0
- package/dist/theme/presets/tabletop.d.ts.map +1 -0
- package/dist/theme/presets/tabletop.js +262 -0
- package/dist/theme/tokens.d.ts +345 -0
- package/dist/theme/tokens.d.ts.map +1 -0
- package/dist/theme/tokens.js +57 -0
- package/dist/types/player-state.d.ts +337 -0
- package/dist/types/player-state.d.ts.map +1 -0
- package/dist/types/player-state.js +1 -0
- package/dist/types/plugin-state.d.ts +324 -0
- package/dist/types/plugin-state.d.ts.map +1 -0
- package/dist/types/plugin-state.js +1 -0
- package/dist/types/reducer-state.d.ts +10 -0
- package/dist/types/reducer-state.d.ts.map +1 -0
- package/dist/types/reducer-state.js +1 -0
- package/dist/types/runtime-api.d.ts +99 -0
- package/dist/types/runtime-api.d.ts.map +1 -0
- package/dist/types/runtime-api.js +1 -0
- package/dist/types/tiled-board.d.ts +187 -0
- package/dist/types/tiled-board.d.ts.map +1 -0
- package/dist/types/tiled-board.js +226 -0
- package/dist/ui-contract.d.ts +78 -0
- package/dist/ui-contract.d.ts.map +1 -0
- package/dist/ui-contract.js +15 -0
- package/dist/ui-sdk.d.ts +3409 -0
- package/dist/utils/interaction-inputs.d.ts +22 -0
- package/dist/utils/interaction-inputs.d.ts.map +1 -0
- package/dist/utils/interaction-inputs.js +219 -0
- package/dist/utils/interaction-labels.d.ts +4 -0
- package/dist/utils/interaction-labels.d.ts.map +1 -0
- package/dist/utils/interaction-labels.js +18 -0
- package/dist/utils/interaction-status.d.ts +15 -0
- package/dist/utils/interaction-status.d.ts.map +1 -0
- package/dist/utils/interaction-status.js +31 -0
- package/package.json +101 -0
- package/src/components/ActionButton.tsx +48 -0
- package/src/components/ActionPanel.tsx +310 -0
- package/src/components/Card.tsx +385 -0
- package/src/components/ChromeSuppressionContext.tsx +70 -0
- package/src/components/CostDisplay.test.tsx +23 -0
- package/src/components/CostDisplay.tsx +145 -0
- package/src/components/DiceRoller.tsx +601 -0
- package/src/components/Drawer.tsx +179 -0
- package/src/components/ErrorBoundary.tsx +119 -0
- package/src/components/GameEndDisplay.test.tsx +19 -0
- package/src/components/GameEndDisplay.tsx +398 -0
- package/src/components/GameSkeleton.tsx +260 -0
- package/src/components/Hand.tsx +387 -0
- package/src/components/HandDock.tsx +257 -0
- package/src/components/InteractionForm.test.tsx +303 -0
- package/src/components/InteractionForm.tsx +1029 -0
- package/src/components/MoreActions.test.tsx +93 -0
- package/src/components/MoreActions.tsx +143 -0
- package/src/components/PhaseIndicator.tsx +341 -0
- package/src/components/PlayArea.tsx +125 -0
- package/src/components/PluginRuntime.tsx +92 -0
- package/src/components/PrimaryActionButton.test.tsx +138 -0
- package/src/components/PrimaryActionButton.tsx +351 -0
- package/src/components/PrimaryButton.tsx +44 -0
- package/src/components/PromptDialogHost.tsx +92 -0
- package/src/components/ResourceCounter.test.tsx +29 -0
- package/src/components/ResourceCounter.tsx +275 -0
- package/src/components/ThemedButton.tsx +78 -0
- package/src/components/Toast.tsx +251 -0
- package/src/components/__fixtures__/ActionButton.fixture.tsx +234 -0
- package/src/components/__fixtures__/ActionPanel.fixture.tsx +298 -0
- package/src/components/__fixtures__/Card.fixture.tsx +185 -0
- package/src/components/__fixtures__/CostDisplay.fixture.tsx +156 -0
- package/src/components/__fixtures__/DiceRoller.fixture.tsx +435 -0
- package/src/components/__fixtures__/Drawer.fixture.tsx +113 -0
- package/src/components/__fixtures__/ErrorBoundary.fixture.tsx +82 -0
- package/src/components/__fixtures__/GameEndDisplay.fixture.tsx +188 -0
- package/src/components/__fixtures__/GameSkeleton.fixture.tsx +46 -0
- package/src/components/__fixtures__/Hand.fixture.tsx +522 -0
- package/src/components/__fixtures__/HexGrid.fixture.tsx +1181 -0
- package/src/components/__fixtures__/NetworkGraph.fixture.tsx +599 -0
- package/src/components/__fixtures__/PhaseIndicator.fixture.tsx +181 -0
- package/src/components/__fixtures__/PlayArea.fixture.tsx +221 -0
- package/src/components/__fixtures__/ResourceCounter.fixture.tsx +227 -0
- package/src/components/__fixtures__/SlotSystem.fixture.tsx +824 -0
- package/src/components/__fixtures__/SquareGrid.fixture.tsx +764 -0
- package/src/components/__fixtures__/Toast.fixture.tsx +97 -0
- package/src/components/__fixtures__/TrackBoard.fixture.tsx +685 -0
- package/src/components/__fixtures__/ZoneMap.fixture.tsx +754 -0
- package/src/components/board/HexGrid.tsx +1294 -0
- package/src/components/board/NetworkGraph.tsx +476 -0
- package/src/components/board/SlotSystem.tsx +339 -0
- package/src/components/board/SquareGrid.tsx +1165 -0
- package/src/components/board/TrackBoard.tsx +496 -0
- package/src/components/board/ZoneMap.tsx +448 -0
- package/src/components/board/hex-board-view.test.tsx +114 -0
- package/src/components/board/hex-board-view.ts +123 -0
- package/src/components/board/index.ts +142 -0
- package/src/components/board/interaction-accessibility.ts +21 -0
- package/src/components/board/target-layer-grids.test.tsx +420 -0
- package/src/components/board/target-layer.ts +30 -0
- package/src/components/card-render-content.type-test.ts +27 -0
- package/src/components/index.ts +208 -0
- package/src/components/interaction-dialog-behavior.test.ts +23 -0
- package/src/components/interaction-dialog-behavior.ts +22 -0
- package/src/components/surfaces/BlockerSurface.test.tsx +158 -0
- package/src/components/surfaces/BlockerSurface.tsx +127 -0
- package/src/components/surfaces/BoardSurface.tsx +340 -0
- package/src/components/surfaces/ChromeSurface.tsx +123 -0
- package/src/components/surfaces/ExhaustivenessAudit.tsx +91 -0
- package/src/components/surfaces/InboxSurface.test.tsx +149 -0
- package/src/components/surfaces/InboxSurface.tsx +245 -0
- package/src/components/surfaces/MarketSurface.tsx +544 -0
- package/src/components/surfaces/PanelSurface.test.tsx +496 -0
- package/src/components/surfaces/PanelSurface.tsx +458 -0
- package/src/components/surfaces/PlayerCardsSurface.tsx +525 -0
- package/src/components/surfaces/internal/CardZoneFollowUpForm.tsx +35 -0
- package/src/components/surfaces/internal/DefaultInteractionButton.tsx +219 -0
- package/src/components/surfaces/internal/useCardZoneInteractions.ts +311 -0
- package/src/components/surfaces/types.ts +100 -0
- package/src/context/ClientParamSchemaContext.tsx +44 -0
- package/src/context/InteractionDraftContext.tsx +204 -0
- package/src/context/PluginSessionContext.tsx +47 -0
- package/src/context/PluginStateContext.tsx +254 -0
- package/src/context/RuntimeContext.tsx +96 -0
- package/src/defaults/components.tsx +442 -0
- package/src/defaults/defaults.test.tsx +230 -0
- package/src/defaults/index.ts +1 -0
- package/src/errors/ValidationError.ts +29 -0
- package/src/helpers/cards.ts +19 -0
- package/src/helpers/track-board.ts +211 -0
- package/src/hooks/useActivePlayers.ts +19 -0
- package/src/hooks/useBoardInteractions.test.tsx +622 -0
- package/src/hooks/useBoardInteractions.ts +434 -0
- package/src/hooks/useBoardTopology.ts +316 -0
- package/src/hooks/useCards.test.tsx +129 -0
- package/src/hooks/useCards.ts +10 -0
- package/src/hooks/useGameSelector.ts +105 -0
- package/src/hooks/useGameView.ts +9 -0
- package/src/hooks/useHandLayout.ts +349 -0
- package/src/hooks/useHexBoard.ts +74 -0
- package/src/hooks/useHexGrid.ts +185 -0
- package/src/hooks/useInteractionByKey.ts +349 -0
- package/src/hooks/useInteractionHandle.ts +437 -0
- package/src/hooks/useIsMobile.ts +35 -0
- package/src/hooks/useIsMyTurn.test.tsx +99 -0
- package/src/hooks/useIsMyTurn.ts +15 -0
- package/src/hooks/useLobby.ts +76 -0
- package/src/hooks/useMe.ts +48 -0
- package/src/hooks/usePanZoom.ts +278 -0
- package/src/hooks/usePlayerInfo.ts +28 -0
- package/src/hooks/usePlayerTurnOrder.ts +23 -0
- package/src/hooks/usePluginRuntime.test.tsx +102 -0
- package/src/hooks/usePluginRuntime.ts +130 -0
- package/src/hooks/useSeatInbox.ts +61 -0
- package/src/hooks/useSimultaneousPhase.ts +10 -0
- package/src/hooks/useSquareBoard.ts +124 -0
- package/src/hooks/useSquareGrid.ts +328 -0
- package/src/index.test.ts +474 -0
- package/src/index.ts +148 -0
- package/src/internal/ui/alert.tsx +51 -0
- package/src/internal/ui/button.tsx +58 -0
- package/src/internal/ui/dialog.tsx +134 -0
- package/src/internal/ui/input.tsx +21 -0
- package/src/internal/ui/label.tsx +21 -0
- package/src/internal/ui/select.tsx +129 -0
- package/src/internal/ui/tooltip.tsx +54 -0
- package/src/internal/ui/utils.ts +5 -0
- package/src/internal.ts +18 -0
- package/src/plugin-styles.css +246 -0
- package/src/primitives/board.test.tsx +139 -0
- package/src/primitives/board.tsx +267 -0
- package/src/primitives/game-ui-provider.tsx +35 -0
- package/src/primitives/index.ts +83 -0
- package/src/primitives/interaction.test.tsx +420 -0
- package/src/primitives/interaction.tsx +405 -0
- package/src/primitives/phase.test.tsx +82 -0
- package/src/primitives/phase.tsx +43 -0
- package/src/primitives/player-roster.test.tsx +168 -0
- package/src/primitives/player-roster.tsx +301 -0
- package/src/primitives/primitive-props.tsx +82 -0
- package/src/primitives/prompt.test.tsx +159 -0
- package/src/primitives/prompt.tsx +203 -0
- package/src/primitives/zone.tsx +113 -0
- package/src/reducer.ts +42 -0
- package/src/runtime/createPluginRuntimeAPI.ts +605 -0
- package/src/theme/ThemeProvider.test.tsx +36 -0
- package/src/theme/ThemeProvider.tsx +252 -0
- package/src/theme/board.ts +61 -0
- package/src/theme/css-vars.ts +105 -0
- package/src/theme/derive.ts +240 -0
- package/src/theme/index.ts +61 -0
- package/src/theme/presets/arcade.ts +261 -0
- package/src/theme/presets/studio.ts +261 -0
- package/src/theme/presets/tabletop.ts +266 -0
- package/src/theme/theme.test.ts +258 -0
- package/src/theme/tokens.ts +392 -0
- package/src/types/player-state.ts +445 -0
- package/src/types/plugin-state.ts +407 -0
- package/src/types/reducer-state.ts +24 -0
- package/src/types/runtime-api.ts +114 -0
- package/src/types/tiled-board.ts +785 -0
- package/src/ui-contract.ts +168 -0
- package/src/utils/interaction-inputs.test.ts +109 -0
- package/src/utils/interaction-inputs.ts +331 -0
- package/src/utils/interaction-labels.ts +23 -0
- package/src/utils/interaction-status.ts +59 -0
|
@@ -0,0 +1,419 @@
|
|
|
1
|
+
import { z } from "zod";
|
|
2
|
+
/**
|
|
3
|
+
* Message schemas from main app to plugin
|
|
4
|
+
* We define them here to avoid circular dependencies with apps/web
|
|
5
|
+
*/
|
|
6
|
+
// Main → Plugin: Initialize plugin with session info
|
|
7
|
+
const InitMessageSchema = z.object({
|
|
8
|
+
type: z.literal("init"),
|
|
9
|
+
sessionId: z.string(),
|
|
10
|
+
controllablePlayerIds: z.array(z.string()),
|
|
11
|
+
controllingPlayerId: z.string(),
|
|
12
|
+
userId: z.string().nullable(),
|
|
13
|
+
});
|
|
14
|
+
// Main → Plugin: Health check ping
|
|
15
|
+
const PingMessageSchema = z.object({
|
|
16
|
+
type: z.literal("ping"),
|
|
17
|
+
});
|
|
18
|
+
// Main → Plugin: State sync - sends complete state snapshot
|
|
19
|
+
const StateSyncMessageSchema = z.object({
|
|
20
|
+
type: z.literal("state-sync"),
|
|
21
|
+
syncId: z.number(),
|
|
22
|
+
state: z.custom((data) => {
|
|
23
|
+
return (typeof data === "object" &&
|
|
24
|
+
data !== null &&
|
|
25
|
+
"session" in data &&
|
|
26
|
+
"notifications" in data);
|
|
27
|
+
}),
|
|
28
|
+
});
|
|
29
|
+
// Main → Plugin: Validation result response
|
|
30
|
+
const ValidateInteractionResultMessageSchema = z.object({
|
|
31
|
+
type: z.literal("validate-interaction-result"),
|
|
32
|
+
messageId: z.string(),
|
|
33
|
+
result: z.object({
|
|
34
|
+
valid: z.boolean(),
|
|
35
|
+
errorCode: z.string().nullable().optional(),
|
|
36
|
+
message: z.string().nullable().optional(),
|
|
37
|
+
}),
|
|
38
|
+
});
|
|
39
|
+
const SubmitResultMessageSchema = z.object({
|
|
40
|
+
type: z.literal("submit-result"),
|
|
41
|
+
messageId: z.string(),
|
|
42
|
+
accepted: z.boolean(),
|
|
43
|
+
errorCode: z.string().nullable().optional(),
|
|
44
|
+
message: z.string().nullable().optional(),
|
|
45
|
+
});
|
|
46
|
+
// Union of all messages from main → plugin
|
|
47
|
+
const MainToPluginMessageSchema = z.discriminatedUnion("type", [
|
|
48
|
+
InitMessageSchema,
|
|
49
|
+
PingMessageSchema,
|
|
50
|
+
StateSyncMessageSchema,
|
|
51
|
+
ValidateInteractionResultMessageSchema,
|
|
52
|
+
SubmitResultMessageSchema,
|
|
53
|
+
]);
|
|
54
|
+
/**
|
|
55
|
+
* Mint a client-side correlation id for a single submitted interaction.
|
|
56
|
+
* This id flows plugin -> host gateway -> backend HTTP header
|
|
57
|
+
* (`X-Dreamboard-Client-Action-Id`), and back to the host via the recorded
|
|
58
|
+
* `version -> actionId` map so the full t0..t8 latency trace can be
|
|
59
|
+
* assembled for Tier-0 input-latency observability. Falls back to a
|
|
60
|
+
* timestamp-seeded pseudo-uuid on environments without `crypto.randomUUID`
|
|
61
|
+
* (older sandboxed browsers in tests) so we never crash the plugin.
|
|
62
|
+
*/
|
|
63
|
+
const mintClientActionId = () => {
|
|
64
|
+
const cryptoLike = globalThis.crypto;
|
|
65
|
+
if (cryptoLike?.randomUUID) {
|
|
66
|
+
try {
|
|
67
|
+
return cryptoLike.randomUUID();
|
|
68
|
+
}
|
|
69
|
+
catch {
|
|
70
|
+
// fall through to fallback
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
const rand = Math.random().toString(16).slice(2);
|
|
74
|
+
return `cid-${Date.now().toString(16)}-${rand}`;
|
|
75
|
+
};
|
|
76
|
+
const PLUGIN_RUNTIME_SINGLETON_KEY = "__dreamboardPluginRuntimeApi";
|
|
77
|
+
/**
|
|
78
|
+
* Creates a RuntimeAPI implementation for plugin iframes.
|
|
79
|
+
*
|
|
80
|
+
* Architecture (state-sync):
|
|
81
|
+
* - Host maintains all state in GameSessionStore
|
|
82
|
+
* - Host sends complete state snapshots via state-sync messages
|
|
83
|
+
* - Plugin stores received state and notifies subscribers
|
|
84
|
+
* - No buffering needed - plugin only renders when state exists
|
|
85
|
+
*
|
|
86
|
+
* Security:
|
|
87
|
+
* - Plugin runs in sandboxed iframe (no network access, no same-origin)
|
|
88
|
+
* - All backend communication goes through main app
|
|
89
|
+
*
|
|
90
|
+
* @returns PluginRuntimeAPI instance
|
|
91
|
+
*/
|
|
92
|
+
export function createPluginRuntimeAPI() {
|
|
93
|
+
const existingRuntime = globalThis[PLUGIN_RUNTIME_SINGLETON_KEY];
|
|
94
|
+
if (existingRuntime) {
|
|
95
|
+
return existingRuntime;
|
|
96
|
+
}
|
|
97
|
+
// State-sync state
|
|
98
|
+
let currentStateSnapshot = null;
|
|
99
|
+
const stateListeners = new Set();
|
|
100
|
+
// Session state
|
|
101
|
+
const sessionState = {
|
|
102
|
+
status: "loading",
|
|
103
|
+
sessionId: null,
|
|
104
|
+
controllablePlayerIds: [],
|
|
105
|
+
controllingPlayerId: null,
|
|
106
|
+
userId: null,
|
|
107
|
+
};
|
|
108
|
+
const sessionStateListeners = new Set();
|
|
109
|
+
// Pending validation requests
|
|
110
|
+
const pendingValidations = new Map();
|
|
111
|
+
let validationIdCounter = 0;
|
|
112
|
+
const pendingSubmissions = new Map();
|
|
113
|
+
let submitIdCounter = 0;
|
|
114
|
+
// Helper functions
|
|
115
|
+
const notifySessionStateChange = () => {
|
|
116
|
+
sessionStateListeners.forEach((listener) => {
|
|
117
|
+
try {
|
|
118
|
+
listener({ ...sessionState });
|
|
119
|
+
}
|
|
120
|
+
catch {
|
|
121
|
+
// Silently catch listener errors
|
|
122
|
+
}
|
|
123
|
+
});
|
|
124
|
+
};
|
|
125
|
+
const notifyStateListeners = () => {
|
|
126
|
+
if (!currentStateSnapshot)
|
|
127
|
+
return;
|
|
128
|
+
const snapshot = currentStateSnapshot;
|
|
129
|
+
stateListeners.forEach((listener) => {
|
|
130
|
+
try {
|
|
131
|
+
listener(snapshot);
|
|
132
|
+
}
|
|
133
|
+
catch {
|
|
134
|
+
// Silently catch listener errors
|
|
135
|
+
}
|
|
136
|
+
});
|
|
137
|
+
};
|
|
138
|
+
const createSubmissionError = (errorCode, message) => {
|
|
139
|
+
const error = new Error(message ?? "Submission failed");
|
|
140
|
+
error.name = "SubmissionError";
|
|
141
|
+
error.errorCode = errorCode;
|
|
142
|
+
return error;
|
|
143
|
+
};
|
|
144
|
+
const submitViaParent = (payload) => new Promise((resolve, reject) => {
|
|
145
|
+
const messageId = `submit-${++submitIdCounter}`;
|
|
146
|
+
pendingSubmissions.set(messageId, { resolve, reject });
|
|
147
|
+
// Plugin-iframe `Date.now()` ships alongside the postMessage
|
|
148
|
+
// as the `t0_click` timestamp for Tier-0 input-latency
|
|
149
|
+
// observability. Date.now() (not performance.now()) is
|
|
150
|
+
// intentional: the iframe and the host share a wall-clock
|
|
151
|
+
// base but not a `performance.now()` origin.
|
|
152
|
+
const clientSubmittedAtMs = Date.now();
|
|
153
|
+
if (payload.clientActionId && typeof performance !== "undefined") {
|
|
154
|
+
try {
|
|
155
|
+
performance.mark(`dreamboard.t0_click.${payload.clientActionId}`, {
|
|
156
|
+
detail: { clientActionId: payload.clientActionId },
|
|
157
|
+
});
|
|
158
|
+
}
|
|
159
|
+
catch {
|
|
160
|
+
// performance.mark detail arg not supported in older browsers; ignore
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
window.parent.postMessage({ ...payload, messageId, clientSubmittedAtMs }, "*");
|
|
164
|
+
setTimeout(() => {
|
|
165
|
+
const pending = pendingSubmissions.get(messageId);
|
|
166
|
+
if (!pending) {
|
|
167
|
+
return;
|
|
168
|
+
}
|
|
169
|
+
pendingSubmissions.delete(messageId);
|
|
170
|
+
pending.reject(createSubmissionError("submission-timeout", "Submission request timed out"));
|
|
171
|
+
}, 10000);
|
|
172
|
+
});
|
|
173
|
+
// Message handler
|
|
174
|
+
const handleMessage = (event) => {
|
|
175
|
+
const rawMessage = event.data;
|
|
176
|
+
const parseResult = MainToPluginMessageSchema.safeParse(rawMessage);
|
|
177
|
+
if (!parseResult.success) {
|
|
178
|
+
// Only warn for messages that look like they're meant for us
|
|
179
|
+
if (rawMessage?.type && typeof rawMessage.type === "string") {
|
|
180
|
+
// eslint-disable-next-line no-console
|
|
181
|
+
console.warn("[Plugin RuntimeAPI] Invalid message received:", rawMessage.type);
|
|
182
|
+
}
|
|
183
|
+
return;
|
|
184
|
+
}
|
|
185
|
+
const message = parseResult.data;
|
|
186
|
+
switch (message.type) {
|
|
187
|
+
case "init": {
|
|
188
|
+
// eslint-disable-next-line no-console
|
|
189
|
+
console.log("[Plugin RuntimeAPI] Received init message");
|
|
190
|
+
sessionState.status = "ready";
|
|
191
|
+
sessionState.sessionId = message.sessionId;
|
|
192
|
+
sessionState.controllablePlayerIds =
|
|
193
|
+
message.controllablePlayerIds;
|
|
194
|
+
sessionState.controllingPlayerId =
|
|
195
|
+
message.controllingPlayerId;
|
|
196
|
+
sessionState.userId = message.userId;
|
|
197
|
+
notifySessionStateChange();
|
|
198
|
+
window.parent.postMessage({ type: "ready" }, "*");
|
|
199
|
+
break;
|
|
200
|
+
}
|
|
201
|
+
case "ping": {
|
|
202
|
+
window.parent.postMessage({ type: "pong" }, "*");
|
|
203
|
+
break;
|
|
204
|
+
}
|
|
205
|
+
case "state-sync": {
|
|
206
|
+
// Handle state-sync from host
|
|
207
|
+
// eslint-disable-next-line no-console
|
|
208
|
+
console.log("[Plugin RuntimeAPI] Received state-sync, syncId:", message.syncId);
|
|
209
|
+
// Tier-0 perf: capture `t7_state_sync_received` wall-clock
|
|
210
|
+
// timestamp up-front so the host can stitch it onto the
|
|
211
|
+
// perf HUD via the outgoing state-ack message.
|
|
212
|
+
const clientReceivedAtMs = Date.now();
|
|
213
|
+
if (typeof performance !== "undefined") {
|
|
214
|
+
try {
|
|
215
|
+
performance.mark(`dreamboard.t7_state_sync_received.sync-${message.syncId}`, { detail: { syncId: message.syncId } });
|
|
216
|
+
}
|
|
217
|
+
catch {
|
|
218
|
+
// performance.mark detail arg not supported; ignore
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
currentStateSnapshot = message.state;
|
|
222
|
+
// Update session state from snapshot
|
|
223
|
+
if (message.state.session) {
|
|
224
|
+
sessionState.sessionId = message.state.session.sessionId;
|
|
225
|
+
sessionState.controllablePlayerIds =
|
|
226
|
+
message.state.session.controllablePlayerIds;
|
|
227
|
+
sessionState.controllingPlayerId =
|
|
228
|
+
message.state.session.controllingPlayerId;
|
|
229
|
+
sessionState.userId = message.state.session.userId;
|
|
230
|
+
sessionState.status = "ready";
|
|
231
|
+
notifySessionStateChange();
|
|
232
|
+
}
|
|
233
|
+
// Notify state listeners
|
|
234
|
+
notifyStateListeners();
|
|
235
|
+
// Send acknowledgment (carrying t7 timestamp for the host
|
|
236
|
+
// perf HUD; host ignores it when perf is disabled).
|
|
237
|
+
window.parent.postMessage({
|
|
238
|
+
type: "state-ack",
|
|
239
|
+
syncId: message.syncId,
|
|
240
|
+
clientReceivedAtMs,
|
|
241
|
+
}, "*");
|
|
242
|
+
// Tier-0 perf: after notifyStateListeners has kicked React's
|
|
243
|
+
// render, schedule a post-commit microtask + rAF chain so the
|
|
244
|
+
// follow-up `state-rendered` message lands close to when the
|
|
245
|
+
// plugin's DOM would have been painted. `queueMicrotask` is
|
|
246
|
+
// used first because most React listeners finish synchronously;
|
|
247
|
+
// `requestAnimationFrame` then bounces to the next paint tick.
|
|
248
|
+
const schedulePostRender = () => {
|
|
249
|
+
const send = () => {
|
|
250
|
+
const clientRenderedAtMs = Date.now();
|
|
251
|
+
if (typeof performance !== "undefined") {
|
|
252
|
+
try {
|
|
253
|
+
performance.mark(`dreamboard.t8_render_commit.sync-${message.syncId}`, { detail: { syncId: message.syncId } });
|
|
254
|
+
}
|
|
255
|
+
catch {
|
|
256
|
+
// ignore
|
|
257
|
+
}
|
|
258
|
+
}
|
|
259
|
+
window.parent.postMessage({
|
|
260
|
+
type: "state-rendered",
|
|
261
|
+
syncId: message.syncId,
|
|
262
|
+
clientReceivedAtMs,
|
|
263
|
+
clientRenderedAtMs,
|
|
264
|
+
}, "*");
|
|
265
|
+
};
|
|
266
|
+
if (typeof requestAnimationFrame === "function") {
|
|
267
|
+
requestAnimationFrame(send);
|
|
268
|
+
}
|
|
269
|
+
else {
|
|
270
|
+
queueMicrotask(send);
|
|
271
|
+
}
|
|
272
|
+
};
|
|
273
|
+
queueMicrotask(schedulePostRender);
|
|
274
|
+
break;
|
|
275
|
+
}
|
|
276
|
+
case "validate-interaction-result": {
|
|
277
|
+
const resolver = pendingValidations.get(message.messageId);
|
|
278
|
+
if (resolver) {
|
|
279
|
+
pendingValidations.delete(message.messageId);
|
|
280
|
+
resolver({
|
|
281
|
+
valid: message.result.valid,
|
|
282
|
+
errorCode: message.result.errorCode ?? undefined,
|
|
283
|
+
message: message.result.message ?? undefined,
|
|
284
|
+
});
|
|
285
|
+
}
|
|
286
|
+
break;
|
|
287
|
+
}
|
|
288
|
+
case "submit-result": {
|
|
289
|
+
const pending = pendingSubmissions.get(message.messageId);
|
|
290
|
+
if (!pending) {
|
|
291
|
+
break;
|
|
292
|
+
}
|
|
293
|
+
if (typeof performance !== "undefined") {
|
|
294
|
+
try {
|
|
295
|
+
performance.mark(`dreamboard.t3b_ack.${message.messageId}`, {
|
|
296
|
+
detail: { messageId: message.messageId },
|
|
297
|
+
});
|
|
298
|
+
}
|
|
299
|
+
catch {
|
|
300
|
+
// ignore
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
pendingSubmissions.delete(message.messageId);
|
|
304
|
+
if (message.accepted) {
|
|
305
|
+
pending.resolve();
|
|
306
|
+
}
|
|
307
|
+
else {
|
|
308
|
+
pending.reject(createSubmissionError(message.errorCode ?? undefined, message.message ?? undefined));
|
|
309
|
+
}
|
|
310
|
+
break;
|
|
311
|
+
}
|
|
312
|
+
}
|
|
313
|
+
};
|
|
314
|
+
window.addEventListener("message", handleMessage);
|
|
315
|
+
// Error handlers
|
|
316
|
+
const sendErrorToParent = (message, code, stack) => {
|
|
317
|
+
// eslint-disable-next-line no-console
|
|
318
|
+
console.error(`[Plugin RuntimeAPI] ${code}:`, message, stack || "");
|
|
319
|
+
window.parent.postMessage({
|
|
320
|
+
type: "error",
|
|
321
|
+
message: stack ? `${message}\n${stack}` : message,
|
|
322
|
+
code,
|
|
323
|
+
}, "*");
|
|
324
|
+
};
|
|
325
|
+
window.onerror = (message, source, lineno, colno, error) => {
|
|
326
|
+
const errorMessage = typeof message === "string" ? message : error?.message || "Unknown error";
|
|
327
|
+
const location = source ? ` at ${source}:${lineno}:${colno}` : "";
|
|
328
|
+
sendErrorToParent(errorMessage + location, "UNCAUGHT_ERROR", error?.stack);
|
|
329
|
+
return false;
|
|
330
|
+
};
|
|
331
|
+
window.onunhandledrejection = (event) => {
|
|
332
|
+
const reason = event.reason;
|
|
333
|
+
const message = reason instanceof Error
|
|
334
|
+
? reason.message
|
|
335
|
+
: typeof reason === "string"
|
|
336
|
+
? reason
|
|
337
|
+
: JSON.stringify(reason);
|
|
338
|
+
const stack = reason instanceof Error ? reason.stack : undefined;
|
|
339
|
+
sendErrorToParent(message, "UNHANDLED_REJECTION", stack);
|
|
340
|
+
};
|
|
341
|
+
// eslint-disable-next-line no-console
|
|
342
|
+
console.log("[Plugin RuntimeAPI] ✅ Initialized (state-sync architecture)");
|
|
343
|
+
const runtime = {
|
|
344
|
+
// State-sync methods
|
|
345
|
+
getSnapshot: () => currentStateSnapshot,
|
|
346
|
+
subscribeToState: (listener) => {
|
|
347
|
+
stateListeners.add(listener);
|
|
348
|
+
// Immediately notify with current state if available
|
|
349
|
+
if (currentStateSnapshot) {
|
|
350
|
+
try {
|
|
351
|
+
listener(currentStateSnapshot);
|
|
352
|
+
}
|
|
353
|
+
catch {
|
|
354
|
+
// Silently catch listener errors
|
|
355
|
+
}
|
|
356
|
+
}
|
|
357
|
+
return () => {
|
|
358
|
+
stateListeners.delete(listener);
|
|
359
|
+
};
|
|
360
|
+
},
|
|
361
|
+
validateInteraction: async (playerId, interactionId, params) => {
|
|
362
|
+
return new Promise((resolve) => {
|
|
363
|
+
const messageId = `validate-${++validationIdCounter}`;
|
|
364
|
+
pendingValidations.set(messageId, resolve);
|
|
365
|
+
window.parent.postMessage({
|
|
366
|
+
type: "validate-interaction",
|
|
367
|
+
playerId,
|
|
368
|
+
interactionId,
|
|
369
|
+
params,
|
|
370
|
+
messageId,
|
|
371
|
+
}, "*");
|
|
372
|
+
// Timeout after 10 seconds to avoid hanging forever
|
|
373
|
+
setTimeout(() => {
|
|
374
|
+
if (pendingValidations.has(messageId)) {
|
|
375
|
+
pendingValidations.delete(messageId);
|
|
376
|
+
resolve({
|
|
377
|
+
valid: false,
|
|
378
|
+
errorCode: "validation-timeout",
|
|
379
|
+
message: "Validation request timed out",
|
|
380
|
+
});
|
|
381
|
+
}
|
|
382
|
+
}, 10000);
|
|
383
|
+
});
|
|
384
|
+
},
|
|
385
|
+
submitInteraction: async (playerId, interactionId, params) => submitViaParent({
|
|
386
|
+
type: "interaction",
|
|
387
|
+
playerId,
|
|
388
|
+
interactionId,
|
|
389
|
+
params,
|
|
390
|
+
clientActionId: mintClientActionId(),
|
|
391
|
+
}),
|
|
392
|
+
getSessionState: () => ({ ...sessionState }),
|
|
393
|
+
disconnect: () => {
|
|
394
|
+
window.removeEventListener("message", handleMessage);
|
|
395
|
+
window.onerror = null;
|
|
396
|
+
window.onunhandledrejection = null;
|
|
397
|
+
sessionStateListeners.clear();
|
|
398
|
+
stateListeners.clear();
|
|
399
|
+
pendingValidations.clear();
|
|
400
|
+
pendingSubmissions.clear();
|
|
401
|
+
currentStateSnapshot = null;
|
|
402
|
+
},
|
|
403
|
+
switchPlayer: (playerId) => {
|
|
404
|
+
window.parent.postMessage({ type: "switch-player", playerId }, "*");
|
|
405
|
+
},
|
|
406
|
+
restoreHistory: (entryId) => {
|
|
407
|
+
window.parent.postMessage({ type: "restore-history", entryId }, "*");
|
|
408
|
+
},
|
|
409
|
+
_subscribeToSessionState: (listener) => {
|
|
410
|
+
sessionStateListeners.add(listener);
|
|
411
|
+
listener({ ...sessionState });
|
|
412
|
+
return () => {
|
|
413
|
+
sessionStateListeners.delete(listener);
|
|
414
|
+
};
|
|
415
|
+
},
|
|
416
|
+
};
|
|
417
|
+
globalThis[PLUGIN_RUNTIME_SINGLETON_KEY] = runtime;
|
|
418
|
+
return runtime;
|
|
419
|
+
}
|
|
@@ -0,0 +1,98 @@
|
|
|
1
|
+
import { type CSSProperties, type ReactNode } from "react";
|
|
2
|
+
import { type Theme, type ThemeOverride } from "./tokens.js";
|
|
3
|
+
/**
|
|
4
|
+
* The set of preset theme ids ships in `@dreamboard/ui-sdk`. Authors
|
|
5
|
+
* may register additional ones by passing a fully-resolved {@link Theme}
|
|
6
|
+
* instead of an id.
|
|
7
|
+
*/
|
|
8
|
+
export type ThemePresetId = "tabletop" | "arcade" | "studio";
|
|
9
|
+
/**
|
|
10
|
+
* Resolve a preset id (or a full theme) into a {@link Theme}.
|
|
11
|
+
*
|
|
12
|
+
* Used internally by {@link ThemeProvider}; exported so authors who
|
|
13
|
+
* compose at the call site (e.g. for a side-by-side preview) can do
|
|
14
|
+
* the same resolution without mounting a provider.
|
|
15
|
+
*/
|
|
16
|
+
export declare function resolveTheme(input: ThemePresetId | Theme | undefined): Theme;
|
|
17
|
+
/** Look up a registered preset by id. Returns `undefined` if missing. */
|
|
18
|
+
export declare function getThemePreset(id: ThemePresetId): Theme | undefined;
|
|
19
|
+
/**
|
|
20
|
+
* Subset returned by {@link useTheme} — the resolved theme plus a
|
|
21
|
+
* stable copy of its CSS-variable map so consumers can reuse it (for
|
|
22
|
+
* example, to apply tokens to a portal that escapes the provider).
|
|
23
|
+
*/
|
|
24
|
+
export interface ThemeContextValue {
|
|
25
|
+
/** Fully resolved theme. */
|
|
26
|
+
readonly theme: Theme;
|
|
27
|
+
/** CSS-var map consumable as `style` on a wrapper. */
|
|
28
|
+
readonly cssVars: CSSProperties;
|
|
29
|
+
}
|
|
30
|
+
declare const ThemeCtx: import("react").Context<ThemeContextValue | null>;
|
|
31
|
+
export interface ThemeProviderProps {
|
|
32
|
+
/**
|
|
33
|
+
* Preset id, a full {@link Theme}, or omitted to use the `tabletop`
|
|
34
|
+
* default.
|
|
35
|
+
*/
|
|
36
|
+
theme?: ThemePresetId | Theme;
|
|
37
|
+
/**
|
|
38
|
+
* Deep-partial overrides merged onto the resolved base theme. Use
|
|
39
|
+
* for one-off tweaks (e.g. swapping the player palette per game)
|
|
40
|
+
* without writing a full theme.
|
|
41
|
+
*/
|
|
42
|
+
override?: ThemeOverride;
|
|
43
|
+
/**
|
|
44
|
+
* Render mode for reduced-motion enforcement.
|
|
45
|
+
*
|
|
46
|
+
* - `auto` (default): respect the user's OS-level preference via
|
|
47
|
+
* `prefers-reduced-motion: reduce`.
|
|
48
|
+
* - `force`: force `motion.reducedMotion = "true"` regardless of the
|
|
49
|
+
* OS preference (useful for screenshot CI).
|
|
50
|
+
* - `ignore`: never override motion (use only when the embedding
|
|
51
|
+
* shell already decides motion behaviour).
|
|
52
|
+
*/
|
|
53
|
+
reducedMotion?: "auto" | "force" | "ignore";
|
|
54
|
+
/**
|
|
55
|
+
* Where to mount the wrapper. `block` (default) renders a `div` and
|
|
56
|
+
* applies the CSS variables to it. `none` skips the wrapper entirely
|
|
57
|
+
* — only useful when the parent is already a Dreamboard provider and
|
|
58
|
+
* just needs to publish a different sub-tree (rare).
|
|
59
|
+
*/
|
|
60
|
+
as?: "block" | "none";
|
|
61
|
+
/** Extra style merged after the CSS-var map. */
|
|
62
|
+
style?: CSSProperties;
|
|
63
|
+
/** Extra className for the wrapper. */
|
|
64
|
+
className?: string;
|
|
65
|
+
children: ReactNode;
|
|
66
|
+
}
|
|
67
|
+
/**
|
|
68
|
+
* Mounts a Dreamboard theme. The provider:
|
|
69
|
+
*
|
|
70
|
+
* 1. Resolves the preset (or accepts a full theme).
|
|
71
|
+
* 2. Applies any deep-partial `override`.
|
|
72
|
+
* 3. Optionally clamps `motion.reducedMotion` to `"true"` based on the
|
|
73
|
+
* OS-level preference (or a forced override).
|
|
74
|
+
* 4. Serialises the resolved theme into CSS variables on a wrapper
|
|
75
|
+
* element so descendants can read tokens via `useTheme()` *or* via
|
|
76
|
+
* `var(--db-...)` references in their own CSS / Tailwind.
|
|
77
|
+
*
|
|
78
|
+
* Defaults such as `<GameLayout>` mount this for you with the configured
|
|
79
|
+
* theme; mount it manually only when you need a sub-tree to use a different
|
|
80
|
+
* theme (e.g. a dark sheet over a light layout).
|
|
81
|
+
*/
|
|
82
|
+
export declare function ThemeProvider({ theme: themeInput, override, reducedMotion, as, style, className, children, }: ThemeProviderProps): import("react/jsx-runtime").JSX.Element;
|
|
83
|
+
/**
|
|
84
|
+
* Read the active {@link Theme} from a {@link ThemeProvider} ancestor.
|
|
85
|
+
*
|
|
86
|
+
* Falls back to the `tabletop` preset (with no overrides) when called
|
|
87
|
+
* outside a provider — components remain renderable in isolated tests
|
|
88
|
+
* and Storybook/Cosmos fixtures without a wrapping shell.
|
|
89
|
+
*/
|
|
90
|
+
export declare function useTheme(): Theme;
|
|
91
|
+
/**
|
|
92
|
+
* Read the {@link CSSProperties} that publish the active theme as CSS
|
|
93
|
+
* variables. Useful when porting a theme into a portal or drawer that
|
|
94
|
+
* is rendered outside the provider's DOM subtree.
|
|
95
|
+
*/
|
|
96
|
+
export declare function useThemeCssVars(): CSSProperties;
|
|
97
|
+
export { ThemeCtx };
|
|
98
|
+
//# sourceMappingURL=ThemeProvider.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ThemeProvider.d.ts","sourceRoot":"","sources":["../../src/theme/ThemeProvider.tsx"],"names":[],"mappings":"AAAA,OAAO,EAML,KAAK,aAAa,EAClB,KAAK,SAAS,EACf,MAAM,OAAO,CAAC;AAKf,OAAO,EAAc,KAAK,KAAK,EAAE,KAAK,aAAa,EAAE,MAAM,aAAa,CAAC;AAEzE;;;;GAIG;AACH,MAAM,MAAM,aAAa,GAAG,UAAU,GAAG,QAAQ,GAAG,QAAQ,CAAC;AAQ7D;;;;;;GAMG;AACH,wBAAgB,YAAY,CAAC,KAAK,EAAE,aAAa,GAAG,KAAK,GAAG,SAAS,GAAG,KAAK,CAc5E;AAED,yEAAyE;AACzE,wBAAgB,cAAc,CAAC,EAAE,EAAE,aAAa,GAAG,KAAK,GAAG,SAAS,CAEnE;AAED;;;;GAIG;AACH,MAAM,WAAW,iBAAiB;IAChC,4BAA4B;IAC5B,QAAQ,CAAC,KAAK,EAAE,KAAK,CAAC;IACtB,sDAAsD;IACtD,QAAQ,CAAC,OAAO,EAAE,aAAa,CAAC;CACjC;AAED,QAAA,MAAM,QAAQ,mDAAgD,CAAC;AAE/D,MAAM,WAAW,kBAAkB;IACjC;;;OAGG;IACH,KAAK,CAAC,EAAE,aAAa,GAAG,KAAK,CAAC;IAC9B;;;;OAIG;IACH,QAAQ,CAAC,EAAE,aAAa,CAAC;IACzB;;;;;;;;;OASG;IACH,aAAa,CAAC,EAAE,MAAM,GAAG,OAAO,GAAG,QAAQ,CAAC;IAC5C;;;;;OAKG;IACH,EAAE,CAAC,EAAE,OAAO,GAAG,MAAM,CAAC;IACtB,gDAAgD;IAChD,KAAK,CAAC,EAAE,aAAa,CAAC;IACtB,uCAAuC;IACvC,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,SAAS,CAAC;CACrB;AAeD;;;;;;;;;;;;;;GAcG;AACH,wBAAgB,aAAa,CAAC,EAC5B,KAAK,EAAE,UAAU,EACjB,QAAQ,EACR,aAAsB,EACtB,EAAY,EACZ,KAAK,EACL,SAAS,EACT,QAAQ,GACT,EAAE,kBAAkB,2CAgDpB;AAED;;;;;;GAMG;AACH,wBAAgB,QAAQ,IAAI,KAAK,CAGhC;AAED;;;;GAIG;AACH,wBAAgB,eAAe,IAAI,aAAa,CAQ/C;AAED,OAAO,EAAE,QAAQ,EAAE,CAAC"}
|
|
@@ -0,0 +1,148 @@
|
|
|
1
|
+
import { jsx as _jsx } from "react/jsx-runtime";
|
|
2
|
+
import { createContext, useContext, useEffect, useMemo, useState, } from "react";
|
|
3
|
+
import { themeToCssVars } from "./css-vars.js";
|
|
4
|
+
import { arcadeTheme } from "./presets/arcade.js";
|
|
5
|
+
import { studioTheme } from "./presets/studio.js";
|
|
6
|
+
import { tabletopTheme } from "./presets/tabletop.js";
|
|
7
|
+
import { mergeTheme } from "./tokens.js";
|
|
8
|
+
const PRESETS = {
|
|
9
|
+
tabletop: tabletopTheme,
|
|
10
|
+
arcade: arcadeTheme,
|
|
11
|
+
studio: studioTheme,
|
|
12
|
+
};
|
|
13
|
+
/**
|
|
14
|
+
* Resolve a preset id (or a full theme) into a {@link Theme}.
|
|
15
|
+
*
|
|
16
|
+
* Used internally by {@link ThemeProvider}; exported so authors who
|
|
17
|
+
* compose at the call site (e.g. for a side-by-side preview) can do
|
|
18
|
+
* the same resolution without mounting a provider.
|
|
19
|
+
*/
|
|
20
|
+
export function resolveTheme(input) {
|
|
21
|
+
if (!input)
|
|
22
|
+
return tabletopTheme;
|
|
23
|
+
if (typeof input === "string") {
|
|
24
|
+
const preset = PRESETS[input];
|
|
25
|
+
if (!preset) {
|
|
26
|
+
throw new Error(`[ui-sdk] Unknown theme preset "${input}". Pass a full Theme object or one of: ${Object.keys(PRESETS).join(", ")}.`);
|
|
27
|
+
}
|
|
28
|
+
return preset;
|
|
29
|
+
}
|
|
30
|
+
return input;
|
|
31
|
+
}
|
|
32
|
+
/** Look up a registered preset by id. Returns `undefined` if missing. */
|
|
33
|
+
export function getThemePreset(id) {
|
|
34
|
+
return PRESETS[id];
|
|
35
|
+
}
|
|
36
|
+
const ThemeCtx = createContext(null);
|
|
37
|
+
function usePrefersReducedMotion() {
|
|
38
|
+
const [prefers, setPrefers] = useState(false);
|
|
39
|
+
useEffect(() => {
|
|
40
|
+
if (typeof window === "undefined" || !window.matchMedia)
|
|
41
|
+
return;
|
|
42
|
+
const media = window.matchMedia("(prefers-reduced-motion: reduce)");
|
|
43
|
+
setPrefers(media.matches);
|
|
44
|
+
const handle = (event) => setPrefers(event.matches);
|
|
45
|
+
media.addEventListener("change", handle);
|
|
46
|
+
return () => media.removeEventListener("change", handle);
|
|
47
|
+
}, []);
|
|
48
|
+
return prefers;
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* Mounts a Dreamboard theme. The provider:
|
|
52
|
+
*
|
|
53
|
+
* 1. Resolves the preset (or accepts a full theme).
|
|
54
|
+
* 2. Applies any deep-partial `override`.
|
|
55
|
+
* 3. Optionally clamps `motion.reducedMotion` to `"true"` based on the
|
|
56
|
+
* OS-level preference (or a forced override).
|
|
57
|
+
* 4. Serialises the resolved theme into CSS variables on a wrapper
|
|
58
|
+
* element so descendants can read tokens via `useTheme()` *or* via
|
|
59
|
+
* `var(--db-...)` references in their own CSS / Tailwind.
|
|
60
|
+
*
|
|
61
|
+
* Defaults such as `<GameLayout>` mount this for you with the configured
|
|
62
|
+
* theme; mount it manually only when you need a sub-tree to use a different
|
|
63
|
+
* theme (e.g. a dark sheet over a light layout).
|
|
64
|
+
*/
|
|
65
|
+
export function ThemeProvider({ theme: themeInput, override, reducedMotion = "auto", as = "block", style, className, children, }) {
|
|
66
|
+
const prefersReduced = usePrefersReducedMotion();
|
|
67
|
+
const value = useMemo(() => {
|
|
68
|
+
const base = resolveTheme(themeInput);
|
|
69
|
+
const merged = mergeTheme(base, override);
|
|
70
|
+
const motionResolved = reducedMotion === "force"
|
|
71
|
+
? "true"
|
|
72
|
+
: reducedMotion === "ignore"
|
|
73
|
+
? merged.motion.reducedMotion
|
|
74
|
+
: prefersReduced
|
|
75
|
+
? "true"
|
|
76
|
+
: merged.motion.reducedMotion;
|
|
77
|
+
const final = motionResolved === merged.motion.reducedMotion
|
|
78
|
+
? merged
|
|
79
|
+
: {
|
|
80
|
+
...merged,
|
|
81
|
+
motion: { ...merged.motion, reducedMotion: motionResolved },
|
|
82
|
+
};
|
|
83
|
+
return { theme: final, cssVars: themeToCssVars(final) };
|
|
84
|
+
}, [themeInput, override, reducedMotion, prefersReduced]);
|
|
85
|
+
if (as === "none") {
|
|
86
|
+
return _jsx(ThemeCtx.Provider, { value: value, children: children });
|
|
87
|
+
}
|
|
88
|
+
const wrapperStyle = {
|
|
89
|
+
...themeToShadcnVars(value.theme),
|
|
90
|
+
...value.cssVars,
|
|
91
|
+
fontFamily: value.theme.typography.fontFamily.body,
|
|
92
|
+
...style,
|
|
93
|
+
};
|
|
94
|
+
return (_jsx(ThemeCtx.Provider, { value: value, children: _jsx("div", { "data-dreamboard-theme": value.theme.meta.id, "data-dreamboard-mode": value.theme.meta.mode, "data-dreamboard-reduced-motion": value.theme.motion.reducedMotion, className: className, style: wrapperStyle, children: children }) }));
|
|
95
|
+
}
|
|
96
|
+
/**
|
|
97
|
+
* Read the active {@link Theme} from a {@link ThemeProvider} ancestor.
|
|
98
|
+
*
|
|
99
|
+
* Falls back to the `tabletop` preset (with no overrides) when called
|
|
100
|
+
* outside a provider — components remain renderable in isolated tests
|
|
101
|
+
* and Storybook/Cosmos fixtures without a wrapping shell.
|
|
102
|
+
*/
|
|
103
|
+
export function useTheme() {
|
|
104
|
+
const ctx = useContext(ThemeCtx);
|
|
105
|
+
return ctx?.theme ?? tabletopTheme;
|
|
106
|
+
}
|
|
107
|
+
/**
|
|
108
|
+
* Read the {@link CSSProperties} that publish the active theme as CSS
|
|
109
|
+
* variables. Useful when porting a theme into a portal or drawer that
|
|
110
|
+
* is rendered outside the provider's DOM subtree.
|
|
111
|
+
*/
|
|
112
|
+
export function useThemeCssVars() {
|
|
113
|
+
const ctx = useContext(ThemeCtx);
|
|
114
|
+
if (ctx)
|
|
115
|
+
return { ...themeToShadcnVars(ctx.theme), ...ctx.cssVars };
|
|
116
|
+
// Match the fallback theme exposed by `useTheme()`.
|
|
117
|
+
return {
|
|
118
|
+
...themeToShadcnVars(tabletopTheme),
|
|
119
|
+
...themeToCssVars(tabletopTheme),
|
|
120
|
+
};
|
|
121
|
+
}
|
|
122
|
+
export { ThemeCtx };
|
|
123
|
+
function themeToShadcnVars(theme) {
|
|
124
|
+
const vars = {
|
|
125
|
+
"--background": theme.semantic.surface.app,
|
|
126
|
+
"--foreground": theme.semantic.text.primary,
|
|
127
|
+
"--card": theme.semantic.surface.card,
|
|
128
|
+
"--card-foreground": theme.semantic.text.primary,
|
|
129
|
+
"--popover": theme.semantic.surface.sheet,
|
|
130
|
+
"--popover-foreground": theme.semantic.text.primary,
|
|
131
|
+
"--primary": theme.semantic.intent.primary.solid,
|
|
132
|
+
"--primary-foreground": theme.semantic.intent.primary.on,
|
|
133
|
+
"--secondary": theme.semantic.intent.secondary.solid,
|
|
134
|
+
"--secondary-foreground": theme.semantic.intent.secondary.on,
|
|
135
|
+
"--muted": theme.semantic.surface.inset,
|
|
136
|
+
"--muted-foreground": theme.semantic.text.muted,
|
|
137
|
+
"--accent": theme.semantic.intent.info.soft,
|
|
138
|
+
"--accent-foreground": theme.semantic.intent.info.onSoft,
|
|
139
|
+
"--destructive": theme.semantic.intent.danger.solid,
|
|
140
|
+
"--destructive-foreground": theme.semantic.intent.danger.on,
|
|
141
|
+
"--border": theme.semantic.border.default,
|
|
142
|
+
"--input": theme.semantic.border.default,
|
|
143
|
+
"--ring": theme.semantic.border.focus,
|
|
144
|
+
"--font-sans": theme.typography.fontFamily.body,
|
|
145
|
+
"--font-display": theme.typography.fontFamily.display,
|
|
146
|
+
};
|
|
147
|
+
return vars;
|
|
148
|
+
}
|