@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,252 @@
|
|
|
1
|
+
import {
|
|
2
|
+
createContext,
|
|
3
|
+
useContext,
|
|
4
|
+
useEffect,
|
|
5
|
+
useMemo,
|
|
6
|
+
useState,
|
|
7
|
+
type CSSProperties,
|
|
8
|
+
type ReactNode,
|
|
9
|
+
} from "react";
|
|
10
|
+
import { themeToCssVars } from "./css-vars.js";
|
|
11
|
+
import { arcadeTheme } from "./presets/arcade.js";
|
|
12
|
+
import { studioTheme } from "./presets/studio.js";
|
|
13
|
+
import { tabletopTheme } from "./presets/tabletop.js";
|
|
14
|
+
import { mergeTheme, type Theme, type ThemeOverride } from "./tokens.js";
|
|
15
|
+
|
|
16
|
+
/**
|
|
17
|
+
* The set of preset theme ids ships in `@dreamboard/ui-sdk`. Authors
|
|
18
|
+
* may register additional ones by passing a fully-resolved {@link Theme}
|
|
19
|
+
* instead of an id.
|
|
20
|
+
*/
|
|
21
|
+
export type ThemePresetId = "tabletop" | "arcade" | "studio";
|
|
22
|
+
|
|
23
|
+
const PRESETS: Readonly<Record<ThemePresetId, Theme>> = {
|
|
24
|
+
tabletop: tabletopTheme,
|
|
25
|
+
arcade: arcadeTheme,
|
|
26
|
+
studio: studioTheme,
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
/**
|
|
30
|
+
* Resolve a preset id (or a full theme) into a {@link Theme}.
|
|
31
|
+
*
|
|
32
|
+
* Used internally by {@link ThemeProvider}; exported so authors who
|
|
33
|
+
* compose at the call site (e.g. for a side-by-side preview) can do
|
|
34
|
+
* the same resolution without mounting a provider.
|
|
35
|
+
*/
|
|
36
|
+
export function resolveTheme(input: ThemePresetId | Theme | undefined): Theme {
|
|
37
|
+
if (!input) return tabletopTheme;
|
|
38
|
+
if (typeof input === "string") {
|
|
39
|
+
const preset = PRESETS[input];
|
|
40
|
+
if (!preset) {
|
|
41
|
+
throw new Error(
|
|
42
|
+
`[ui-sdk] Unknown theme preset "${input}". Pass a full Theme object or one of: ${Object.keys(
|
|
43
|
+
PRESETS,
|
|
44
|
+
).join(", ")}.`,
|
|
45
|
+
);
|
|
46
|
+
}
|
|
47
|
+
return preset;
|
|
48
|
+
}
|
|
49
|
+
return input;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
/** Look up a registered preset by id. Returns `undefined` if missing. */
|
|
53
|
+
export function getThemePreset(id: ThemePresetId): Theme | undefined {
|
|
54
|
+
return PRESETS[id];
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
/**
|
|
58
|
+
* Subset returned by {@link useTheme} — the resolved theme plus a
|
|
59
|
+
* stable copy of its CSS-variable map so consumers can reuse it (for
|
|
60
|
+
* example, to apply tokens to a portal that escapes the provider).
|
|
61
|
+
*/
|
|
62
|
+
export interface ThemeContextValue {
|
|
63
|
+
/** Fully resolved theme. */
|
|
64
|
+
readonly theme: Theme;
|
|
65
|
+
/** CSS-var map consumable as `style` on a wrapper. */
|
|
66
|
+
readonly cssVars: CSSProperties;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
const ThemeCtx = createContext<ThemeContextValue | null>(null);
|
|
70
|
+
|
|
71
|
+
export interface ThemeProviderProps {
|
|
72
|
+
/**
|
|
73
|
+
* Preset id, a full {@link Theme}, or omitted to use the `tabletop`
|
|
74
|
+
* default.
|
|
75
|
+
*/
|
|
76
|
+
theme?: ThemePresetId | Theme;
|
|
77
|
+
/**
|
|
78
|
+
* Deep-partial overrides merged onto the resolved base theme. Use
|
|
79
|
+
* for one-off tweaks (e.g. swapping the player palette per game)
|
|
80
|
+
* without writing a full theme.
|
|
81
|
+
*/
|
|
82
|
+
override?: ThemeOverride;
|
|
83
|
+
/**
|
|
84
|
+
* Render mode for reduced-motion enforcement.
|
|
85
|
+
*
|
|
86
|
+
* - `auto` (default): respect the user's OS-level preference via
|
|
87
|
+
* `prefers-reduced-motion: reduce`.
|
|
88
|
+
* - `force`: force `motion.reducedMotion = "true"` regardless of the
|
|
89
|
+
* OS preference (useful for screenshot CI).
|
|
90
|
+
* - `ignore`: never override motion (use only when the embedding
|
|
91
|
+
* shell already decides motion behaviour).
|
|
92
|
+
*/
|
|
93
|
+
reducedMotion?: "auto" | "force" | "ignore";
|
|
94
|
+
/**
|
|
95
|
+
* Where to mount the wrapper. `block` (default) renders a `div` and
|
|
96
|
+
* applies the CSS variables to it. `none` skips the wrapper entirely
|
|
97
|
+
* — only useful when the parent is already a Dreamboard provider and
|
|
98
|
+
* just needs to publish a different sub-tree (rare).
|
|
99
|
+
*/
|
|
100
|
+
as?: "block" | "none";
|
|
101
|
+
/** Extra style merged after the CSS-var map. */
|
|
102
|
+
style?: CSSProperties;
|
|
103
|
+
/** Extra className for the wrapper. */
|
|
104
|
+
className?: string;
|
|
105
|
+
children: ReactNode;
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
function usePrefersReducedMotion(): boolean {
|
|
109
|
+
const [prefers, setPrefers] = useState(false);
|
|
110
|
+
useEffect(() => {
|
|
111
|
+
if (typeof window === "undefined" || !window.matchMedia) return;
|
|
112
|
+
const media = window.matchMedia("(prefers-reduced-motion: reduce)");
|
|
113
|
+
setPrefers(media.matches);
|
|
114
|
+
const handle = (event: MediaQueryListEvent) => setPrefers(event.matches);
|
|
115
|
+
media.addEventListener("change", handle);
|
|
116
|
+
return () => media.removeEventListener("change", handle);
|
|
117
|
+
}, []);
|
|
118
|
+
return prefers;
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
/**
|
|
122
|
+
* Mounts a Dreamboard theme. The provider:
|
|
123
|
+
*
|
|
124
|
+
* 1. Resolves the preset (or accepts a full theme).
|
|
125
|
+
* 2. Applies any deep-partial `override`.
|
|
126
|
+
* 3. Optionally clamps `motion.reducedMotion` to `"true"` based on the
|
|
127
|
+
* OS-level preference (or a forced override).
|
|
128
|
+
* 4. Serialises the resolved theme into CSS variables on a wrapper
|
|
129
|
+
* element so descendants can read tokens via `useTheme()` *or* via
|
|
130
|
+
* `var(--db-...)` references in their own CSS / Tailwind.
|
|
131
|
+
*
|
|
132
|
+
* Defaults such as `<GameLayout>` mount this for you with the configured
|
|
133
|
+
* theme; mount it manually only when you need a sub-tree to use a different
|
|
134
|
+
* theme (e.g. a dark sheet over a light layout).
|
|
135
|
+
*/
|
|
136
|
+
export function ThemeProvider({
|
|
137
|
+
theme: themeInput,
|
|
138
|
+
override,
|
|
139
|
+
reducedMotion = "auto",
|
|
140
|
+
as = "block",
|
|
141
|
+
style,
|
|
142
|
+
className,
|
|
143
|
+
children,
|
|
144
|
+
}: ThemeProviderProps) {
|
|
145
|
+
const prefersReduced = usePrefersReducedMotion();
|
|
146
|
+
|
|
147
|
+
const value = useMemo<ThemeContextValue>(() => {
|
|
148
|
+
const base = resolveTheme(themeInput);
|
|
149
|
+
const merged = mergeTheme(base, override);
|
|
150
|
+
const motionResolved =
|
|
151
|
+
reducedMotion === "force"
|
|
152
|
+
? "true"
|
|
153
|
+
: reducedMotion === "ignore"
|
|
154
|
+
? merged.motion.reducedMotion
|
|
155
|
+
: prefersReduced
|
|
156
|
+
? "true"
|
|
157
|
+
: merged.motion.reducedMotion;
|
|
158
|
+
const final: Theme =
|
|
159
|
+
motionResolved === merged.motion.reducedMotion
|
|
160
|
+
? merged
|
|
161
|
+
: {
|
|
162
|
+
...merged,
|
|
163
|
+
motion: { ...merged.motion, reducedMotion: motionResolved },
|
|
164
|
+
};
|
|
165
|
+
return { theme: final, cssVars: themeToCssVars(final) };
|
|
166
|
+
}, [themeInput, override, reducedMotion, prefersReduced]);
|
|
167
|
+
|
|
168
|
+
if (as === "none") {
|
|
169
|
+
return <ThemeCtx.Provider value={value}>{children}</ThemeCtx.Provider>;
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
const wrapperStyle: CSSProperties = {
|
|
173
|
+
...themeToShadcnVars(value.theme),
|
|
174
|
+
...value.cssVars,
|
|
175
|
+
fontFamily: value.theme.typography.fontFamily.body,
|
|
176
|
+
...style,
|
|
177
|
+
};
|
|
178
|
+
|
|
179
|
+
return (
|
|
180
|
+
<ThemeCtx.Provider value={value}>
|
|
181
|
+
<div
|
|
182
|
+
data-dreamboard-theme={value.theme.meta.id}
|
|
183
|
+
data-dreamboard-mode={value.theme.meta.mode}
|
|
184
|
+
data-dreamboard-reduced-motion={value.theme.motion.reducedMotion}
|
|
185
|
+
className={className}
|
|
186
|
+
style={wrapperStyle}
|
|
187
|
+
>
|
|
188
|
+
{children}
|
|
189
|
+
</div>
|
|
190
|
+
</ThemeCtx.Provider>
|
|
191
|
+
);
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
/**
|
|
195
|
+
* Read the active {@link Theme} from a {@link ThemeProvider} ancestor.
|
|
196
|
+
*
|
|
197
|
+
* Falls back to the `tabletop` preset (with no overrides) when called
|
|
198
|
+
* outside a provider — components remain renderable in isolated tests
|
|
199
|
+
* and Storybook/Cosmos fixtures without a wrapping shell.
|
|
200
|
+
*/
|
|
201
|
+
export function useTheme(): Theme {
|
|
202
|
+
const ctx = useContext(ThemeCtx);
|
|
203
|
+
return ctx?.theme ?? tabletopTheme;
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
/**
|
|
207
|
+
* Read the {@link CSSProperties} that publish the active theme as CSS
|
|
208
|
+
* variables. Useful when porting a theme into a portal or drawer that
|
|
209
|
+
* is rendered outside the provider's DOM subtree.
|
|
210
|
+
*/
|
|
211
|
+
export function useThemeCssVars(): CSSProperties {
|
|
212
|
+
const ctx = useContext(ThemeCtx);
|
|
213
|
+
if (ctx) return { ...themeToShadcnVars(ctx.theme), ...ctx.cssVars };
|
|
214
|
+
// Match the fallback theme exposed by `useTheme()`.
|
|
215
|
+
return {
|
|
216
|
+
...themeToShadcnVars(tabletopTheme),
|
|
217
|
+
...themeToCssVars(tabletopTheme),
|
|
218
|
+
};
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
export { ThemeCtx };
|
|
222
|
+
|
|
223
|
+
type CssVariableStyle = CSSProperties & {
|
|
224
|
+
[K in `--${string}`]?: string | number;
|
|
225
|
+
};
|
|
226
|
+
|
|
227
|
+
function themeToShadcnVars(theme: Theme): CSSProperties {
|
|
228
|
+
const vars: CssVariableStyle = {
|
|
229
|
+
"--background": theme.semantic.surface.app,
|
|
230
|
+
"--foreground": theme.semantic.text.primary,
|
|
231
|
+
"--card": theme.semantic.surface.card,
|
|
232
|
+
"--card-foreground": theme.semantic.text.primary,
|
|
233
|
+
"--popover": theme.semantic.surface.sheet,
|
|
234
|
+
"--popover-foreground": theme.semantic.text.primary,
|
|
235
|
+
"--primary": theme.semantic.intent.primary.solid,
|
|
236
|
+
"--primary-foreground": theme.semantic.intent.primary.on,
|
|
237
|
+
"--secondary": theme.semantic.intent.secondary.solid,
|
|
238
|
+
"--secondary-foreground": theme.semantic.intent.secondary.on,
|
|
239
|
+
"--muted": theme.semantic.surface.inset,
|
|
240
|
+
"--muted-foreground": theme.semantic.text.muted,
|
|
241
|
+
"--accent": theme.semantic.intent.info.soft,
|
|
242
|
+
"--accent-foreground": theme.semantic.intent.info.onSoft,
|
|
243
|
+
"--destructive": theme.semantic.intent.danger.solid,
|
|
244
|
+
"--destructive-foreground": theme.semantic.intent.danger.on,
|
|
245
|
+
"--border": theme.semantic.border.default,
|
|
246
|
+
"--input": theme.semantic.border.default,
|
|
247
|
+
"--ring": theme.semantic.border.focus,
|
|
248
|
+
"--font-sans": theme.typography.fontFamily.body,
|
|
249
|
+
"--font-display": theme.typography.fontFamily.display,
|
|
250
|
+
};
|
|
251
|
+
return vars;
|
|
252
|
+
}
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Board theming helpers.
|
|
3
|
+
*
|
|
4
|
+
* The SDK's board renderers (`<HexGrid>`, `<SquareGrid>`,
|
|
5
|
+
* `<NetworkGraph>`, `<ZoneMap>`, `<TrackBoard>`, `<SlotSystem>`) need a
|
|
6
|
+
* tiny but consistent set of visual tokens — frame border, frame fill,
|
|
7
|
+
* eligible-target hint, and active hover ring — so authors don't have
|
|
8
|
+
* to re-pick colours per board. {@link useBoardTheme} exposes those
|
|
9
|
+
* tokens, derived from the active {@link Theme}.
|
|
10
|
+
*/
|
|
11
|
+
|
|
12
|
+
import { useMemo } from "react";
|
|
13
|
+
import { useTheme } from "./ThemeProvider.js";
|
|
14
|
+
import type { Theme } from "./tokens.js";
|
|
15
|
+
|
|
16
|
+
/**
|
|
17
|
+
* Resolved board tokens consumed by board renderers and per-tile
|
|
18
|
+
* default renderers. All values are CSS-ready strings.
|
|
19
|
+
*/
|
|
20
|
+
export interface BoardTheme {
|
|
21
|
+
/** Border drawn around the board's outer frame. */
|
|
22
|
+
readonly frameBorder: string;
|
|
23
|
+
/** Background painted under the tile renderer (visible at tile gaps). */
|
|
24
|
+
readonly frameBackground: string;
|
|
25
|
+
/** Hover-state ring drawn over an interactive target. */
|
|
26
|
+
readonly hoverRing: string;
|
|
27
|
+
/** Idle hint colour for an eligible-but-unhovered target. */
|
|
28
|
+
readonly eligibleHint: string;
|
|
29
|
+
/** Border drawn on inactive tiles (matches semantic.border.default). */
|
|
30
|
+
readonly tileBorder: string;
|
|
31
|
+
/** Subtle dotted border for non-interactive scaffolding (e.g. ports). */
|
|
32
|
+
readonly tileBorderSubtle: string;
|
|
33
|
+
/** Foreground used for board labels (numbers, port ratios). */
|
|
34
|
+
readonly tileText: string;
|
|
35
|
+
/** Recommended ring stroke width in px for hover/eligible markers. */
|
|
36
|
+
readonly ringWidth: number;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
/**
|
|
40
|
+
* Derive a {@link BoardTheme} from the active theme. Memoised on the
|
|
41
|
+
* theme reference so consumers can pass it straight to a board's
|
|
42
|
+
* `defaultTileProps` / `defaultEdgeProps` slots.
|
|
43
|
+
*/
|
|
44
|
+
export function useBoardTheme(): BoardTheme {
|
|
45
|
+
const theme = useTheme();
|
|
46
|
+
return useMemo(() => deriveBoardTheme(theme), [theme]);
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
/** Standalone derivation for renderers that already hold a {@link Theme}. */
|
|
50
|
+
export function deriveBoardTheme(theme: Theme): BoardTheme {
|
|
51
|
+
return {
|
|
52
|
+
frameBorder: theme.component.board.frameBorder,
|
|
53
|
+
frameBackground: theme.component.board.frameBackground,
|
|
54
|
+
hoverRing: theme.component.board.hoverRing,
|
|
55
|
+
eligibleHint: theme.component.board.eligibleHint,
|
|
56
|
+
tileBorder: theme.semantic.border.default,
|
|
57
|
+
tileBorderSubtle: theme.semantic.border.subtle,
|
|
58
|
+
tileText: theme.semantic.text.primary,
|
|
59
|
+
ringWidth: 3,
|
|
60
|
+
};
|
|
61
|
+
}
|
|
@@ -0,0 +1,105 @@
|
|
|
1
|
+
import type { CSSProperties } from "react";
|
|
2
|
+
import type { Theme } from "./tokens.js";
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* CSS variable bridge for {@link Theme}.
|
|
6
|
+
*
|
|
7
|
+
* The {@link ThemeProvider} mounts a wrapper element and applies the
|
|
8
|
+
* map returned by {@link themeToCssVars} as inline `style`, which
|
|
9
|
+
* declares one CSS custom property per leaf token. Components inside
|
|
10
|
+
* the provider can:
|
|
11
|
+
*
|
|
12
|
+
* 1. Read tokens via `useTheme()` (typed) and use them inline, or
|
|
13
|
+
* 2. Read the CSS variable directly in Tailwind / className styling
|
|
14
|
+
* via the helpers in {@link var()} below.
|
|
15
|
+
*
|
|
16
|
+
* The variable naming scheme is `--db-<dot.path>` with `.` and any
|
|
17
|
+
* non-identifier characters replaced by `-`. Example:
|
|
18
|
+
*
|
|
19
|
+
* - `theme.semantic.intent.primary.solid` → `--db-semantic-intent-primary-solid`
|
|
20
|
+
* - `theme.player[0].solid` → `--db-player-1-solid`
|
|
21
|
+
* - `theme.color.brand.600` → `--db-color-brand-600`
|
|
22
|
+
*
|
|
23
|
+
* The 1-based player suffix (rather than 0-based) keeps the variable
|
|
24
|
+
* name aligned with how seats are labelled in product UI ("Player 1").
|
|
25
|
+
*/
|
|
26
|
+
|
|
27
|
+
const PREFIX = "--db";
|
|
28
|
+
|
|
29
|
+
/** Convert a token-tree path to a kebab-case CSS variable name. */
|
|
30
|
+
function toVarName(parts: ReadonlyArray<string | number>): string {
|
|
31
|
+
const tail = parts
|
|
32
|
+
.map((part) => String(part))
|
|
33
|
+
.join("-")
|
|
34
|
+
.replace(/[^a-zA-Z0-9-]/g, "-")
|
|
35
|
+
.replace(/-+/g, "-");
|
|
36
|
+
return `${PREFIX}-${tail}`;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
/**
|
|
40
|
+
* Walk the theme tree and produce a flat `{ "--db-...": value }` map.
|
|
41
|
+
* Arrays are walked positionally with a 1-based index (so
|
|
42
|
+
* `theme.player[0]` → `--db-player-1-...`).
|
|
43
|
+
*/
|
|
44
|
+
export function themeToCssVars(theme: Theme): CSSProperties {
|
|
45
|
+
const out: Record<string, string> = {};
|
|
46
|
+
walk(theme as unknown as Record<string, unknown>, [], out);
|
|
47
|
+
return out as CSSProperties;
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
function walk(
|
|
51
|
+
node: unknown,
|
|
52
|
+
path: ReadonlyArray<string | number>,
|
|
53
|
+
out: Record<string, string>,
|
|
54
|
+
): void {
|
|
55
|
+
if (node === null || node === undefined) return;
|
|
56
|
+
if (typeof node === "string" || typeof node === "number") {
|
|
57
|
+
out[toVarName(path)] = String(node);
|
|
58
|
+
return;
|
|
59
|
+
}
|
|
60
|
+
if (Array.isArray(node)) {
|
|
61
|
+
node.forEach((child, idx) => {
|
|
62
|
+
// 1-based index for player slots, etc.
|
|
63
|
+
walk(child, [...path, idx + 1], out);
|
|
64
|
+
});
|
|
65
|
+
return;
|
|
66
|
+
}
|
|
67
|
+
if (typeof node === "object") {
|
|
68
|
+
for (const [key, value] of Object.entries(
|
|
69
|
+
node as Record<string, unknown>,
|
|
70
|
+
)) {
|
|
71
|
+
walk(value, [...path, key], out);
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
/**
|
|
77
|
+
* Build a `var(--db-...)` reference for a token path.
|
|
78
|
+
*
|
|
79
|
+
* @example
|
|
80
|
+
* ```ts
|
|
81
|
+
* cssVar("semantic", "intent", "primary", "solid")
|
|
82
|
+
* // => "var(--db-semantic-intent-primary-solid)"
|
|
83
|
+
* cssVar("player", 1, "solid")
|
|
84
|
+
* // => "var(--db-player-1-solid)"
|
|
85
|
+
* ```
|
|
86
|
+
*/
|
|
87
|
+
export function cssVar(...path: Array<string | number>): string {
|
|
88
|
+
return `var(${toVarName(path)})`;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
/**
|
|
92
|
+
* Convenience for declaring a `var(--db-...)` with an inline fallback.
|
|
93
|
+
*
|
|
94
|
+
* @example
|
|
95
|
+
* ```ts
|
|
96
|
+
* cssVarOr("transparent", "semantic", "border", "default")
|
|
97
|
+
* // => "var(--db-semantic-border-default, transparent)"
|
|
98
|
+
* ```
|
|
99
|
+
*/
|
|
100
|
+
export function cssVarOr(
|
|
101
|
+
fallback: string,
|
|
102
|
+
...path: Array<string | number>
|
|
103
|
+
): string {
|
|
104
|
+
return `var(${toVarName(path)}, ${fallback})`;
|
|
105
|
+
}
|
|
@@ -0,0 +1,240 @@
|
|
|
1
|
+
import type { CSSProperties } from "react";
|
|
2
|
+
import type { IntentColor, Theme } from "./tokens.js";
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Helpers that derive component-specific style fragments from a
|
|
6
|
+
* resolved {@link Theme}. They live alongside the theme module (rather
|
|
7
|
+
* than in each component) so that:
|
|
8
|
+
*
|
|
9
|
+
* - There is one canonical mapping of "intent → button style", "intent →
|
|
10
|
+
* chip style", etc.
|
|
11
|
+
* - Component visuals stay aligned automatically when an author tweaks
|
|
12
|
+
* the theme; nobody has to remember to update a bespoke calc inside
|
|
13
|
+
* `<Card>` or `<DefaultInteractionButton>`.
|
|
14
|
+
*
|
|
15
|
+
* Each helper returns a plain {@link CSSProperties} fragment so the
|
|
16
|
+
* caller can spread it into an inline style (or pass it through to
|
|
17
|
+
* Framer Motion's `style` prop).
|
|
18
|
+
*/
|
|
19
|
+
|
|
20
|
+
/** Button visual variants derived from semantic intent slots. */
|
|
21
|
+
export type ButtonVariant =
|
|
22
|
+
| "primary"
|
|
23
|
+
| "secondary"
|
|
24
|
+
| "danger"
|
|
25
|
+
| "submitted"
|
|
26
|
+
| "success"
|
|
27
|
+
| "warning"
|
|
28
|
+
| "info"
|
|
29
|
+
| "ghost";
|
|
30
|
+
|
|
31
|
+
export type ButtonSize = "sm" | "md" | "lg";
|
|
32
|
+
|
|
33
|
+
/**
|
|
34
|
+
* Compute the inline style for a button in `variant`/`size` from the
|
|
35
|
+
* theme. `disabled` and `pressed` are visual-only — they affect colour
|
|
36
|
+
* intensity and elevation but never touch interactivity (the consuming
|
|
37
|
+
* component owns `aria-disabled` and event wiring).
|
|
38
|
+
*/
|
|
39
|
+
export function buttonStyle(
|
|
40
|
+
theme: Theme,
|
|
41
|
+
options: {
|
|
42
|
+
variant?: ButtonVariant;
|
|
43
|
+
size?: ButtonSize;
|
|
44
|
+
disabled?: boolean;
|
|
45
|
+
pressed?: boolean;
|
|
46
|
+
} = {},
|
|
47
|
+
): CSSProperties {
|
|
48
|
+
const { variant = "primary", size = "md", disabled, pressed } = options;
|
|
49
|
+
const intent = intentForVariant(theme, variant);
|
|
50
|
+
const sizing = BUTTON_SIZE[size];
|
|
51
|
+
const visuallyDisabled = disabled && variant !== "submitted";
|
|
52
|
+
|
|
53
|
+
const baseShadow =
|
|
54
|
+
variant === "ghost"
|
|
55
|
+
? "none"
|
|
56
|
+
: pressed
|
|
57
|
+
? theme.elevation.rest
|
|
58
|
+
: theme.elevation.hover;
|
|
59
|
+
|
|
60
|
+
return {
|
|
61
|
+
display: "inline-flex",
|
|
62
|
+
alignItems: "center",
|
|
63
|
+
justifyContent: "center",
|
|
64
|
+
gap: theme.space[2],
|
|
65
|
+
minHeight: sizing.minHeight,
|
|
66
|
+
paddingBlock: sizing.paddingBlock,
|
|
67
|
+
paddingInline: sizing.paddingInline,
|
|
68
|
+
borderRadius: theme.radius.md,
|
|
69
|
+
border:
|
|
70
|
+
variant === "ghost"
|
|
71
|
+
? "1px solid transparent"
|
|
72
|
+
: `1px solid ${visuallyDisabled ? theme.semantic.border.subtle : intent.border}`,
|
|
73
|
+
background:
|
|
74
|
+
variant === "ghost"
|
|
75
|
+
? "transparent"
|
|
76
|
+
: visuallyDisabled
|
|
77
|
+
? theme.semantic.surface.inset
|
|
78
|
+
: intent.solid,
|
|
79
|
+
color: visuallyDisabled
|
|
80
|
+
? theme.semantic.text.disabled
|
|
81
|
+
: variant === "ghost"
|
|
82
|
+
? theme.semantic.text.primary
|
|
83
|
+
: intent.on,
|
|
84
|
+
fontFamily: theme.typography.fontFamily.body,
|
|
85
|
+
fontSize: sizing.fontSize,
|
|
86
|
+
fontWeight: theme.typography.fontWeight.bold,
|
|
87
|
+
lineHeight: theme.typography.lineHeight.tight,
|
|
88
|
+
letterSpacing: theme.typography.letterSpacing.normal,
|
|
89
|
+
cursor: disabled ? "not-allowed" : "pointer",
|
|
90
|
+
boxShadow: disabled ? "none" : baseShadow,
|
|
91
|
+
transition: `background-color ${theme.motion.duration.fast} ${theme.motion.easing.out}, transform ${theme.motion.duration.fast} ${theme.motion.easing.out}, box-shadow ${theme.motion.duration.normal} ${theme.motion.easing.out}`,
|
|
92
|
+
userSelect: "none",
|
|
93
|
+
WebkitTapHighlightColor: "transparent",
|
|
94
|
+
};
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
const BUTTON_SIZE: Record<
|
|
98
|
+
ButtonSize,
|
|
99
|
+
{
|
|
100
|
+
minHeight: number;
|
|
101
|
+
paddingBlock: string;
|
|
102
|
+
paddingInline: string;
|
|
103
|
+
fontSize: string;
|
|
104
|
+
}
|
|
105
|
+
> = {
|
|
106
|
+
sm: {
|
|
107
|
+
minHeight: 32,
|
|
108
|
+
paddingBlock: "0.25rem",
|
|
109
|
+
paddingInline: "0.625rem",
|
|
110
|
+
fontSize: "0.8125rem",
|
|
111
|
+
},
|
|
112
|
+
md: {
|
|
113
|
+
minHeight: 40,
|
|
114
|
+
paddingBlock: "0.5rem",
|
|
115
|
+
paddingInline: "0.875rem",
|
|
116
|
+
fontSize: "0.9375rem",
|
|
117
|
+
},
|
|
118
|
+
lg: {
|
|
119
|
+
minHeight: 52,
|
|
120
|
+
paddingBlock: "0.75rem",
|
|
121
|
+
paddingInline: "1.125rem",
|
|
122
|
+
fontSize: "1.0625rem",
|
|
123
|
+
},
|
|
124
|
+
};
|
|
125
|
+
|
|
126
|
+
/**
|
|
127
|
+
* Soft-intent chip ("Your turn", "Largest army", "Waiting…"). Reads
|
|
128
|
+
* `intent.<variant>.soft` for background and `.onSoft` for text.
|
|
129
|
+
*/
|
|
130
|
+
export function chipStyle(
|
|
131
|
+
theme: Theme,
|
|
132
|
+
options: {
|
|
133
|
+
variant?: Exclude<ButtonVariant, "ghost">;
|
|
134
|
+
size?: "sm" | "md";
|
|
135
|
+
} = {},
|
|
136
|
+
): CSSProperties {
|
|
137
|
+
const { variant = "secondary", size = "sm" } = options;
|
|
138
|
+
const intent = intentForVariant(theme, variant);
|
|
139
|
+
return {
|
|
140
|
+
display: "inline-flex",
|
|
141
|
+
alignItems: "center",
|
|
142
|
+
gap: theme.space[1],
|
|
143
|
+
paddingBlock: size === "md" ? "0.25rem" : "0.125rem",
|
|
144
|
+
paddingInline: size === "md" ? "0.625rem" : "0.5rem",
|
|
145
|
+
borderRadius: theme.radius.pill,
|
|
146
|
+
border: `1px solid ${intent.border}`,
|
|
147
|
+
background: intent.soft,
|
|
148
|
+
color: intent.onSoft,
|
|
149
|
+
fontFamily: theme.typography.fontFamily.body,
|
|
150
|
+
fontSize:
|
|
151
|
+
size === "md"
|
|
152
|
+
? theme.typography.fontSize.sm
|
|
153
|
+
: theme.typography.fontSize.xs,
|
|
154
|
+
fontWeight: theme.typography.fontWeight.bold,
|
|
155
|
+
letterSpacing: theme.typography.letterSpacing.wide,
|
|
156
|
+
textTransform: "uppercase",
|
|
157
|
+
whiteSpace: "nowrap",
|
|
158
|
+
};
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
/**
|
|
162
|
+
* Standard surface card (player card, action panel, hand drawer). The
|
|
163
|
+
* `tone` selects which `surface.*` slot to read; `interactive` adds
|
|
164
|
+
* the hover-elevation transition baseline.
|
|
165
|
+
*/
|
|
166
|
+
export function surfaceStyle(
|
|
167
|
+
theme: Theme,
|
|
168
|
+
options: {
|
|
169
|
+
tone?: keyof Theme["semantic"]["surface"];
|
|
170
|
+
interactive?: boolean;
|
|
171
|
+
/** Override radius to one of the theme tokens. Defaults to `lg`. */
|
|
172
|
+
radius?: keyof Theme["radius"];
|
|
173
|
+
} = {},
|
|
174
|
+
): CSSProperties {
|
|
175
|
+
const { tone = "card", interactive, radius = "lg" } = options;
|
|
176
|
+
return {
|
|
177
|
+
background: theme.semantic.surface[tone],
|
|
178
|
+
color: theme.semantic.text.primary,
|
|
179
|
+
border: `1px solid ${theme.semantic.border.subtle}`,
|
|
180
|
+
borderRadius: theme.radius[radius],
|
|
181
|
+
boxShadow: theme.elevation.rest,
|
|
182
|
+
transition: interactive
|
|
183
|
+
? `box-shadow ${theme.motion.duration.normal} ${theme.motion.easing.out}, transform ${theme.motion.duration.normal} ${theme.motion.easing.out}`
|
|
184
|
+
: undefined,
|
|
185
|
+
};
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
/** Resolve a {@link ButtonVariant} to its underlying {@link IntentColor}. */
|
|
189
|
+
export function intentForVariant(
|
|
190
|
+
theme: Theme,
|
|
191
|
+
variant: ButtonVariant,
|
|
192
|
+
): IntentColor {
|
|
193
|
+
switch (variant) {
|
|
194
|
+
case "primary":
|
|
195
|
+
return theme.semantic.intent.primary;
|
|
196
|
+
case "secondary":
|
|
197
|
+
return theme.semantic.intent.secondary;
|
|
198
|
+
case "danger":
|
|
199
|
+
return theme.semantic.intent.danger;
|
|
200
|
+
case "submitted":
|
|
201
|
+
return theme.semantic.intent.success;
|
|
202
|
+
case "success":
|
|
203
|
+
return theme.semantic.intent.success;
|
|
204
|
+
case "warning":
|
|
205
|
+
return theme.semantic.intent.warning;
|
|
206
|
+
case "info":
|
|
207
|
+
return theme.semantic.intent.info;
|
|
208
|
+
case "ghost":
|
|
209
|
+
return theme.semantic.intent.secondary;
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
/**
|
|
214
|
+
* Pick a {@link Theme.player} entry by 0-based seat slot, wrapping
|
|
215
|
+
* around when there are more players than palette entries (so a
|
|
216
|
+
* 7-player game cycles back to player 1's palette).
|
|
217
|
+
*/
|
|
218
|
+
export function playerColor(theme: Theme, slot: number) {
|
|
219
|
+
const length = theme.player.length;
|
|
220
|
+
const idx = ((slot % length) + length) % length;
|
|
221
|
+
// The {@link Theme.player} contract pins length === 6 at the type
|
|
222
|
+
// level, but indexing by a runtime-computed value widens the result
|
|
223
|
+
// to `PlayerColor | undefined`. Falling back to slot 0 (also typed
|
|
224
|
+
// `PlayerColor`) keeps the return type narrow without a non-null
|
|
225
|
+
// assertion.
|
|
226
|
+
return theme.player[idx] ?? theme.player[0];
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
/**
|
|
230
|
+
* Returns `theme.motion.duration.<key>` zeroed out when the theme has
|
|
231
|
+
* `motion.reducedMotion === "true"`. Use this in any component that
|
|
232
|
+
* starts an animation so `prefers-reduced-motion` is honoured uniformly.
|
|
233
|
+
*/
|
|
234
|
+
export function motionDuration(
|
|
235
|
+
theme: Theme,
|
|
236
|
+
key: keyof Theme["motion"]["duration"],
|
|
237
|
+
): string {
|
|
238
|
+
if (theme.motion.reducedMotion === "true") return "0ms";
|
|
239
|
+
return theme.motion.duration[key];
|
|
240
|
+
}
|