@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,437 @@
|
|
|
1
|
+
import { useCallback, useEffect, useMemo, useRef } from "react";
|
|
2
|
+
import {
|
|
3
|
+
useArmedInteraction,
|
|
4
|
+
useInteractionDraft,
|
|
5
|
+
useInteractionSubmitting,
|
|
6
|
+
useInteractionUiStore,
|
|
7
|
+
} from "../context/InteractionDraftContext.js";
|
|
8
|
+
import { useClientParamSchema } from "../context/ClientParamSchemaContext.js";
|
|
9
|
+
import { usePluginState } from "../context/PluginStateContext.js";
|
|
10
|
+
import { usePluginSession } from "../context/PluginSessionContext.js";
|
|
11
|
+
import { useRuntimeContext } from "../context/RuntimeContext.js";
|
|
12
|
+
import {
|
|
13
|
+
ValidationError,
|
|
14
|
+
validationErrorFromUnknown,
|
|
15
|
+
} from "../errors/ValidationError.js";
|
|
16
|
+
import type { InteractionDescriptor } from "../types/plugin-state.js";
|
|
17
|
+
import {
|
|
18
|
+
applyInteractionInputDefaults,
|
|
19
|
+
hasInteractionFieldErrors,
|
|
20
|
+
inputByKey,
|
|
21
|
+
isInputValueReady,
|
|
22
|
+
interactionArmScope,
|
|
23
|
+
interactionInputKeys,
|
|
24
|
+
mergeInteractionFieldErrors,
|
|
25
|
+
validateInteractionInputDomains,
|
|
26
|
+
} from "../utils/interaction-inputs.js";
|
|
27
|
+
|
|
28
|
+
/**
|
|
29
|
+
* Anything that can be used as a submit params object. Subset of TS
|
|
30
|
+
* objects so generic defaults and `keyof Params & string` hold.
|
|
31
|
+
*/
|
|
32
|
+
export type InteractionParamsShape = Record<string, unknown>;
|
|
33
|
+
|
|
34
|
+
export type InteractionHandleStatus = "open" | "submitting" | "submitted";
|
|
35
|
+
|
|
36
|
+
export type DraftValidation<
|
|
37
|
+
Params extends InteractionParamsShape = InteractionParamsShape,
|
|
38
|
+
> =
|
|
39
|
+
| {
|
|
40
|
+
ok: true;
|
|
41
|
+
params: Params;
|
|
42
|
+
fieldErrors: Partial<Record<keyof Params & string, readonly string[]>>;
|
|
43
|
+
formErrors: readonly string[];
|
|
44
|
+
missing: ReadonlyArray<keyof Params & string>;
|
|
45
|
+
}
|
|
46
|
+
| {
|
|
47
|
+
ok: false;
|
|
48
|
+
fieldErrors: Partial<Record<keyof Params & string, readonly string[]>>;
|
|
49
|
+
formErrors: readonly string[];
|
|
50
|
+
missing: ReadonlyArray<keyof Params & string>;
|
|
51
|
+
};
|
|
52
|
+
|
|
53
|
+
/**
|
|
54
|
+
* Bound handle around an {@link InteractionDescriptor}. Surfaces call into
|
|
55
|
+
* this hook to submit/validate an interaction, track draft input state for
|
|
56
|
+
* multi-step prompts, and arm/disarm themselves on a surface.
|
|
57
|
+
*
|
|
58
|
+
* Availability flags are mirrored from the authoritative descriptor; UI
|
|
59
|
+
* MUST NOT recompute eligibility locally.
|
|
60
|
+
*
|
|
61
|
+
* When the surrounding workspace knows the concrete params shape (e.g.
|
|
62
|
+
* obtained from the generated `InteractionParamsOf<Id>` alias), parameterise
|
|
63
|
+
* this handle on `Params` so `submit`, `draft`, and `setInput` are all
|
|
64
|
+
* statically typed. The default `Record<string, unknown>` preserves the
|
|
65
|
+
* loosely-typed fallback for generic infrastructure.
|
|
66
|
+
*/
|
|
67
|
+
export interface InteractionHandle<
|
|
68
|
+
Params extends InteractionParamsShape = InteractionParamsShape,
|
|
69
|
+
DefaultedKeys extends keyof Params & string = never,
|
|
70
|
+
> {
|
|
71
|
+
descriptor: InteractionDescriptor;
|
|
72
|
+
/** Draft commit policy projected by the reducer. */
|
|
73
|
+
commit: InteractionDescriptor["commit"];
|
|
74
|
+
/**
|
|
75
|
+
* Submit the interaction. When `params` is omitted the current draft (as
|
|
76
|
+
* mutated by {@link InteractionHandle.setInput}) is used instead, which is
|
|
77
|
+
* the common case for multi-input prompts.
|
|
78
|
+
*/
|
|
79
|
+
submit: (params?: Params) => Promise<void>;
|
|
80
|
+
/** Run server validation using `params` (or the current draft by default). */
|
|
81
|
+
validate: (params?: Params) => Promise<void>;
|
|
82
|
+
/** Run local generated client-schema validation against the current draft. */
|
|
83
|
+
validateDraft: () => DraftValidation<Params>;
|
|
84
|
+
/** Run server validation against the current draft. */
|
|
85
|
+
validateDraftServer: () => Promise<void>;
|
|
86
|
+
/**
|
|
87
|
+
* Validate the current draft locally, submit parsed params, then clear the
|
|
88
|
+
* draft only when submission succeeds.
|
|
89
|
+
*/
|
|
90
|
+
submitDraft: () => Promise<void>;
|
|
91
|
+
/** Mirrors `descriptor.available`. */
|
|
92
|
+
available: boolean;
|
|
93
|
+
/** Mirrors `descriptor.unavailableReason`. */
|
|
94
|
+
unavailableReason?: string;
|
|
95
|
+
/** Local/authoritative progress for this interaction. */
|
|
96
|
+
status: InteractionHandleStatus;
|
|
97
|
+
|
|
98
|
+
// --- Draft state --------------------------------------------------------
|
|
99
|
+
|
|
100
|
+
/** Live draft values for this interaction. Empty object when unset. */
|
|
101
|
+
draft: Readonly<Partial<Params>>;
|
|
102
|
+
/**
|
|
103
|
+
* Draft values with authored input defaults applied. Fields with declared
|
|
104
|
+
* defaults are typed as present; other draft fields remain partial.
|
|
105
|
+
*/
|
|
106
|
+
values: Readonly<Partial<Params> & Pick<Params, DefaultedKeys>>;
|
|
107
|
+
/** Set a single input on the draft. */
|
|
108
|
+
setInput: <K extends keyof Params & string>(key: K, value: Params[K]) => void;
|
|
109
|
+
/** Clear a single input (or the whole draft when `key` is omitted). */
|
|
110
|
+
clearInput: (key?: keyof Params & string) => void;
|
|
111
|
+
/**
|
|
112
|
+
* True when every key declared on `descriptor.inputs` has a value in
|
|
113
|
+
* the draft. Falls back to `true` when the descriptor declares no inputs.
|
|
114
|
+
*/
|
|
115
|
+
isReady: boolean;
|
|
116
|
+
/**
|
|
117
|
+
* True when this interaction is the currently armed one on its surface.
|
|
118
|
+
* Armed interactions are the ones that surfaces like `BoardSurface` use
|
|
119
|
+
* to highlight eligible targets and route clicks.
|
|
120
|
+
*/
|
|
121
|
+
isArmed: boolean;
|
|
122
|
+
/** Arm this interaction on its surface (disarms any previously armed). */
|
|
123
|
+
arm: () => void;
|
|
124
|
+
/** Disarm this interaction (if it was armed). */
|
|
125
|
+
disarm: () => void;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
/**
|
|
129
|
+
* Bind an {@link InteractionDescriptor} to submit/validate helpers plus
|
|
130
|
+
* draft + arming state. Use this from any surface; draft state is shared
|
|
131
|
+
* across components through {@link InteractionUiProvider}, which
|
|
132
|
+
* `PluginRuntime` auto-mounts.
|
|
133
|
+
*
|
|
134
|
+
* Example:
|
|
135
|
+
* ```tsx
|
|
136
|
+
* const handle = useInteractionHandle(placeThing);
|
|
137
|
+
* handle.setInput("cardId", card.id);
|
|
138
|
+
* handle.setInput("ringId", ring.id);
|
|
139
|
+
* if (handle.isReady) await handle.submit();
|
|
140
|
+
* ```
|
|
141
|
+
*/
|
|
142
|
+
export function useInteractionHandle<
|
|
143
|
+
Params extends InteractionParamsShape = InteractionParamsShape,
|
|
144
|
+
DefaultedKeys extends keyof Params & string = never,
|
|
145
|
+
>(descriptor: InteractionDescriptor): InteractionHandle<Params, DefaultedKeys> {
|
|
146
|
+
const runtime = useRuntimeContext();
|
|
147
|
+
const { controllingPlayerId } = usePluginSession();
|
|
148
|
+
const store = useInteractionUiStore();
|
|
149
|
+
const submittingRef = useRef(false);
|
|
150
|
+
const autoSubmitSignatureRef = useRef<string | null>(null);
|
|
151
|
+
const simultaneousPhase = usePluginState(
|
|
152
|
+
(state) => state.gameplay.simultaneousPhase,
|
|
153
|
+
);
|
|
154
|
+
|
|
155
|
+
const { interactionId, interactionKey, phaseName } = descriptor;
|
|
156
|
+
const armScope = interactionArmScope(descriptor);
|
|
157
|
+
const inputKeys = useMemo(
|
|
158
|
+
() => interactionInputKeys(descriptor),
|
|
159
|
+
[descriptor],
|
|
160
|
+
);
|
|
161
|
+
const paramsSchema = useClientParamSchema(phaseName, interactionId);
|
|
162
|
+
const draft = useInteractionDraft(interactionKey) as Readonly<
|
|
163
|
+
Partial<Params>
|
|
164
|
+
>;
|
|
165
|
+
const values = useMemo(
|
|
166
|
+
() =>
|
|
167
|
+
applyInteractionInputDefaults<Params>(descriptor, draft) as Readonly<
|
|
168
|
+
Partial<Params> & Pick<Params, DefaultedKeys>
|
|
169
|
+
>,
|
|
170
|
+
[descriptor, draft],
|
|
171
|
+
);
|
|
172
|
+
const armedId = useArmedInteraction(armScope);
|
|
173
|
+
const isArmed = armedId === interactionKey;
|
|
174
|
+
const submitting = useInteractionSubmitting(interactionKey);
|
|
175
|
+
const submitted =
|
|
176
|
+
controllingPlayerId !== null &&
|
|
177
|
+
simultaneousPhase?.phaseName === phaseName &&
|
|
178
|
+
simultaneousPhase.interactionId === interactionId &&
|
|
179
|
+
simultaneousPhase.sealedPlayerIds.includes(controllingPlayerId);
|
|
180
|
+
const status: InteractionHandleStatus = submitted
|
|
181
|
+
? "submitted"
|
|
182
|
+
: submitting
|
|
183
|
+
? "submitting"
|
|
184
|
+
: "open";
|
|
185
|
+
|
|
186
|
+
const isReady = useMemo(() => {
|
|
187
|
+
const required = inputKeys;
|
|
188
|
+
if (required.length === 0) return true;
|
|
189
|
+
return required.every((key) => {
|
|
190
|
+
const input = inputByKey(descriptor, key);
|
|
191
|
+
const value = (values as Record<string, unknown>)[key];
|
|
192
|
+
return input
|
|
193
|
+
? isInputValueReady(input, value)
|
|
194
|
+
: value !== null && value !== undefined;
|
|
195
|
+
});
|
|
196
|
+
}, [descriptor, values, inputKeys]);
|
|
197
|
+
|
|
198
|
+
const requirePlayer = useCallback(() => {
|
|
199
|
+
if (!controllingPlayerId) {
|
|
200
|
+
throw new Error("useInteractionHandle: no controlling player available");
|
|
201
|
+
}
|
|
202
|
+
return controllingPlayerId;
|
|
203
|
+
}, [controllingPlayerId]);
|
|
204
|
+
|
|
205
|
+
const submit = useCallback(
|
|
206
|
+
async (params?: Params) => {
|
|
207
|
+
if (status !== "open" || submittingRef.current) {
|
|
208
|
+
throw new ValidationError(
|
|
209
|
+
status === "submitted" ? "ALREADY_SUBMITTED" : "SUBMITTING",
|
|
210
|
+
status === "submitted"
|
|
211
|
+
? "Interaction has already been submitted."
|
|
212
|
+
: "Interaction submission is already in progress.",
|
|
213
|
+
);
|
|
214
|
+
}
|
|
215
|
+
submittingRef.current = true;
|
|
216
|
+
store.setSubmitting(interactionKey, true);
|
|
217
|
+
const finalParams = applyInteractionInputDefaults<Params>(
|
|
218
|
+
descriptor,
|
|
219
|
+
params ?? values,
|
|
220
|
+
) as Params;
|
|
221
|
+
try {
|
|
222
|
+
await runtime.submitInteraction(
|
|
223
|
+
requirePlayer(),
|
|
224
|
+
interactionId,
|
|
225
|
+
finalParams as Record<string, unknown>,
|
|
226
|
+
);
|
|
227
|
+
store.clearInput(interactionKey);
|
|
228
|
+
if (store.getArmed(armScope) === interactionKey) {
|
|
229
|
+
store.arm(armScope, null);
|
|
230
|
+
}
|
|
231
|
+
} catch (error) {
|
|
232
|
+
throw validationErrorFromUnknown(error);
|
|
233
|
+
} finally {
|
|
234
|
+
submittingRef.current = false;
|
|
235
|
+
store.setSubmitting(interactionKey, false);
|
|
236
|
+
}
|
|
237
|
+
},
|
|
238
|
+
[
|
|
239
|
+
descriptor,
|
|
240
|
+
values,
|
|
241
|
+
runtime,
|
|
242
|
+
requirePlayer,
|
|
243
|
+
interactionId,
|
|
244
|
+
interactionKey,
|
|
245
|
+
armScope,
|
|
246
|
+
store,
|
|
247
|
+
status,
|
|
248
|
+
],
|
|
249
|
+
);
|
|
250
|
+
|
|
251
|
+
const validate = useCallback(
|
|
252
|
+
async (params?: Params) => {
|
|
253
|
+
const finalParams = applyInteractionInputDefaults<Params>(
|
|
254
|
+
descriptor,
|
|
255
|
+
params ?? values,
|
|
256
|
+
) as Params;
|
|
257
|
+
const result = await runtime.validateInteraction(
|
|
258
|
+
requirePlayer(),
|
|
259
|
+
interactionId,
|
|
260
|
+
finalParams as Record<string, unknown>,
|
|
261
|
+
);
|
|
262
|
+
if (!result.valid) {
|
|
263
|
+
throw new ValidationError(result.errorCode, result.message);
|
|
264
|
+
}
|
|
265
|
+
},
|
|
266
|
+
[descriptor, values, runtime, requirePlayer, interactionId],
|
|
267
|
+
);
|
|
268
|
+
|
|
269
|
+
const validateDraft = useCallback((): DraftValidation<Params> => {
|
|
270
|
+
const rawDraft = { ...values } as Record<string, unknown>;
|
|
271
|
+
const required = inputKeys;
|
|
272
|
+
const missing = required.filter((key) => {
|
|
273
|
+
const input = inputByKey(descriptor, key);
|
|
274
|
+
const value = rawDraft[key];
|
|
275
|
+
return input
|
|
276
|
+
? !isInputValueReady(input, value)
|
|
277
|
+
: value === null || value === undefined;
|
|
278
|
+
}) as Array<keyof Params & string>;
|
|
279
|
+
const domainFieldErrors = validateInteractionInputDomains(
|
|
280
|
+
descriptor,
|
|
281
|
+
rawDraft,
|
|
282
|
+
) as Partial<Record<keyof Params & string, readonly string[]>>;
|
|
283
|
+
|
|
284
|
+
if (!paramsSchema) {
|
|
285
|
+
if (missing.length > 0 || hasInteractionFieldErrors(domainFieldErrors)) {
|
|
286
|
+
return {
|
|
287
|
+
ok: false,
|
|
288
|
+
fieldErrors: domainFieldErrors,
|
|
289
|
+
formErrors: [],
|
|
290
|
+
missing,
|
|
291
|
+
};
|
|
292
|
+
}
|
|
293
|
+
return {
|
|
294
|
+
ok: true,
|
|
295
|
+
params: rawDraft as Params,
|
|
296
|
+
fieldErrors: {},
|
|
297
|
+
formErrors: [],
|
|
298
|
+
missing: [],
|
|
299
|
+
};
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
const result = paramsSchema.safeParse(rawDraft);
|
|
303
|
+
if (result.success) {
|
|
304
|
+
if (hasInteractionFieldErrors(domainFieldErrors)) {
|
|
305
|
+
return {
|
|
306
|
+
ok: false,
|
|
307
|
+
fieldErrors: domainFieldErrors,
|
|
308
|
+
formErrors: [],
|
|
309
|
+
missing,
|
|
310
|
+
};
|
|
311
|
+
}
|
|
312
|
+
return {
|
|
313
|
+
ok: true,
|
|
314
|
+
params: result.data as Params,
|
|
315
|
+
fieldErrors: {},
|
|
316
|
+
formErrors: [],
|
|
317
|
+
missing: [],
|
|
318
|
+
};
|
|
319
|
+
}
|
|
320
|
+
|
|
321
|
+
const fieldErrors: Partial<
|
|
322
|
+
Record<keyof Params & string, readonly string[]>
|
|
323
|
+
> = {};
|
|
324
|
+
const formErrors: string[] = [];
|
|
325
|
+
const missingSet = new Set<PropertyKey>(missing);
|
|
326
|
+
for (const issue of result.error.issues) {
|
|
327
|
+
const [first] = issue.path;
|
|
328
|
+
if (typeof first === "string" && required.includes(first)) {
|
|
329
|
+
if (missingSet.has(first)) continue;
|
|
330
|
+
const key = first as keyof Params & string;
|
|
331
|
+
fieldErrors[key] = [...(fieldErrors[key] ?? []), issue.message];
|
|
332
|
+
} else {
|
|
333
|
+
formErrors.push(issue.message);
|
|
334
|
+
}
|
|
335
|
+
}
|
|
336
|
+
return {
|
|
337
|
+
ok: false,
|
|
338
|
+
fieldErrors: mergeInteractionFieldErrors(
|
|
339
|
+
fieldErrors,
|
|
340
|
+
domainFieldErrors,
|
|
341
|
+
) as Partial<Record<keyof Params & string, readonly string[]>>,
|
|
342
|
+
formErrors,
|
|
343
|
+
missing,
|
|
344
|
+
};
|
|
345
|
+
}, [descriptor, values, inputKeys, paramsSchema]);
|
|
346
|
+
|
|
347
|
+
const validateDraftServer = useCallback(async () => {
|
|
348
|
+
await validate({ ...values } as Params);
|
|
349
|
+
}, [values, validate]);
|
|
350
|
+
|
|
351
|
+
const submitDraft = useCallback(async () => {
|
|
352
|
+
const validation = validateDraft();
|
|
353
|
+
if (!validation.ok) {
|
|
354
|
+
const message =
|
|
355
|
+
validation.formErrors[0] ??
|
|
356
|
+
Object.values(validation.fieldErrors).flat()[0] ??
|
|
357
|
+
(validation.missing.length > 0
|
|
358
|
+
? "Required inputs are missing"
|
|
359
|
+
: "Draft validation failed");
|
|
360
|
+
throw new ValidationError("INVALID_DRAFT", message);
|
|
361
|
+
}
|
|
362
|
+
await submit(validation.params);
|
|
363
|
+
}, [submit, validateDraft]);
|
|
364
|
+
|
|
365
|
+
useEffect(() => {
|
|
366
|
+
if (descriptor.commit.mode !== "autoWhenReady") {
|
|
367
|
+
autoSubmitSignatureRef.current = null;
|
|
368
|
+
return;
|
|
369
|
+
}
|
|
370
|
+
if (!descriptor.available || status !== "open" || !isReady) {
|
|
371
|
+
if (!isReady) autoSubmitSignatureRef.current = null;
|
|
372
|
+
return;
|
|
373
|
+
}
|
|
374
|
+
const validation = validateDraft();
|
|
375
|
+
if (!validation.ok) return;
|
|
376
|
+
const signature = `${interactionKey}:${JSON.stringify(validation.params)}`;
|
|
377
|
+
if (autoSubmitSignatureRef.current === signature) return;
|
|
378
|
+
autoSubmitSignatureRef.current = signature;
|
|
379
|
+
void submit(validation.params).catch(() => {
|
|
380
|
+
// Runtime error channels surface the failure. Keep the draft intact and
|
|
381
|
+
// suppress repeated attempts until the player changes the draft.
|
|
382
|
+
});
|
|
383
|
+
}, [
|
|
384
|
+
descriptor.available,
|
|
385
|
+
descriptor.commit.mode,
|
|
386
|
+
interactionKey,
|
|
387
|
+
isReady,
|
|
388
|
+
status,
|
|
389
|
+
submit,
|
|
390
|
+
validateDraft,
|
|
391
|
+
]);
|
|
392
|
+
|
|
393
|
+
const setInput = useCallback(
|
|
394
|
+
<K extends keyof Params & string>(key: K, value: Params[K]) => {
|
|
395
|
+
store.setInput(interactionKey, key, value as unknown);
|
|
396
|
+
},
|
|
397
|
+
[store, interactionKey],
|
|
398
|
+
);
|
|
399
|
+
|
|
400
|
+
const clearInput = useCallback(
|
|
401
|
+
(key?: keyof Params & string) => {
|
|
402
|
+
store.clearInput(interactionKey, key);
|
|
403
|
+
},
|
|
404
|
+
[store, interactionKey],
|
|
405
|
+
);
|
|
406
|
+
|
|
407
|
+
const arm = useCallback(() => {
|
|
408
|
+
store.arm(armScope, interactionKey);
|
|
409
|
+
}, [store, armScope, interactionKey]);
|
|
410
|
+
|
|
411
|
+
const disarm = useCallback(() => {
|
|
412
|
+
if (store.getArmed(armScope) === interactionKey) {
|
|
413
|
+
store.arm(armScope, null);
|
|
414
|
+
}
|
|
415
|
+
}, [store, armScope, interactionKey]);
|
|
416
|
+
|
|
417
|
+
return {
|
|
418
|
+
descriptor,
|
|
419
|
+
commit: descriptor.commit,
|
|
420
|
+
available: descriptor.available,
|
|
421
|
+
unavailableReason: descriptor.unavailableReason,
|
|
422
|
+
status,
|
|
423
|
+
submit,
|
|
424
|
+
validate,
|
|
425
|
+
validateDraft,
|
|
426
|
+
validateDraftServer,
|
|
427
|
+
submitDraft,
|
|
428
|
+
draft,
|
|
429
|
+
values,
|
|
430
|
+
setInput,
|
|
431
|
+
clearInput,
|
|
432
|
+
isReady,
|
|
433
|
+
isArmed,
|
|
434
|
+
arm,
|
|
435
|
+
disarm,
|
|
436
|
+
};
|
|
437
|
+
}
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import { useState, useEffect } from "react";
|
|
2
|
+
|
|
3
|
+
const MOBILE_BREAKPOINT = 768;
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* Hook to detect if the user is on a mobile/small screen device
|
|
7
|
+
* @param breakpoint - The width threshold in pixels (default: 768)
|
|
8
|
+
* @returns boolean indicating if the screen is mobile-sized
|
|
9
|
+
*/
|
|
10
|
+
export function useIsMobile(breakpoint: number = MOBILE_BREAKPOINT): boolean {
|
|
11
|
+
const [isMobile, setIsMobile] = useState(() => {
|
|
12
|
+
// SSR-safe: default to false if window is not available
|
|
13
|
+
if (typeof window === "undefined") return false;
|
|
14
|
+
return window.innerWidth < breakpoint;
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
useEffect(() => {
|
|
18
|
+
if (typeof window === "undefined") return;
|
|
19
|
+
|
|
20
|
+
const mediaQuery = window.matchMedia(`(max-width: ${breakpoint - 1}px)`);
|
|
21
|
+
|
|
22
|
+
const handleChange = (e: MediaQueryListEvent | MediaQueryList) => {
|
|
23
|
+
setIsMobile(e.matches);
|
|
24
|
+
};
|
|
25
|
+
|
|
26
|
+
// Set initial value
|
|
27
|
+
handleChange(mediaQuery);
|
|
28
|
+
|
|
29
|
+
// Listen for changes
|
|
30
|
+
mediaQuery.addEventListener("change", handleChange);
|
|
31
|
+
return () => mediaQuery.removeEventListener("change", handleChange);
|
|
32
|
+
}, [breakpoint]);
|
|
33
|
+
|
|
34
|
+
return isMobile;
|
|
35
|
+
}
|
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
import { expect, test } from "bun:test";
|
|
2
|
+
import { renderToString } from "react-dom/server";
|
|
3
|
+
import { RuntimeContext } from "../context/RuntimeContext.js";
|
|
4
|
+
import { PluginSessionContext } from "../context/PluginSessionContext.js";
|
|
5
|
+
import type { PluginStateSnapshot } from "../types/plugin-state.js";
|
|
6
|
+
import type { PluginSessionState, RuntimeAPI } from "../types/runtime-api.js";
|
|
7
|
+
import { useIsMyTurn } from "./useIsMyTurn.js";
|
|
8
|
+
|
|
9
|
+
function createSessionState(
|
|
10
|
+
overrides: Partial<PluginSessionState> = {},
|
|
11
|
+
): PluginSessionState {
|
|
12
|
+
return {
|
|
13
|
+
status: "ready",
|
|
14
|
+
sessionId: "session-1",
|
|
15
|
+
controllablePlayerIds: ["player-1"],
|
|
16
|
+
controllingPlayerId: "player-1",
|
|
17
|
+
userId: "user-1",
|
|
18
|
+
...overrides,
|
|
19
|
+
};
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
function createSnapshot(
|
|
23
|
+
session: PluginSessionState,
|
|
24
|
+
activePlayers: string[],
|
|
25
|
+
): PluginStateSnapshot {
|
|
26
|
+
return {
|
|
27
|
+
view: null,
|
|
28
|
+
gameplay: {
|
|
29
|
+
currentPhase: "takeTurn",
|
|
30
|
+
currentStage: null,
|
|
31
|
+
activePlayers,
|
|
32
|
+
availableInteractions: [],
|
|
33
|
+
zones: {},
|
|
34
|
+
},
|
|
35
|
+
lobby: null,
|
|
36
|
+
notifications: [],
|
|
37
|
+
session,
|
|
38
|
+
history: null,
|
|
39
|
+
syncId: 1,
|
|
40
|
+
};
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
function createRuntime(snapshot: PluginStateSnapshot): RuntimeAPI & {
|
|
44
|
+
getSnapshot: () => PluginStateSnapshot;
|
|
45
|
+
subscribeToState: (
|
|
46
|
+
listener: (state: PluginStateSnapshot) => void,
|
|
47
|
+
) => () => void;
|
|
48
|
+
} {
|
|
49
|
+
return {
|
|
50
|
+
validateInteraction: async () => ({ valid: true }),
|
|
51
|
+
submitInteraction: async () => undefined,
|
|
52
|
+
getSessionState: () => snapshot.session,
|
|
53
|
+
disconnect: () => undefined,
|
|
54
|
+
getSnapshot: () => snapshot,
|
|
55
|
+
subscribeToState: () => () => undefined,
|
|
56
|
+
};
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
test("useIsMyTurn returns true when the controlled player is active", () => {
|
|
60
|
+
const session = createSessionState();
|
|
61
|
+
const runtime = createRuntime(createSnapshot(session, ["player-1"]));
|
|
62
|
+
let captured = false;
|
|
63
|
+
|
|
64
|
+
function Harness() {
|
|
65
|
+
captured = useIsMyTurn();
|
|
66
|
+
return null;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
renderToString(
|
|
70
|
+
<RuntimeContext.Provider value={runtime}>
|
|
71
|
+
<PluginSessionContext.Provider value={session}>
|
|
72
|
+
<Harness />
|
|
73
|
+
</PluginSessionContext.Provider>
|
|
74
|
+
</RuntimeContext.Provider>,
|
|
75
|
+
);
|
|
76
|
+
|
|
77
|
+
expect(captured).toBe(true);
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
test("useIsMyTurn returns false when the controlled player is not active", () => {
|
|
81
|
+
const session = createSessionState();
|
|
82
|
+
const runtime = createRuntime(createSnapshot(session, ["player-2"]));
|
|
83
|
+
let captured = true;
|
|
84
|
+
|
|
85
|
+
function Harness() {
|
|
86
|
+
captured = useIsMyTurn();
|
|
87
|
+
return null;
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
renderToString(
|
|
91
|
+
<RuntimeContext.Provider value={runtime}>
|
|
92
|
+
<PluginSessionContext.Provider value={session}>
|
|
93
|
+
<Harness />
|
|
94
|
+
</PluginSessionContext.Provider>
|
|
95
|
+
</RuntimeContext.Provider>,
|
|
96
|
+
);
|
|
97
|
+
|
|
98
|
+
expect(captured).toBe(false);
|
|
99
|
+
});
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { usePluginState } from "../context/PluginStateContext.js";
|
|
2
|
+
import { usePluginSession } from "../context/PluginSessionContext.js";
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Returns whether the currently controlled player is one of the engine-tracked
|
|
6
|
+
* active players for the current gameplay snapshot.
|
|
7
|
+
*/
|
|
8
|
+
export function useIsMyTurn(): boolean {
|
|
9
|
+
const { controllingPlayerId } = usePluginSession();
|
|
10
|
+
const activePlayers = usePluginState((state) => state.gameplay.activePlayers);
|
|
11
|
+
|
|
12
|
+
return (
|
|
13
|
+
controllingPlayerId !== null && activePlayers.includes(controllingPlayerId)
|
|
14
|
+
);
|
|
15
|
+
}
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
import { useState, useEffect } from "react";
|
|
2
|
+
import { useRuntimeContext } from "../context/RuntimeContext.js";
|
|
3
|
+
import type { LobbyState } from "../types/plugin-state.js";
|
|
4
|
+
import type { PluginRuntimeAPI } from "../runtime/createPluginRuntimeAPI.js";
|
|
5
|
+
|
|
6
|
+
// Re-export LobbyState for convenience
|
|
7
|
+
export type { LobbyState };
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* Subscribes to lobby updates from the plugin runtime snapshot when present.
|
|
11
|
+
* Returns `null` until the host provides lobby state (SSR/tests/minimal runtimes).
|
|
12
|
+
*/
|
|
13
|
+
export function useLobbyState(): LobbyState | null {
|
|
14
|
+
const runtime = useRuntimeContext() as PluginRuntimeAPI;
|
|
15
|
+
|
|
16
|
+
const getStateFromSnapshot = (): LobbyState | null => {
|
|
17
|
+
if (!runtime.getSnapshot) return null;
|
|
18
|
+
const snapshot = runtime.getSnapshot();
|
|
19
|
+
if (!snapshot?.lobby) return null;
|
|
20
|
+
return snapshot.lobby;
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
const [lobbyState, setLobbyState] = useState<LobbyState | null>(
|
|
24
|
+
getStateFromSnapshot,
|
|
25
|
+
);
|
|
26
|
+
|
|
27
|
+
useEffect(() => {
|
|
28
|
+
if (!runtime.subscribeToState) {
|
|
29
|
+
return;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
const initialState = runtime.getSnapshot?.();
|
|
33
|
+
if (initialState?.lobby) {
|
|
34
|
+
setLobbyState(initialState.lobby);
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
return runtime.subscribeToState((snapshot) => {
|
|
38
|
+
if (snapshot.lobby) {
|
|
39
|
+
setLobbyState(snapshot.lobby);
|
|
40
|
+
}
|
|
41
|
+
});
|
|
42
|
+
}, [runtime]);
|
|
43
|
+
|
|
44
|
+
return lobbyState;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
/**
|
|
48
|
+
* Hook to subscribe to lobby state updates.
|
|
49
|
+
* Returns the latest lobby information from state-sync messages.
|
|
50
|
+
*
|
|
51
|
+
* State is provided by PluginStateProvider from host's state-sync messages.
|
|
52
|
+
* The host transforms raw SSE LOBBY_UPDATE messages into clean LobbyState objects.
|
|
53
|
+
*
|
|
54
|
+
* @returns Current lobby state (never null - throws if not available)
|
|
55
|
+
* @throws Error if lobby state is not available
|
|
56
|
+
*
|
|
57
|
+
* @example
|
|
58
|
+
* ```typescript
|
|
59
|
+
* function LobbyScreen() {
|
|
60
|
+
* const lobby = useLobby();
|
|
61
|
+
* // lobby is guaranteed to be non-null
|
|
62
|
+
* return <div>{lobby.seats.length} seats</div>;
|
|
63
|
+
* }
|
|
64
|
+
* ```
|
|
65
|
+
*/
|
|
66
|
+
export function useLobby(): LobbyState {
|
|
67
|
+
const lobbyState = useLobbyState();
|
|
68
|
+
if (lobbyState === null) {
|
|
69
|
+
throw new Error(
|
|
70
|
+
"useLobby: Lobby state not available. " +
|
|
71
|
+
"The host should only render the plugin when lobby state is ready.",
|
|
72
|
+
);
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
return lobbyState;
|
|
76
|
+
}
|