@dreamboard-games/sdk 0.2.0
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.md +96 -0
- package/README.md +12 -0
- package/dist/HandView-ncJIVLhN.d.ts +193 -0
- package/dist/ResourceCounter-CTREyF73.d.ts +102 -0
- package/dist/ThemeProvider-fy0_QzgO.d.ts +99 -0
- package/dist/bundle-TIZcw8LB.d.ts +281 -0
- package/dist/cards-Sl3b40Mv.d.ts +13 -0
- package/dist/chunk-7YAHLYBR.js +481 -0
- package/dist/chunk-7YAHLYBR.js.map +1 -0
- package/dist/chunk-FDNZTDD6.js +8085 -0
- package/dist/chunk-FDNZTDD6.js.map +1 -0
- package/dist/chunk-GKKBPPSW.js +598 -0
- package/dist/chunk-GKKBPPSW.js.map +1 -0
- package/dist/chunk-I46YJSOD.js +1 -0
- package/dist/chunk-I46YJSOD.js.map +1 -0
- package/dist/chunk-KAELH4KC.js +104 -0
- package/dist/chunk-KAELH4KC.js.map +1 -0
- package/dist/chunk-PZ5AY32C.js +10 -0
- package/dist/chunk-PZ5AY32C.js.map +1 -0
- package/dist/chunk-T3ZKNUZ7.js +1 -0
- package/dist/chunk-T3ZKNUZ7.js.map +1 -0
- package/dist/chunk-T52J5RMF.js +1 -0
- package/dist/chunk-T52J5RMF.js.map +1 -0
- package/dist/chunk-TDSWKVZ4.js +5401 -0
- package/dist/chunk-TDSWKVZ4.js.map +1 -0
- package/dist/chunk-U5C6BONG.js +34 -0
- package/dist/chunk-U5C6BONG.js.map +1 -0
- package/dist/chunk-VDXOF4FW.js +69 -0
- package/dist/chunk-VDXOF4FW.js.map +1 -0
- package/dist/chunk-VFTAA4WO.js +115 -0
- package/dist/chunk-VFTAA4WO.js.map +1 -0
- package/dist/chunk-WN74KVNY.js +17 -0
- package/dist/chunk-WN74KVNY.js.map +1 -0
- package/dist/chunk-WYPQ3GG5.js +10990 -0
- package/dist/chunk-WYPQ3GG5.js.map +1 -0
- package/dist/components-D5ZRE2Hl.d.ts +1451 -0
- package/dist/generated/runtime/primitives.d.ts +12 -0
- package/dist/generated/runtime/primitives.js +180 -0
- package/dist/generated/runtime/primitives.js.map +1 -0
- package/dist/generated/runtime-api.d.ts +3 -0
- package/dist/generated/runtime-api.js +2 -0
- package/dist/generated/runtime-api.js.map +1 -0
- package/dist/generated/runtime.d.ts +14 -0
- package/dist/generated/runtime.js +18 -0
- package/dist/generated/runtime.js.map +1 -0
- package/dist/generated/workspace-contract.d.ts +14 -0
- package/dist/generated/workspace-contract.js +14 -0
- package/dist/generated/workspace-contract.js.map +1 -0
- package/dist/hex-board-view-D_07hO6O.d.ts +933 -0
- package/dist/hex-color-MhOyuY-o.d.ts +8 -0
- package/dist/index-BwqPQtBu.d.ts +1433 -0
- package/dist/index.d.ts +1 -0
- package/dist/index.js +12 -0
- package/dist/index.js.map +1 -0
- package/dist/infrastructure/reducer-bundle-abi.d.ts +1083 -0
- package/dist/infrastructure/reducer-bundle-abi.js +14 -0
- package/dist/infrastructure/reducer-bundle-abi.js.map +1 -0
- package/dist/infrastructure/workspace-codegen.d.ts +53 -0
- package/dist/infrastructure/workspace-codegen.js +44 -0
- package/dist/infrastructure/workspace-codegen.js.map +1 -0
- package/dist/manifest-contract-BNHVGFtU.d.ts +9 -0
- package/dist/package-set.d.ts +13 -0
- package/dist/package-set.js +12 -0
- package/dist/package-set.js.map +1 -0
- package/dist/primitive-props-DpKs-GCr.d.ts +11 -0
- package/dist/reducer.d.ts +3786 -0
- package/dist/reducer.js +8131 -0
- package/dist/reducer.js.map +1 -0
- package/dist/runtime/primitives.d.ts +226 -0
- package/dist/runtime/primitives.js +180 -0
- package/dist/runtime/primitives.js.map +1 -0
- package/dist/runtime/types/runtime-api.d.ts +1 -0
- package/dist/runtime/types/runtime-api.js +2 -0
- package/dist/runtime/types/runtime-api.js.map +1 -0
- package/dist/runtime/workspace-contract.d.ts +172 -0
- package/dist/runtime/workspace-contract.js +14 -0
- package/dist/runtime/workspace-contract.js.map +1 -0
- package/dist/runtime-api-3dshj6kK.d.ts +101 -0
- package/dist/runtime-api-DWxvTr-O.d.ts +379 -0
- package/dist/runtime.d.ts +58 -0
- package/dist/runtime.js +13 -0
- package/dist/runtime.js.map +1 -0
- package/dist/slots-1GPGihk8.d.ts +8 -0
- package/dist/testing.d.ts +149 -0
- package/dist/testing.js +513 -0
- package/dist/testing.js.map +1 -0
- package/dist/types.d.ts +496 -0
- package/dist/types.js +28 -0
- package/dist/types.js.map +1 -0
- package/dist/ui/components.d.ts +16 -0
- package/dist/ui/components.js +192 -0
- package/dist/ui/components.js.map +1 -0
- package/dist/ui/defaults.d.ts +19 -0
- package/dist/ui/defaults.js +104 -0
- package/dist/ui/defaults.js.map +1 -0
- package/dist/ui/plugin-styles.css +250 -0
- package/dist/ui/types/player-state.d.ts +365 -0
- package/dist/ui/types/player-state.js +1 -0
- package/dist/ui/types/player-state.js.map +1 -0
- package/dist/ui-contract-iQfTtUSL.d.ts +1161 -0
- package/dist/ui.d.ts +320 -0
- package/dist/ui.js +253 -0
- package/dist/ui.js.map +1 -0
- package/package.json +199 -0
- package/src/generated/reducer-contract/builders.ts +90 -0
- package/src/generated/reducer-contract/version.ts +9 -0
- package/src/generated/reducer-contract/wire.ts +100 -0
- package/src/generated/reducer-contract/zod.ts +101 -0
- package/src/generated/runtime/primitives.ts +2 -0
- package/src/generated/runtime-api.ts +5 -0
- package/src/generated/runtime.ts +35 -0
- package/src/generated/workspace-contract.ts +2 -0
- package/src/index.ts +7 -0
- package/src/infrastructure/reducer-bundle-abi.ts +8 -0
- package/src/infrastructure/reducer-contract/bundle.ts +37 -0
- package/src/infrastructure/workspace-codegen/hex-geometry.ts +69 -0
- package/src/infrastructure/workspace-codegen/index.ts +64 -0
- package/src/infrastructure/workspace-codegen/manifest-contract.ts +6632 -0
- package/src/infrastructure/workspace-codegen/manifest-validation.ts +795 -0
- package/src/infrastructure/workspace-codegen/ownership.ts +131 -0
- package/src/infrastructure/workspace-codegen/preset-card-sets.ts +169 -0
- package/src/infrastructure/workspace-codegen/seeds.ts +1705 -0
- package/src/infrastructure/workspace-codegen.ts +1 -0
- package/src/package-set.ts +19 -0
- package/src/reducer/authoring/contract.ts +157 -0
- package/src/reducer/authoring/effect.ts +224 -0
- package/src/reducer/authoring/game.ts +23 -0
- package/src/reducer/authoring/interaction.ts +98 -0
- package/src/reducer/authoring/phase.ts +300 -0
- package/src/reducer/authoring/types.ts +70 -0
- package/src/reducer/authoring/validation.ts +382 -0
- package/src/reducer/authoring/view-stage.ts +68 -0
- package/src/reducer/authoring.ts +29 -0
- package/src/reducer/bundle/ingress-bundle.ts +491 -0
- package/src/reducer/bundle/trusted/engine-instruction-resolver.ts +254 -0
- package/src/reducer/bundle/trusted/flow-instruction-resolver.ts +73 -0
- package/src/reducer/bundle/trusted/instruction-runner.ts +414 -0
- package/src/reducer/bundle/trusted/interaction-authorization.ts +137 -0
- package/src/reducer/bundle/trusted/interaction-collectors.ts +859 -0
- package/src/reducer/bundle/trusted/interaction-decision.ts +747 -0
- package/src/reducer/bundle/trusted/interaction-resolver.ts +95 -0
- package/src/reducer/bundle/trusted/interaction-types.ts +171 -0
- package/src/reducer/bundle/trusted/lifecycle-runner.ts +427 -0
- package/src/reducer/bundle/trusted/projection-builder.ts +356 -0
- package/src/reducer/bundle/trusted/projection-context.ts +39 -0
- package/src/reducer/bundle/trusted/rng-sampler.ts +150 -0
- package/src/reducer/bundle/trusted/runtime-registry.ts +120 -0
- package/src/reducer/bundle/trusted/runtime-scope.ts +336 -0
- package/src/reducer/bundle/trusted/simultaneous-player.ts +97 -0
- package/src/reducer/bundle/trusted/stage-resolver.ts +87 -0
- package/src/reducer/bundle/trusted/static-projection.ts +116 -0
- package/src/reducer/bundle/trusted/trusted-runtime-args.ts +97 -0
- package/src/reducer/bundle/trusted/trusted-runtime-result.ts +39 -0
- package/src/reducer/bundle/trusted/trusted-setup-profiles.ts +43 -0
- package/src/reducer/bundle/trusted/trusted-state-codec.ts +48 -0
- package/src/reducer/bundle/trusted-bundle.ts +97 -0
- package/src/reducer/bundle/types.ts +171 -0
- package/src/reducer/bundle.ts +2 -0
- package/src/reducer/client-param-schemas.ts +57 -0
- package/src/reducer/compose.ts +34 -0
- package/src/reducer/core/runtime-input.ts +30 -0
- package/src/reducer/core/runtime-instruction.ts +59 -0
- package/src/reducer/core/types.ts +62 -0
- package/src/reducer/definition-index.ts +277 -0
- package/src/reducer/derived.ts +106 -0
- package/src/reducer/effects.ts +92 -0
- package/src/reducer/engine/runtime-instruction-engine.ts +155 -0
- package/src/reducer/ingress/decode-runtime-input.ts +7 -0
- package/src/reducer/ingress/decode-session-state.ts +9 -0
- package/src/reducer/ingress/encode-session-state.ts +6 -0
- package/src/reducer/ingress/input-codec.ts +18 -0
- package/src/reducer/ingress/phase-schemas.ts +62 -0
- package/src/reducer/ingress/raw-types.ts +107 -0
- package/src/reducer/ingress/runtime-codec.ts +14 -0
- package/src/reducer/ingress/runtime-payload.ts +13 -0
- package/src/reducer/ingress/session-codec.ts +392 -0
- package/src/reducer/ingress/types.ts +6 -0
- package/src/reducer/inputs/boardInput.ts +217 -0
- package/src/reducer/inputs/boardTarget.ts +190 -0
- package/src/reducer/inputs/cardInput.ts +86 -0
- package/src/reducer/inputs/cardTarget.ts +101 -0
- package/src/reducer/inputs/choiceTarget.ts +104 -0
- package/src/reducer/inputs/defineInputs.ts +71 -0
- package/src/reducer/inputs/formInput.ts +809 -0
- package/src/reducer/inputs/many.ts +120 -0
- package/src/reducer/inputs/promptInput.ts +87 -0
- package/src/reducer/inputs/rngInput.ts +58 -0
- package/src/reducer/inputs/targetRule.ts +123 -0
- package/src/reducer/inputs.ts +41 -0
- package/src/reducer/model/definition.ts +1072 -0
- package/src/reducer/model/extract.ts +745 -0
- package/src/reducer/model/manifest.ts +570 -0
- package/src/reducer/model/queries.ts +641 -0
- package/src/reducer/model/runtime.ts +264 -0
- package/src/reducer/model/spec.ts +1386 -0
- package/src/reducer/model/table.ts +260 -0
- package/src/reducer/model.ts +7 -0
- package/src/reducer/ops.ts +1034 -0
- package/src/reducer/parse-utils.ts +28 -0
- package/src/reducer/per-player.ts +422 -0
- package/src/reducer/rng.ts +69 -0
- package/src/reducer/schema-helpers.ts +185 -0
- package/src/reducer/setup-bootstrap-helpers.ts +171 -0
- package/src/reducer/setup-bootstrap.ts +481 -0
- package/src/reducer/table-ops.ts +2671 -0
- package/src/reducer/table-queries.ts +372 -0
- package/src/reducer/transaction.ts +120 -0
- package/src/reducer.ts +314 -0
- package/src/runtime/primitives.ts +1 -0
- package/src/runtime/types/runtime-api.ts +1 -0
- package/src/runtime/workspace-contract.ts +32 -0
- package/src/runtime-internal/components/InteractionForm.tsx +1309 -0
- package/src/runtime-internal/components/PluginRuntime.tsx +103 -0
- package/src/runtime-internal/components/board/target-layer.ts +70 -0
- package/src/runtime-internal/context/ClientParamSchemaContext.tsx +44 -0
- package/src/runtime-internal/context/InteractionDraftContext.tsx +279 -0
- package/src/runtime-internal/context/PluginSessionContext.tsx +47 -0
- package/src/runtime-internal/context/PluginStateContext.tsx +262 -0
- package/src/runtime-internal/context/RuntimeContext.tsx +96 -0
- package/src/runtime-internal/defaults/components.tsx +409 -0
- package/src/runtime-internal/defaults/index.ts +11 -0
- package/src/runtime-internal/errors/ValidationError.ts +29 -0
- package/src/runtime-internal/hooks/useActivePlayers.ts +33 -0
- package/src/runtime-internal/hooks/useBoardInteractions.ts +665 -0
- package/src/runtime-internal/hooks/useGameSelector.ts +105 -0
- package/src/runtime-internal/hooks/useGameView.ts +9 -0
- package/src/runtime-internal/hooks/useInteractionByKey.ts +354 -0
- package/src/runtime-internal/hooks/useInteractionHandle.ts +438 -0
- package/src/runtime-internal/hooks/useIsMyTurn.ts +20 -0
- package/src/runtime-internal/hooks/useLobby.ts +76 -0
- package/src/runtime-internal/hooks/useMe.ts +48 -0
- package/src/runtime-internal/hooks/usePlayerInfo.ts +28 -0
- package/src/runtime-internal/hooks/usePlayerTurnOrder.ts +23 -0
- package/src/runtime-internal/hooks/usePluginRuntime.ts +147 -0
- package/src/runtime-internal/hooks/useSeatInbox.ts +61 -0
- package/src/runtime-internal/hooks/useSimultaneousPhase.ts +10 -0
- package/src/runtime-internal/index.ts +42 -0
- package/src/runtime-internal/internal.ts +43 -0
- package/src/runtime-internal/plugin-styles.css +250 -0
- package/src/runtime-internal/primitives/board.tsx +459 -0
- package/src/runtime-internal/primitives/dialog-lifecycle.ts +58 -0
- package/src/runtime-internal/primitives/dice.tsx +79 -0
- package/src/runtime-internal/primitives/game-ui-provider.tsx +35 -0
- package/src/runtime-internal/primitives/game.tsx +387 -0
- package/src/runtime-internal/primitives/hand-intent-adapter.ts +147 -0
- package/src/runtime-internal/primitives/hand-surface.tsx +594 -0
- package/src/runtime-internal/primitives/index.ts +196 -0
- package/src/runtime-internal/primitives/interaction-form-binding.tsx +56 -0
- package/src/runtime-internal/primitives/interaction-submit.ts +90 -0
- package/src/runtime-internal/primitives/interaction.tsx +987 -0
- package/src/runtime-internal/primitives/phase.tsx +43 -0
- package/src/runtime-internal/primitives/player-roster.tsx +302 -0
- package/src/runtime-internal/primitives/primitive-props.tsx +101 -0
- package/src/runtime-internal/primitives/prompt.tsx +255 -0
- package/src/runtime-internal/primitives/ui.tsx +60 -0
- package/src/runtime-internal/primitives/zone.tsx +791 -0
- package/src/runtime-internal/reducer.ts +30 -0
- package/src/runtime-internal/runtime/createPluginRuntimeAPI.ts +605 -0
- package/src/runtime-internal/types/plugin-state.ts +508 -0
- package/src/runtime-internal/types/reducer-state.ts +24 -0
- package/src/runtime-internal/types/runtime-api.ts +114 -0
- package/src/runtime-internal/ui-contract.ts +519 -0
- package/src/runtime-internal/utils/card-intent-adapter.ts +546 -0
- package/src/runtime-internal/utils/interaction-inputs.ts +492 -0
- package/src/runtime-internal/utils/interaction-labels.ts +23 -0
- package/src/runtime-internal/utils/interaction-router.ts +273 -0
- package/src/runtime-internal/utils/interaction-status.ts +74 -0
- package/src/runtime-internal/workspace-contract.ts +1170 -0
- package/src/runtime.ts +34 -0
- package/src/testing/create-expect-api.ts +352 -0
- package/src/testing/create-test-runtime.ts +381 -0
- package/src/testing/definitions.ts +127 -0
- package/src/testing/index.ts +3 -0
- package/src/testing.ts +1 -0
- package/src/type-stubs/manifest-contract.d.ts +42 -0
- package/src/type-stubs/manifest-contract.js +72 -0
- package/src/type-stubs/ui-contract.d.ts +5 -0
- package/src/type-stubs/ui-contract.js +1 -0
- package/src/types/authoring-card-properties.type-test.ts +266 -0
- package/src/types/authoring.ts +1282 -0
- package/src/types/cards.ts +19 -0
- package/src/types/contracts.ts +1550 -0
- package/src/types/generated-helpers.ts +35 -0
- package/src/types/index.ts +147 -0
- package/src/types/slots.ts +11 -0
- package/src/types.ts +1 -0
- package/src/ui/components/ActionButton.tsx +97 -0
- package/src/ui/components/ActionPanel.tsx +315 -0
- package/src/ui/components/Card.tsx +378 -0
- package/src/ui/components/CardDragSurface.tsx +1076 -0
- package/src/ui/components/ChromeSuppressionContext.tsx +70 -0
- package/src/ui/components/CostDisplay.tsx +145 -0
- package/src/ui/components/DiceRoller.tsx +581 -0
- package/src/ui/components/Drawer.tsx +180 -0
- package/src/ui/components/ErrorBoundary.tsx +275 -0
- package/src/ui/components/GameEndDisplay.tsx +398 -0
- package/src/ui/components/GameSkeleton.tsx +260 -0
- package/src/ui/components/Hand.tsx +468 -0
- package/src/ui/components/HandDock.tsx +299 -0
- package/src/ui/components/HandView.tsx +441 -0
- package/src/ui/components/MobileHandTray.tsx +381 -0
- package/src/ui/components/MoreActions.tsx +143 -0
- package/src/ui/components/PhaseIndicator.tsx +341 -0
- package/src/ui/components/PlayArea.tsx +146 -0
- package/src/ui/components/PrimaryActionButton.tsx +336 -0
- package/src/ui/components/PrimaryButton.tsx +45 -0
- package/src/ui/components/ResourceCounter.tsx +270 -0
- package/src/ui/components/StagingZone.tsx +134 -0
- package/src/ui/components/ThemedButton.tsx +113 -0
- package/src/ui/components/Toast.tsx +264 -0
- package/src/ui/components/board/HexGrid.tsx +1294 -0
- package/src/ui/components/board/NetworkGraph.tsx +476 -0
- package/src/ui/components/board/SlotSystem.tsx +388 -0
- package/src/ui/components/board/SquareGrid.tsx +1165 -0
- package/src/ui/components/board/TrackBoard.tsx +496 -0
- package/src/ui/components/board/ZoneMap.tsx +448 -0
- package/src/ui/components/board/hex-board-view.ts +123 -0
- package/src/ui/components/board/index.ts +142 -0
- package/src/ui/components/board/interaction-accessibility.ts +21 -0
- package/src/ui/components/board/target-layer.ts +66 -0
- package/src/ui/components/card-render-content.type-test.ts +27 -0
- package/src/ui/components/hand-layout-math.ts +163 -0
- package/src/ui/components/hand-pointer-engine.ts +413 -0
- package/src/ui/components/index.ts +245 -0
- package/src/ui/components.ts +1 -0
- package/src/ui/defaults/components.tsx +106 -0
- package/src/ui/defaults/index.ts +8 -0
- package/src/ui/defaults.ts +1 -0
- package/src/ui/errors/ValidationError.ts +29 -0
- package/src/ui/helpers/cards.ts +19 -0
- package/src/ui/helpers/track-board.ts +211 -0
- package/src/ui/hooks/useBoardTopology.ts +316 -0
- package/src/ui/hooks/useCards.ts +10 -0
- package/src/ui/hooks/useHandCardPointer.ts +381 -0
- package/src/ui/hooks/useHandLayout.ts +378 -0
- package/src/ui/hooks/useHandPresentation.ts +121 -0
- package/src/ui/hooks/useHexBoard.ts +74 -0
- package/src/ui/hooks/useHexGrid.ts +185 -0
- package/src/ui/hooks/useIsMobile.ts +35 -0
- package/src/ui/hooks/usePanZoom.ts +278 -0
- package/src/ui/hooks/useSquareBoard.ts +124 -0
- package/src/ui/hooks/useSquareGrid.ts +328 -0
- package/src/ui/index.ts +98 -0
- package/src/ui/internal/ui/alert.tsx +51 -0
- package/src/ui/internal/ui/button.tsx +58 -0
- package/src/ui/internal/ui/dialog.tsx +134 -0
- package/src/ui/internal/ui/input.tsx +21 -0
- package/src/ui/internal/ui/label.tsx +21 -0
- package/src/ui/internal/ui/select.tsx +129 -0
- package/src/ui/internal/ui/tooltip.tsx +54 -0
- package/src/ui/internal/ui/utils.ts +5 -0
- package/src/ui/plugin-styles.css +250 -0
- package/src/ui/primitives/dialog-lifecycle.ts +58 -0
- package/src/ui/primitives/dice.tsx +79 -0
- package/src/ui/primitives/primitive-props.tsx +101 -0
- package/src/ui/theme/ThemeProvider.tsx +252 -0
- package/src/ui/theme/board.ts +61 -0
- package/src/ui/theme/css-vars.ts +105 -0
- package/src/ui/theme/derive.ts +240 -0
- package/src/ui/theme/index.ts +61 -0
- package/src/ui/theme/presets/arcade.ts +261 -0
- package/src/ui/theme/presets/studio.ts +261 -0
- package/src/ui/theme/presets/tabletop.ts +266 -0
- package/src/ui/theme/tokens.ts +392 -0
- package/src/ui/types/hex-color.ts +20 -0
- package/src/ui/types/player-state.ts +463 -0
- package/src/ui/types/tiled-board.ts +785 -0
- package/src/ui/types/visual-state.ts +137 -0
- package/src/ui.ts +1 -0
|
@@ -0,0 +1,1170 @@
|
|
|
1
|
+
import type { ButtonHTMLAttributes, ReactElement, ReactNode } from "react";
|
|
2
|
+
import { Fragment as ReactFragment, createElement, useMemo } from "react";
|
|
3
|
+
import { clsx } from "clsx";
|
|
4
|
+
import {
|
|
5
|
+
ClientParamSchemaProvider,
|
|
6
|
+
type ClientParamSchemaMap,
|
|
7
|
+
} from "./context/ClientParamSchemaContext.js";
|
|
8
|
+
import { usePluginState } from "./context/PluginStateContext.js";
|
|
9
|
+
import {
|
|
10
|
+
createResourceCounter,
|
|
11
|
+
type ResourceCounterComponents,
|
|
12
|
+
type ResourceDisplayConfig,
|
|
13
|
+
} from "../ui.js";
|
|
14
|
+
import { CardFace, type ViewCard } from "../ui.js";
|
|
15
|
+
import {
|
|
16
|
+
MobileHandTrayProvider,
|
|
17
|
+
useRegisterMobileHand,
|
|
18
|
+
type HandRole,
|
|
19
|
+
} from "../ui/components.js";
|
|
20
|
+
import { ToastProvider, useIsMobile } from "../ui.js";
|
|
21
|
+
import {
|
|
22
|
+
createDreamboardUI,
|
|
23
|
+
type DreamboardUI,
|
|
24
|
+
type TypedGame,
|
|
25
|
+
type UIContract,
|
|
26
|
+
} from "./ui-contract.js";
|
|
27
|
+
import {
|
|
28
|
+
useResolvedCardTargetValue,
|
|
29
|
+
useZoneCards,
|
|
30
|
+
} from "./primitives/index.js";
|
|
31
|
+
import {
|
|
32
|
+
HandSurfaceView,
|
|
33
|
+
HandStagingView,
|
|
34
|
+
dropTargetIdFor,
|
|
35
|
+
type HandSelectionSummary,
|
|
36
|
+
type HandSurfaceViewProps,
|
|
37
|
+
type RuntimeDropTarget,
|
|
38
|
+
} from "./primitives/hand-surface.js";
|
|
39
|
+
import type { AuthoredCardIntent } from "./primitives/hand-intent-adapter.js";
|
|
40
|
+
import type {
|
|
41
|
+
CardDropTargetVisualState,
|
|
42
|
+
HandInteractionPolicy,
|
|
43
|
+
HandLayoutKind,
|
|
44
|
+
HandLayoutPolicy,
|
|
45
|
+
InteractionVisualState,
|
|
46
|
+
} from "../ui.js";
|
|
47
|
+
import type {
|
|
48
|
+
BoardHexGridProps,
|
|
49
|
+
BoardHexViewProps,
|
|
50
|
+
InteractionDialogProps,
|
|
51
|
+
InteractionFormPrimitiveProps,
|
|
52
|
+
InteractionCardInputRenderState,
|
|
53
|
+
InteractionStateProps,
|
|
54
|
+
InteractionSubmitProps,
|
|
55
|
+
InteractionTriggerProps,
|
|
56
|
+
GameMeState,
|
|
57
|
+
GamePlayersState,
|
|
58
|
+
GameRenderState,
|
|
59
|
+
GameTurnState,
|
|
60
|
+
UIRootProps,
|
|
61
|
+
ZoneCardRenderItem,
|
|
62
|
+
} from "./primitives/index.js";
|
|
63
|
+
import type { BoardSpaceTargetProps } from "./primitives/board.js";
|
|
64
|
+
import type { ZoneListProps } from "./primitives/zone.js";
|
|
65
|
+
import type { PluginStateSnapshot } from "./types/plugin-state.js";
|
|
66
|
+
import { isInteractionAvailable } from "./utils/interaction-status.js";
|
|
67
|
+
|
|
68
|
+
export type { BoardSpaceTargetProps } from "./primitives/board.js";
|
|
69
|
+
export type { HandRole } from "../ui/components.js";
|
|
70
|
+
|
|
71
|
+
export type WorkspaceInteractionSlotComponent<Props = object> = (
|
|
72
|
+
props: Props extends { children: unknown }
|
|
73
|
+
? Props
|
|
74
|
+
: Props & { children?: ReactNode },
|
|
75
|
+
) => ReactElement | null;
|
|
76
|
+
|
|
77
|
+
export interface WorkspaceFormInputSlot<Input extends string = string> {
|
|
78
|
+
readonly Field: WorkspaceInteractionSlotComponent;
|
|
79
|
+
readonly Options: WorkspaceInteractionSlotComponent<{
|
|
80
|
+
children?: (option: { value: unknown; label: string }) => ReactNode;
|
|
81
|
+
}>;
|
|
82
|
+
readonly Value: WorkspaceInteractionSlotComponent<{
|
|
83
|
+
children: (value: unknown | undefined) => ReactNode;
|
|
84
|
+
}>;
|
|
85
|
+
readonly Default: WorkspaceInteractionSlotComponent;
|
|
86
|
+
readonly __input?: Input;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
export interface WorkspaceCardInputSlot<Card extends string = string> {
|
|
90
|
+
readonly Card: WorkspaceInteractionSlotComponent<
|
|
91
|
+
{ value: Card } & Omit<
|
|
92
|
+
ButtonHTMLAttributes<HTMLButtonElement>,
|
|
93
|
+
| "children"
|
|
94
|
+
| "disabled"
|
|
95
|
+
| "aria-disabled"
|
|
96
|
+
| "aria-pressed"
|
|
97
|
+
| "onClick"
|
|
98
|
+
| "type"
|
|
99
|
+
| "value"
|
|
100
|
+
>
|
|
101
|
+
>;
|
|
102
|
+
readonly Cards: WorkspaceInteractionSlotComponent<{
|
|
103
|
+
children: (card: { id: Card }) => ReactNode;
|
|
104
|
+
}>;
|
|
105
|
+
readonly Value: WorkspaceInteractionSlotComponent<{
|
|
106
|
+
children: (value: unknown | undefined) => ReactNode;
|
|
107
|
+
}>;
|
|
108
|
+
readonly Default: WorkspaceInteractionSlotComponent;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
export interface WorkspaceBoardTargetInputSlot<
|
|
112
|
+
Kind extends "space" | "edge" | "vertex" | "tile",
|
|
113
|
+
Target extends string = string,
|
|
114
|
+
> {
|
|
115
|
+
readonly Target: WorkspaceInteractionSlotComponent<
|
|
116
|
+
{ value: Target } & Omit<
|
|
117
|
+
ButtonHTMLAttributes<HTMLButtonElement>,
|
|
118
|
+
| "children"
|
|
119
|
+
| "disabled"
|
|
120
|
+
| "aria-disabled"
|
|
121
|
+
| "aria-pressed"
|
|
122
|
+
| "onClick"
|
|
123
|
+
| "type"
|
|
124
|
+
| "value"
|
|
125
|
+
>
|
|
126
|
+
>;
|
|
127
|
+
readonly Value: WorkspaceInteractionSlotComponent<{
|
|
128
|
+
children: (value: unknown | undefined) => ReactNode;
|
|
129
|
+
}>;
|
|
130
|
+
readonly Default: WorkspaceInteractionSlotComponent;
|
|
131
|
+
readonly __kind?: Kind;
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
export interface WorkspaceBoardSurface<
|
|
135
|
+
Space extends string = string,
|
|
136
|
+
Edge extends string = string,
|
|
137
|
+
Vertex extends string = string,
|
|
138
|
+
Tile extends string = string,
|
|
139
|
+
> {
|
|
140
|
+
readonly Root: WorkspaceInteractionSlotComponent;
|
|
141
|
+
readonly Space: <Target extends Space>(
|
|
142
|
+
props: BoardSpaceTargetProps<Target>,
|
|
143
|
+
) => ReactElement | null;
|
|
144
|
+
readonly slot: {
|
|
145
|
+
readonly space: WorkspaceBoardTargetInputSlot<"space", Space>;
|
|
146
|
+
readonly playerSpace: WorkspaceBoardTargetInputSlot<"space", Space>;
|
|
147
|
+
readonly edge: WorkspaceBoardTargetInputSlot<"edge", Edge>;
|
|
148
|
+
readonly vertex: WorkspaceBoardTargetInputSlot<"vertex", Vertex>;
|
|
149
|
+
readonly tile: WorkspaceBoardTargetInputSlot<"tile", Tile>;
|
|
150
|
+
};
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
export type WorkspaceZoneCardsComponent<Card> =
|
|
154
|
+
WorkspaceInteractionSlotComponent<
|
|
155
|
+
Omit<ZoneListProps, "children" | "empty"> & {
|
|
156
|
+
empty?: ReactNode;
|
|
157
|
+
children: (card: Card) => ReactNode;
|
|
158
|
+
}
|
|
159
|
+
>;
|
|
160
|
+
|
|
161
|
+
export type WorkspaceZoneCardComponent<Card> =
|
|
162
|
+
WorkspaceInteractionSlotComponent<
|
|
163
|
+
Omit<ButtonHTMLAttributes<HTMLButtonElement>, "type" | "value"> & {
|
|
164
|
+
card: Card;
|
|
165
|
+
}
|
|
166
|
+
>;
|
|
167
|
+
|
|
168
|
+
export type WorkspaceZoneStagingComponent<Card> =
|
|
169
|
+
WorkspaceInteractionSlotComponent<{
|
|
170
|
+
children: (card: Card) => ReactNode;
|
|
171
|
+
label?: ReactNode;
|
|
172
|
+
renderEmptySlot?: (index: number) => ReactNode;
|
|
173
|
+
cardSize?: "sm" | "md" | "lg";
|
|
174
|
+
ariaLabel?: string;
|
|
175
|
+
className?: string;
|
|
176
|
+
}>;
|
|
177
|
+
|
|
178
|
+
export interface WorkspaceHandSurface<Zone extends string, Card> {
|
|
179
|
+
readonly Hand: WorkspaceZoneCardsComponent<Card>;
|
|
180
|
+
readonly Card: WorkspaceZoneCardComponent<Card>;
|
|
181
|
+
readonly Staging: WorkspaceZoneStagingComponent<Card>;
|
|
182
|
+
readonly slot: {
|
|
183
|
+
readonly card: WorkspaceCardInputSlot<Zone>;
|
|
184
|
+
};
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
export interface WorkspaceHandOptions<Zone extends string = string> {
|
|
188
|
+
zone: Zone;
|
|
189
|
+
role: HandRole;
|
|
190
|
+
label: string;
|
|
191
|
+
order?: number;
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
interface WorkspaceHandComponentOptions {
|
|
195
|
+
name: string;
|
|
196
|
+
zone: string;
|
|
197
|
+
role: HandRole;
|
|
198
|
+
label: string;
|
|
199
|
+
order?: number;
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
export interface WorkspacePileSurface<Card> {
|
|
203
|
+
readonly Pile: WorkspaceZoneCardsComponent<Card>;
|
|
204
|
+
readonly Card: WorkspaceZoneCardComponent<Card>;
|
|
205
|
+
}
|
|
206
|
+
|
|
207
|
+
export interface WorkspaceCardCollectionSurface<Zone extends string, Card> {
|
|
208
|
+
readonly Collection: WorkspaceZoneCardsComponent<Card>;
|
|
209
|
+
readonly Card: WorkspaceZoneCardComponent<Card>;
|
|
210
|
+
readonly slot: {
|
|
211
|
+
readonly card: WorkspaceCardInputSlot<Zone>;
|
|
212
|
+
};
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
export interface WorkspaceBoardSurfaceDescriptor<
|
|
216
|
+
Board extends string = string,
|
|
217
|
+
> {
|
|
218
|
+
readonly kind: "board";
|
|
219
|
+
readonly board: Board;
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
export interface WorkspaceHandSurfaceDescriptor<Zone extends string = string> {
|
|
223
|
+
readonly kind: "hand";
|
|
224
|
+
readonly zone: Zone;
|
|
225
|
+
readonly role: HandRole;
|
|
226
|
+
readonly label: string;
|
|
227
|
+
readonly order?: number;
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
export interface WorkspacePileSurfaceDescriptor<Zone extends string = string> {
|
|
231
|
+
readonly kind: "pile";
|
|
232
|
+
readonly zone: Zone;
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
export interface WorkspacePilesSurfaceDescriptor<
|
|
236
|
+
Zones extends readonly string[] = readonly string[],
|
|
237
|
+
> {
|
|
238
|
+
readonly kind: "piles";
|
|
239
|
+
readonly zones: Zones;
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
export interface WorkspaceCardCollectionSurfaceDescriptor<
|
|
243
|
+
Zones extends readonly string[] = readonly string[],
|
|
244
|
+
> {
|
|
245
|
+
readonly kind: "cardCollection";
|
|
246
|
+
readonly zones: Zones;
|
|
247
|
+
readonly mode?: "all" | "top-card";
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
export interface WorkspaceInteractionFormsDescriptor<
|
|
251
|
+
Interactions extends Readonly<Record<string, string>> = Readonly<
|
|
252
|
+
Record<string, string>
|
|
253
|
+
>,
|
|
254
|
+
> {
|
|
255
|
+
readonly kind: "forms";
|
|
256
|
+
readonly interactions: Interactions;
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
export interface WorkspaceInteractionFormDescriptor<
|
|
260
|
+
Interaction extends string = string,
|
|
261
|
+
> {
|
|
262
|
+
readonly kind: "form";
|
|
263
|
+
readonly interaction: Interaction;
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
export type WorkspaceSurfaceDescriptor =
|
|
267
|
+
| WorkspaceBoardSurfaceDescriptor
|
|
268
|
+
| WorkspaceHandSurfaceDescriptor
|
|
269
|
+
| WorkspacePileSurfaceDescriptor
|
|
270
|
+
| WorkspacePilesSurfaceDescriptor
|
|
271
|
+
| WorkspaceCardCollectionSurfaceDescriptor
|
|
272
|
+
| WorkspaceInteractionFormDescriptor
|
|
273
|
+
| WorkspaceInteractionFormsDescriptor;
|
|
274
|
+
|
|
275
|
+
export interface WorkspaceSurfaceSpec {
|
|
276
|
+
readonly [key: string]: WorkspaceSurfaceDescriptor | WorkspaceSurfaceSpec;
|
|
277
|
+
}
|
|
278
|
+
|
|
279
|
+
type RuntimeInteraction = DreamboardUI["Interaction"] & {
|
|
280
|
+
Root(props: {
|
|
281
|
+
interaction: string;
|
|
282
|
+
children?: ReactNode;
|
|
283
|
+
unavailable?: "render" | "hide";
|
|
284
|
+
}): ReactElement | null;
|
|
285
|
+
CardInput(
|
|
286
|
+
props: Omit<
|
|
287
|
+
ButtonHTMLAttributes<HTMLButtonElement>,
|
|
288
|
+
"type" | "value" | "children"
|
|
289
|
+
> & {
|
|
290
|
+
input: string;
|
|
291
|
+
unsafeCardId?: string;
|
|
292
|
+
children?:
|
|
293
|
+
| ReactNode
|
|
294
|
+
| ((state: InteractionCardInputRenderState) => ReactNode);
|
|
295
|
+
},
|
|
296
|
+
): ReactElement | null;
|
|
297
|
+
Routes(props: {
|
|
298
|
+
routes: Record<string, { collect: Record<string, unknown> }>;
|
|
299
|
+
fallback?: ReactNode;
|
|
300
|
+
includeUnavailable?: boolean | null;
|
|
301
|
+
}): ReactElement;
|
|
302
|
+
};
|
|
303
|
+
|
|
304
|
+
type RuntimeBoard = DreamboardUI["Board"] & {
|
|
305
|
+
SpaceTarget(props: BoardSpaceTargetProps<string>): ReactElement | null;
|
|
306
|
+
EdgeTarget(
|
|
307
|
+
props: { value: string; children?: ReactNode } & Omit<
|
|
308
|
+
ButtonHTMLAttributes<HTMLButtonElement>,
|
|
309
|
+
| "type"
|
|
310
|
+
| "value"
|
|
311
|
+
| "children"
|
|
312
|
+
| "disabled"
|
|
313
|
+
| "aria-disabled"
|
|
314
|
+
| "aria-pressed"
|
|
315
|
+
| "onClick"
|
|
316
|
+
| "onSelect"
|
|
317
|
+
| "onSelectError"
|
|
318
|
+
>,
|
|
319
|
+
): ReactElement | null;
|
|
320
|
+
VertexTarget(
|
|
321
|
+
props: { value: string; children?: ReactNode } & Omit<
|
|
322
|
+
ButtonHTMLAttributes<HTMLButtonElement>,
|
|
323
|
+
| "type"
|
|
324
|
+
| "value"
|
|
325
|
+
| "children"
|
|
326
|
+
| "disabled"
|
|
327
|
+
| "aria-disabled"
|
|
328
|
+
| "aria-pressed"
|
|
329
|
+
| "onClick"
|
|
330
|
+
| "onSelect"
|
|
331
|
+
| "onSelectError"
|
|
332
|
+
>,
|
|
333
|
+
): ReactElement | null;
|
|
334
|
+
};
|
|
335
|
+
|
|
336
|
+
type RuntimeZone<Card> = DreamboardUI["Zone"] & {
|
|
337
|
+
List(
|
|
338
|
+
props: Omit<ZoneListProps, "children" | "empty"> & {
|
|
339
|
+
empty?: ReactNode;
|
|
340
|
+
children: (card: Card) => ReactNode;
|
|
341
|
+
},
|
|
342
|
+
): ReactElement | null;
|
|
343
|
+
};
|
|
344
|
+
|
|
345
|
+
const DEFAULT_ZONE_CARD_CLASS =
|
|
346
|
+
"group relative border-0 bg-transparent p-0 transition-transform enabled:cursor-pointer enabled:hover:-translate-y-2 data-[selected=true]:-translate-y-3 disabled:cursor-not-allowed data-[eligible=false]:opacity-45 data-[eligible=false]:grayscale data-[eligible=false]:hover:translate-y-0 data-[card-available=false]:opacity-45 data-[card-available=false]:grayscale";
|
|
347
|
+
|
|
348
|
+
function cardRenderItemToViewCard(card: unknown, cardId: string): ViewCard {
|
|
349
|
+
if (
|
|
350
|
+
card &&
|
|
351
|
+
typeof card === "object" &&
|
|
352
|
+
"hidden" in card &&
|
|
353
|
+
(card as { hidden?: unknown }).hidden === false
|
|
354
|
+
) {
|
|
355
|
+
return card as unknown as ViewCard;
|
|
356
|
+
}
|
|
357
|
+
return {
|
|
358
|
+
id: cardId,
|
|
359
|
+
cardType: "hidden",
|
|
360
|
+
name: "Hidden card",
|
|
361
|
+
properties: {},
|
|
362
|
+
};
|
|
363
|
+
}
|
|
364
|
+
|
|
365
|
+
export interface WorkspaceContractOptions<
|
|
366
|
+
Contract extends UIContract,
|
|
367
|
+
Resource extends string,
|
|
368
|
+
Card,
|
|
369
|
+
HexBoards extends Record<string, unknown>,
|
|
370
|
+
> {
|
|
371
|
+
readonly uiContract: Contract;
|
|
372
|
+
readonly clientParamSchemasByPhase?: ClientParamSchemaMap;
|
|
373
|
+
readonly formInputKeysForInteraction: (
|
|
374
|
+
interaction: string,
|
|
375
|
+
) => ReadonlySet<string>;
|
|
376
|
+
readonly resourceIds: readonly Resource[];
|
|
377
|
+
readonly resourcePresentationById?: Partial<
|
|
378
|
+
Record<string, { label?: string; icon?: string }>
|
|
379
|
+
>;
|
|
380
|
+
readonly hexStaticBoards: HexBoards;
|
|
381
|
+
readonly cardIdFromZoneCard: (card: Card) => string;
|
|
382
|
+
readonly zoneIdFromZoneCard: (card: Card) => string;
|
|
383
|
+
}
|
|
384
|
+
|
|
385
|
+
/**
|
|
386
|
+
* Render-prop body for the card surface `slot.card.Value`. Surfaces the live
|
|
387
|
+
* draft value for the active interaction's card-target input — the selected
|
|
388
|
+
* card-id array for `selection: "many"` collectors, or the single id for
|
|
389
|
+
* `selection: "one"`. Renders nothing meaningful (`undefined`) outside an
|
|
390
|
+
* `<Interaction.Root>`.
|
|
391
|
+
*/
|
|
392
|
+
function CardSlotValue({
|
|
393
|
+
children,
|
|
394
|
+
}: {
|
|
395
|
+
children: (value: unknown | undefined) => ReactNode;
|
|
396
|
+
}): ReactElement {
|
|
397
|
+
const value = useResolvedCardTargetValue();
|
|
398
|
+
return createElement(ReactFragment, null, children(value));
|
|
399
|
+
}
|
|
400
|
+
|
|
401
|
+
export function createWorkspaceUIContract<
|
|
402
|
+
WorkspaceUI,
|
|
403
|
+
Contract extends UIContract,
|
|
404
|
+
Resource extends string,
|
|
405
|
+
Card,
|
|
406
|
+
HexBoards extends Record<string, unknown>,
|
|
407
|
+
>(
|
|
408
|
+
options: WorkspaceContractOptions<Contract, Resource, Card, HexBoards>,
|
|
409
|
+
): WorkspaceUI {
|
|
410
|
+
const baseUI = createDreamboardUI(options.uiContract);
|
|
411
|
+
const runtimeInteraction = baseUI.Interaction as RuntimeInteraction;
|
|
412
|
+
const runtimeBoard = baseUI.Board as RuntimeBoard;
|
|
413
|
+
const runtimeZone = baseUI.Zone as RuntimeZone<Card>;
|
|
414
|
+
|
|
415
|
+
const resourceDisplayConfig = options.resourceIds.map((resource) => {
|
|
416
|
+
const presentation = options.resourcePresentationById?.[resource];
|
|
417
|
+
return {
|
|
418
|
+
type: resource,
|
|
419
|
+
label: presentation?.label ?? resource,
|
|
420
|
+
icon: presentation?.icon ?? resource,
|
|
421
|
+
};
|
|
422
|
+
}) satisfies ReadonlyArray<ResourceDisplayConfig<Resource>>;
|
|
423
|
+
const resourceCounter = createResourceCounter<Resource>(
|
|
424
|
+
resourceDisplayConfig,
|
|
425
|
+
);
|
|
426
|
+
|
|
427
|
+
function InteractionRoutes({
|
|
428
|
+
routes,
|
|
429
|
+
fallback,
|
|
430
|
+
includeUnavailable,
|
|
431
|
+
}: {
|
|
432
|
+
routes: Record<
|
|
433
|
+
string,
|
|
434
|
+
{
|
|
435
|
+
collect: Record<string, unknown>;
|
|
436
|
+
}
|
|
437
|
+
>;
|
|
438
|
+
fallback?: ReactNode;
|
|
439
|
+
includeUnavailable?: boolean | null;
|
|
440
|
+
}): ReactElement {
|
|
441
|
+
return createElement(runtimeInteraction.Routes, {
|
|
442
|
+
routes,
|
|
443
|
+
fallback,
|
|
444
|
+
includeUnavailable,
|
|
445
|
+
});
|
|
446
|
+
}
|
|
447
|
+
|
|
448
|
+
const withInteractionRoot = (interaction: string, children: ReactNode) =>
|
|
449
|
+
createElement(runtimeInteraction.Root, {
|
|
450
|
+
interaction,
|
|
451
|
+
children,
|
|
452
|
+
});
|
|
453
|
+
|
|
454
|
+
function createFormInputSlot(
|
|
455
|
+
input: string,
|
|
456
|
+
interaction?: string,
|
|
457
|
+
): WorkspaceFormInputSlot {
|
|
458
|
+
return {
|
|
459
|
+
Field: (props: { children?: ReactNode }) => {
|
|
460
|
+
const field = createElement(baseUI.Interaction.Field, {
|
|
461
|
+
...props,
|
|
462
|
+
input: input as never,
|
|
463
|
+
});
|
|
464
|
+
return interaction ? withInteractionRoot(interaction, field) : field;
|
|
465
|
+
},
|
|
466
|
+
Options: () => null,
|
|
467
|
+
Value: ({
|
|
468
|
+
children,
|
|
469
|
+
}: {
|
|
470
|
+
children: (value: unknown | undefined) => ReactNode;
|
|
471
|
+
}) => createElement(ReactFragment, null, children(undefined)),
|
|
472
|
+
Default: ({ children }: { children?: ReactNode }) =>
|
|
473
|
+
createElement(ReactFragment, null, children),
|
|
474
|
+
};
|
|
475
|
+
}
|
|
476
|
+
|
|
477
|
+
function createCardInputSlot(): WorkspaceCardInputSlot {
|
|
478
|
+
return {
|
|
479
|
+
Card: ({ value, ...props }: { value: string; children?: ReactNode }) =>
|
|
480
|
+
createElement(runtimeInteraction.CardInput, {
|
|
481
|
+
...props,
|
|
482
|
+
input: "cardId",
|
|
483
|
+
unsafeCardId: value,
|
|
484
|
+
}),
|
|
485
|
+
Cards: () => null,
|
|
486
|
+
Value: ({
|
|
487
|
+
children,
|
|
488
|
+
}: {
|
|
489
|
+
children: (value: unknown | undefined) => ReactNode;
|
|
490
|
+
}) => createElement(CardSlotValue, { children }),
|
|
491
|
+
Default: ({ children }: { children?: ReactNode }) =>
|
|
492
|
+
createElement(ReactFragment, null, children),
|
|
493
|
+
};
|
|
494
|
+
}
|
|
495
|
+
|
|
496
|
+
function createBoardTargetInputSlot(
|
|
497
|
+
kind: "space" | "edge" | "vertex" | "tile",
|
|
498
|
+
): WorkspaceBoardTargetInputSlot<typeof kind> {
|
|
499
|
+
const Target = ({
|
|
500
|
+
value,
|
|
501
|
+
...props
|
|
502
|
+
}: {
|
|
503
|
+
value: string;
|
|
504
|
+
children?: ReactNode;
|
|
505
|
+
}) => {
|
|
506
|
+
if (kind === "edge") {
|
|
507
|
+
return createElement(runtimeBoard.EdgeTarget, { ...props, value });
|
|
508
|
+
}
|
|
509
|
+
if (kind === "vertex") {
|
|
510
|
+
return createElement(runtimeBoard.VertexTarget, { ...props, value });
|
|
511
|
+
}
|
|
512
|
+
return createElement(runtimeBoard.SpaceTarget, { ...props, value });
|
|
513
|
+
};
|
|
514
|
+
return {
|
|
515
|
+
Target,
|
|
516
|
+
Value: ({
|
|
517
|
+
children,
|
|
518
|
+
}: {
|
|
519
|
+
children: (value: unknown | undefined) => ReactNode;
|
|
520
|
+
}) => createElement(ReactFragment, null, children(undefined)),
|
|
521
|
+
Default: ({ children }: { children?: ReactNode }) =>
|
|
522
|
+
createElement(ReactFragment, null, children),
|
|
523
|
+
};
|
|
524
|
+
}
|
|
525
|
+
|
|
526
|
+
function useInteractionFormSurface(interaction: string) {
|
|
527
|
+
const validInputs = options.formInputKeysForInteraction(interaction);
|
|
528
|
+
const slot = Object.fromEntries(
|
|
529
|
+
[...validInputs].map((input) => [
|
|
530
|
+
input,
|
|
531
|
+
createFormInputSlot(input, interaction),
|
|
532
|
+
]),
|
|
533
|
+
);
|
|
534
|
+
return {
|
|
535
|
+
Root: ({ children }: { children?: ReactNode }) =>
|
|
536
|
+
withInteractionRoot(interaction, children),
|
|
537
|
+
Form: (props: InteractionFormPrimitiveProps) =>
|
|
538
|
+
withInteractionRoot(
|
|
539
|
+
interaction,
|
|
540
|
+
createElement(baseUI.Interaction.Form, props),
|
|
541
|
+
),
|
|
542
|
+
Dialog: (props: InteractionDialogProps) =>
|
|
543
|
+
withInteractionRoot(
|
|
544
|
+
interaction,
|
|
545
|
+
createElement(baseUI.Interaction.Dialog, props),
|
|
546
|
+
),
|
|
547
|
+
State: (props: InteractionStateProps) =>
|
|
548
|
+
withInteractionRoot(
|
|
549
|
+
interaction,
|
|
550
|
+
createElement(baseUI.Interaction.State, props),
|
|
551
|
+
),
|
|
552
|
+
Arm: (props: InteractionTriggerProps) =>
|
|
553
|
+
withInteractionRoot(
|
|
554
|
+
interaction,
|
|
555
|
+
createElement(baseUI.Interaction.Trigger, props),
|
|
556
|
+
),
|
|
557
|
+
Submit: (props: InteractionSubmitProps) =>
|
|
558
|
+
withInteractionRoot(
|
|
559
|
+
interaction,
|
|
560
|
+
createElement(baseUI.Interaction.Submit, props),
|
|
561
|
+
),
|
|
562
|
+
Field: ({ input, ...props }: { input: string; children?: ReactNode }) =>
|
|
563
|
+
withInteractionRoot(
|
|
564
|
+
interaction,
|
|
565
|
+
createElement(baseUI.Interaction.Field, {
|
|
566
|
+
...props,
|
|
567
|
+
input: input as never,
|
|
568
|
+
}),
|
|
569
|
+
),
|
|
570
|
+
slot,
|
|
571
|
+
};
|
|
572
|
+
}
|
|
573
|
+
|
|
574
|
+
function useBoardSurface(_name: string) {
|
|
575
|
+
return {
|
|
576
|
+
Root: ({ children }: { children?: ReactNode }) =>
|
|
577
|
+
createElement(baseUI.Board.Root, { children }),
|
|
578
|
+
Space: (props: BoardSpaceTargetProps<string>) =>
|
|
579
|
+
createElement(runtimeBoard.SpaceTarget, props),
|
|
580
|
+
slot: {
|
|
581
|
+
space: createBoardTargetInputSlot("space"),
|
|
582
|
+
playerSpace: createBoardTargetInputSlot("space"),
|
|
583
|
+
edge: createBoardTargetInputSlot("edge"),
|
|
584
|
+
vertex: createBoardTargetInputSlot("vertex"),
|
|
585
|
+
tile: createBoardTargetInputSlot("tile"),
|
|
586
|
+
},
|
|
587
|
+
};
|
|
588
|
+
}
|
|
589
|
+
|
|
590
|
+
function createZoneCardsComponent(zones: readonly string[]) {
|
|
591
|
+
return ({
|
|
592
|
+
empty,
|
|
593
|
+
children,
|
|
594
|
+
...props
|
|
595
|
+
}: {
|
|
596
|
+
empty?: ReactNode;
|
|
597
|
+
children: (card: Card) => ReactNode;
|
|
598
|
+
} & Omit<ZoneListProps, "children" | "empty">) =>
|
|
599
|
+
createElement(
|
|
600
|
+
ReactFragment,
|
|
601
|
+
null,
|
|
602
|
+
...zones.map((zone) =>
|
|
603
|
+
createElement(baseUI.Zone.Root, {
|
|
604
|
+
key: zone,
|
|
605
|
+
zone: zone as never,
|
|
606
|
+
children: createElement(runtimeZone.List, {
|
|
607
|
+
...props,
|
|
608
|
+
empty,
|
|
609
|
+
children,
|
|
610
|
+
}),
|
|
611
|
+
}),
|
|
612
|
+
),
|
|
613
|
+
);
|
|
614
|
+
}
|
|
615
|
+
|
|
616
|
+
function createHandCardsComponent(options: WorkspaceHandComponentOptions) {
|
|
617
|
+
return ({
|
|
618
|
+
empty,
|
|
619
|
+
children,
|
|
620
|
+
className,
|
|
621
|
+
layout,
|
|
622
|
+
mobileInteraction,
|
|
623
|
+
cardSize,
|
|
624
|
+
ariaLabel,
|
|
625
|
+
dropTargets,
|
|
626
|
+
renderDropTargets,
|
|
627
|
+
onCardIntent,
|
|
628
|
+
renderSummary,
|
|
629
|
+
renderActions,
|
|
630
|
+
onSelectionSummary,
|
|
631
|
+
...props
|
|
632
|
+
}: {
|
|
633
|
+
empty?: ReactNode;
|
|
634
|
+
children: (card: Card, state: InteractionVisualState) => ReactNode;
|
|
635
|
+
className?: string;
|
|
636
|
+
layout?: HandLayoutKind | HandLayoutPolicy;
|
|
637
|
+
mobileInteraction?: HandInteractionPolicy;
|
|
638
|
+
cardSize?: "sm" | "md" | "lg";
|
|
639
|
+
ariaLabel?: string;
|
|
640
|
+
dropTargets?: ReadonlyArray<{
|
|
641
|
+
target:
|
|
642
|
+
| { kind: "card"; card: string }
|
|
643
|
+
| { kind: "space"; target: string }
|
|
644
|
+
| { kind: "edge"; target: string }
|
|
645
|
+
| { kind: "vertex"; target: string }
|
|
646
|
+
| { kind: "tile"; target: string };
|
|
647
|
+
label: string;
|
|
648
|
+
render: (state: CardDropTargetVisualState) => ReactNode;
|
|
649
|
+
className?: string;
|
|
650
|
+
role?: string;
|
|
651
|
+
order?: number;
|
|
652
|
+
}>;
|
|
653
|
+
renderDropTargets?: (children: ReactNode) => ReactNode;
|
|
654
|
+
onCardIntent?: (intent: AuthoredCardIntent) => void;
|
|
655
|
+
renderSummary?: (summary: HandSelectionSummary) => ReactNode;
|
|
656
|
+
renderActions?: (summary: HandSelectionSummary) => ReactNode;
|
|
657
|
+
onSelectionSummary?: (summary: HandSelectionSummary) => void;
|
|
658
|
+
} & Omit<ZoneListProps, "children" | "empty">) =>
|
|
659
|
+
createElement(baseUI.Zone.Root, {
|
|
660
|
+
zone: options.zone as never,
|
|
661
|
+
// Fill the author's container instead of shrink-wrapping to the fan's
|
|
662
|
+
// own measured width. Without a definite width here, a centering parent
|
|
663
|
+
// sizes to the cards row, the hand measures that, recomputes a new fan
|
|
664
|
+
// width, and the layout chases itself (a ResizeObserver loop). `w-full`
|
|
665
|
+
// pins the outermost hand element to the available width and breaks the
|
|
666
|
+
// cycle for every author layout.
|
|
667
|
+
className: "w-full min-w-0",
|
|
668
|
+
children: createElement(GeneratedHandCards, {
|
|
669
|
+
...props,
|
|
670
|
+
hand: options,
|
|
671
|
+
className,
|
|
672
|
+
empty,
|
|
673
|
+
children,
|
|
674
|
+
layout,
|
|
675
|
+
mobileInteraction,
|
|
676
|
+
cardSize,
|
|
677
|
+
ariaLabel,
|
|
678
|
+
dropTargets,
|
|
679
|
+
renderDropTargets,
|
|
680
|
+
onCardIntent,
|
|
681
|
+
renderSummary,
|
|
682
|
+
renderActions,
|
|
683
|
+
onSelectionSummary,
|
|
684
|
+
}),
|
|
685
|
+
});
|
|
686
|
+
}
|
|
687
|
+
|
|
688
|
+
function GeneratedHandCards({
|
|
689
|
+
hand,
|
|
690
|
+
empty,
|
|
691
|
+
children,
|
|
692
|
+
className,
|
|
693
|
+
sort,
|
|
694
|
+
layout,
|
|
695
|
+
mobileInteraction,
|
|
696
|
+
cardSize,
|
|
697
|
+
ariaLabel,
|
|
698
|
+
dropTargets,
|
|
699
|
+
renderDropTargets,
|
|
700
|
+
onCardIntent,
|
|
701
|
+
renderSummary,
|
|
702
|
+
renderActions,
|
|
703
|
+
onSelectionSummary,
|
|
704
|
+
}: {
|
|
705
|
+
hand: WorkspaceHandComponentOptions;
|
|
706
|
+
empty?: ReactNode;
|
|
707
|
+
children: (card: Card, state: InteractionVisualState) => ReactNode;
|
|
708
|
+
className?: string;
|
|
709
|
+
layout?: HandLayoutKind | HandLayoutPolicy;
|
|
710
|
+
mobileInteraction?: HandInteractionPolicy;
|
|
711
|
+
cardSize?: "sm" | "md" | "lg";
|
|
712
|
+
ariaLabel?: string;
|
|
713
|
+
dropTargets?: ReadonlyArray<{
|
|
714
|
+
target:
|
|
715
|
+
| { kind: "card"; card: string }
|
|
716
|
+
| { kind: "space"; target: string }
|
|
717
|
+
| { kind: "edge"; target: string }
|
|
718
|
+
| { kind: "vertex"; target: string }
|
|
719
|
+
| { kind: "tile"; target: string };
|
|
720
|
+
label: string;
|
|
721
|
+
render: (state: CardDropTargetVisualState) => ReactNode;
|
|
722
|
+
className?: string;
|
|
723
|
+
role?: string;
|
|
724
|
+
order?: number;
|
|
725
|
+
}>;
|
|
726
|
+
renderDropTargets?: (children: ReactNode) => ReactNode;
|
|
727
|
+
onCardIntent?: (intent: AuthoredCardIntent) => void;
|
|
728
|
+
renderSummary?: (summary: HandSelectionSummary) => ReactNode;
|
|
729
|
+
renderActions?: (summary: HandSelectionSummary) => ReactNode;
|
|
730
|
+
onSelectionSummary?: (summary: HandSelectionSummary) => void;
|
|
731
|
+
} & Omit<ZoneListProps, "children" | "empty">) {
|
|
732
|
+
const isMobile = useIsMobile();
|
|
733
|
+
const { items, count } = useZoneCards({ sort });
|
|
734
|
+
// One predictable presentation for every hand: the projected surface
|
|
735
|
+
// (fan on desktop, tray on mobile by default). Authors choose other shapes
|
|
736
|
+
// explicitly with `layout` (e.g. layout="strip"). We no longer fall back to
|
|
737
|
+
// a bare scroll strip when no interaction props are passed, so the layout —
|
|
738
|
+
// and the eligible/selected projection — stays consistent across phases.
|
|
739
|
+
const handClassName = clsx("min-h-[112px]", className);
|
|
740
|
+
const resolvedDropTargets = useMemo<RuntimeDropTarget[] | undefined>(() => {
|
|
741
|
+
if (!dropTargets || dropTargets.length === 0) return undefined;
|
|
742
|
+
return dropTargets.map((dt) => {
|
|
743
|
+
const value =
|
|
744
|
+
dt.target.kind === "card" ? dt.target.card : dt.target.target;
|
|
745
|
+
return {
|
|
746
|
+
targetId: dropTargetIdFor(dt.target.kind, value),
|
|
747
|
+
label: dt.label,
|
|
748
|
+
render: dt.render,
|
|
749
|
+
className: dt.className,
|
|
750
|
+
role: dt.role,
|
|
751
|
+
order: dt.order,
|
|
752
|
+
};
|
|
753
|
+
});
|
|
754
|
+
}, [dropTargets]);
|
|
755
|
+
const content = useMemo(
|
|
756
|
+
() =>
|
|
757
|
+
createElement(HandSurfaceView, {
|
|
758
|
+
zone: hand.zone,
|
|
759
|
+
cards: items,
|
|
760
|
+
renderCard: (card, state, _index) => children(card as Card, state),
|
|
761
|
+
layout,
|
|
762
|
+
mobileInteraction,
|
|
763
|
+
dropTargets: resolvedDropTargets,
|
|
764
|
+
renderDropTargets,
|
|
765
|
+
cardSize,
|
|
766
|
+
renderEmpty: empty !== undefined ? () => empty : undefined,
|
|
767
|
+
ariaLabel,
|
|
768
|
+
onIntentRouted: onCardIntent
|
|
769
|
+
? (intent) => onCardIntent(intent)
|
|
770
|
+
: undefined,
|
|
771
|
+
renderSummary,
|
|
772
|
+
renderActions,
|
|
773
|
+
onSelectionSummary,
|
|
774
|
+
className: handClassName,
|
|
775
|
+
} satisfies HandSurfaceViewProps<(typeof items)[number]>),
|
|
776
|
+
[
|
|
777
|
+
ariaLabel,
|
|
778
|
+
cardSize,
|
|
779
|
+
children,
|
|
780
|
+
empty,
|
|
781
|
+
hand.zone,
|
|
782
|
+
handClassName,
|
|
783
|
+
items,
|
|
784
|
+
layout,
|
|
785
|
+
mobileInteraction,
|
|
786
|
+
onCardIntent,
|
|
787
|
+
onSelectionSummary,
|
|
788
|
+
renderActions,
|
|
789
|
+
renderDropTargets,
|
|
790
|
+
renderSummary,
|
|
791
|
+
resolvedDropTargets,
|
|
792
|
+
],
|
|
793
|
+
);
|
|
794
|
+
const active = items.some(
|
|
795
|
+
(item) =>
|
|
796
|
+
!item.hidden &&
|
|
797
|
+
item.interactions.some((descriptor) =>
|
|
798
|
+
isInteractionAvailable(descriptor),
|
|
799
|
+
),
|
|
800
|
+
);
|
|
801
|
+
const autoOpen = items.some(
|
|
802
|
+
(item) =>
|
|
803
|
+
!item.hidden &&
|
|
804
|
+
item.interactions.some(
|
|
805
|
+
(descriptor) =>
|
|
806
|
+
isInteractionAvailable(descriptor) &&
|
|
807
|
+
descriptor.inputs.some(
|
|
808
|
+
(input) =>
|
|
809
|
+
input.domain.type === "cardTarget" &&
|
|
810
|
+
input.domain.selection?.mode === "many",
|
|
811
|
+
),
|
|
812
|
+
),
|
|
813
|
+
);
|
|
814
|
+
const version = items
|
|
815
|
+
.map((item) =>
|
|
816
|
+
item.hidden
|
|
817
|
+
? `${item.id}:hidden`
|
|
818
|
+
: `${item.id}:${item.cardType}:${JSON.stringify(item.properties)}`,
|
|
819
|
+
)
|
|
820
|
+
.join("|");
|
|
821
|
+
const registration = useMemo(
|
|
822
|
+
() => ({
|
|
823
|
+
id: `${hand.name}:${hand.zone}`,
|
|
824
|
+
zone: hand.zone,
|
|
825
|
+
label: hand.label,
|
|
826
|
+
role: hand.role,
|
|
827
|
+
order: hand.order,
|
|
828
|
+
version,
|
|
829
|
+
count,
|
|
830
|
+
active,
|
|
831
|
+
autoOpen,
|
|
832
|
+
content: createElement(baseUI.Zone.Root, {
|
|
833
|
+
zone: hand.zone as never,
|
|
834
|
+
children: content,
|
|
835
|
+
}),
|
|
836
|
+
}),
|
|
837
|
+
[
|
|
838
|
+
active,
|
|
839
|
+
autoOpen,
|
|
840
|
+
content,
|
|
841
|
+
count,
|
|
842
|
+
hand.label,
|
|
843
|
+
hand.name,
|
|
844
|
+
hand.order,
|
|
845
|
+
hand.role,
|
|
846
|
+
hand.zone,
|
|
847
|
+
version,
|
|
848
|
+
],
|
|
849
|
+
);
|
|
850
|
+
useRegisterMobileHand(registration);
|
|
851
|
+
|
|
852
|
+
return isMobile ? null : content;
|
|
853
|
+
}
|
|
854
|
+
|
|
855
|
+
function createZoneCardComponent() {
|
|
856
|
+
return ({
|
|
857
|
+
card,
|
|
858
|
+
children,
|
|
859
|
+
className,
|
|
860
|
+
...props
|
|
861
|
+
}: {
|
|
862
|
+
card: Card;
|
|
863
|
+
children?: ReactNode;
|
|
864
|
+
} & Omit<ButtonHTMLAttributes<HTMLButtonElement>, "type" | "value">) => {
|
|
865
|
+
const cardId = options.cardIdFromZoneCard(card);
|
|
866
|
+
const zone = options.zoneIdFromZoneCard(card);
|
|
867
|
+
const viewCard = cardRenderItemToViewCard(card, cardId);
|
|
868
|
+
const faceDown =
|
|
869
|
+
!!card &&
|
|
870
|
+
typeof card === "object" &&
|
|
871
|
+
"hidden" in card &&
|
|
872
|
+
(card as { hidden?: unknown }).hidden === true;
|
|
873
|
+
const renderCardFace = (state: InteractionCardInputRenderState) =>
|
|
874
|
+
createElement(CardFace, {
|
|
875
|
+
card: viewCard,
|
|
876
|
+
selected: state.selected,
|
|
877
|
+
disabled: state.disabled || !state.eligible,
|
|
878
|
+
faceDown,
|
|
879
|
+
size: "sm",
|
|
880
|
+
children,
|
|
881
|
+
});
|
|
882
|
+
const match = usePluginState((state: PluginStateSnapshot) => {
|
|
883
|
+
const candidates =
|
|
884
|
+
state.gameplay.zones[zone]?.playableByCardId[cardId] ?? [];
|
|
885
|
+
for (const descriptor of candidates) {
|
|
886
|
+
const input = descriptor.inputs.find(
|
|
887
|
+
(candidateInput) =>
|
|
888
|
+
candidateInput.domain.type === "cardTarget" &&
|
|
889
|
+
candidateInput.domain.projection === "resolved" &&
|
|
890
|
+
candidateInput.domain.eligibleTargets.includes(cardId),
|
|
891
|
+
);
|
|
892
|
+
if (input) return { descriptor, input };
|
|
893
|
+
}
|
|
894
|
+
return null;
|
|
895
|
+
});
|
|
896
|
+
if (!match) {
|
|
897
|
+
return createElement(
|
|
898
|
+
"div",
|
|
899
|
+
{
|
|
900
|
+
...props,
|
|
901
|
+
className: clsx("relative inline-flex", className),
|
|
902
|
+
"data-dreamboard-zone-card": "",
|
|
903
|
+
"data-card-id": cardId,
|
|
904
|
+
"data-zone": zone,
|
|
905
|
+
"data-interactive": false,
|
|
906
|
+
},
|
|
907
|
+
createElement(CardFace, {
|
|
908
|
+
card: viewCard,
|
|
909
|
+
faceDown,
|
|
910
|
+
size: "sm",
|
|
911
|
+
children,
|
|
912
|
+
}),
|
|
913
|
+
);
|
|
914
|
+
}
|
|
915
|
+
return createElement(baseUI.Zone.Root, {
|
|
916
|
+
zone: zone as never,
|
|
917
|
+
children: createElement(runtimeInteraction.Root, {
|
|
918
|
+
interaction: match.descriptor.interactionKey,
|
|
919
|
+
children: createElement(runtimeInteraction.CardInput, {
|
|
920
|
+
...props,
|
|
921
|
+
className: clsx(DEFAULT_ZONE_CARD_CLASS, className),
|
|
922
|
+
input: match.input.key,
|
|
923
|
+
unsafeCardId: cardId,
|
|
924
|
+
children: renderCardFace,
|
|
925
|
+
}),
|
|
926
|
+
}),
|
|
927
|
+
});
|
|
928
|
+
};
|
|
929
|
+
}
|
|
930
|
+
|
|
931
|
+
const Board = {
|
|
932
|
+
surface<Board extends string>(
|
|
933
|
+
board: Board,
|
|
934
|
+
): WorkspaceBoardSurfaceDescriptor<Board> {
|
|
935
|
+
return { kind: "board", board };
|
|
936
|
+
},
|
|
937
|
+
useSurface: useBoardSurface,
|
|
938
|
+
HexView({ board: boardId, ...props }: { board: string }) {
|
|
939
|
+
return createElement(
|
|
940
|
+
baseUI.Board.HexView as never,
|
|
941
|
+
{
|
|
942
|
+
...props,
|
|
943
|
+
board: options.hexStaticBoards[boardId],
|
|
944
|
+
} as BoardHexViewProps<never, never>,
|
|
945
|
+
) as ReactElement;
|
|
946
|
+
},
|
|
947
|
+
HexGrid({ board: boardId, ...props }: { board: string }) {
|
|
948
|
+
return createElement(
|
|
949
|
+
baseUI.Board.HexGrid as never,
|
|
950
|
+
{
|
|
951
|
+
...props,
|
|
952
|
+
board: options.hexStaticBoards[boardId],
|
|
953
|
+
} as BoardHexGridProps<never, never>,
|
|
954
|
+
) as ReactElement;
|
|
955
|
+
},
|
|
956
|
+
};
|
|
957
|
+
|
|
958
|
+
function createStagingComponent(options: WorkspaceHandComponentOptions) {
|
|
959
|
+
return ({
|
|
960
|
+
children,
|
|
961
|
+
label,
|
|
962
|
+
renderEmptySlot,
|
|
963
|
+
cardSize,
|
|
964
|
+
ariaLabel,
|
|
965
|
+
className,
|
|
966
|
+
}: {
|
|
967
|
+
children: (card: Card) => ReactNode;
|
|
968
|
+
label?: ReactNode;
|
|
969
|
+
renderEmptySlot?: (index: number) => ReactNode;
|
|
970
|
+
cardSize?: "sm" | "md" | "lg";
|
|
971
|
+
ariaLabel?: string;
|
|
972
|
+
className?: string;
|
|
973
|
+
}) =>
|
|
974
|
+
createElement(HandStagingView, {
|
|
975
|
+
zone: options.zone,
|
|
976
|
+
renderCard: (card) => children(card as Card),
|
|
977
|
+
renderEmptySlot,
|
|
978
|
+
label,
|
|
979
|
+
cardSize,
|
|
980
|
+
ariaLabel,
|
|
981
|
+
className,
|
|
982
|
+
});
|
|
983
|
+
}
|
|
984
|
+
|
|
985
|
+
const Zone = {
|
|
986
|
+
hand<Zone extends string>(
|
|
987
|
+
zone: Zone,
|
|
988
|
+
options: Omit<WorkspaceHandSurfaceDescriptor<Zone>, "kind" | "zone">,
|
|
989
|
+
): WorkspaceHandSurfaceDescriptor<Zone> {
|
|
990
|
+
return { kind: "hand", zone, ...options };
|
|
991
|
+
},
|
|
992
|
+
pile<Zone extends string>(
|
|
993
|
+
zone: Zone,
|
|
994
|
+
): WorkspacePileSurfaceDescriptor<Zone> {
|
|
995
|
+
return { kind: "pile", zone };
|
|
996
|
+
},
|
|
997
|
+
piles<const Zones extends readonly string[]>(
|
|
998
|
+
zones: Zones,
|
|
999
|
+
): WorkspacePilesSurfaceDescriptor<Zones> {
|
|
1000
|
+
return { kind: "piles", zones };
|
|
1001
|
+
},
|
|
1002
|
+
collection<const Zones extends readonly string[]>(
|
|
1003
|
+
zones: Zones,
|
|
1004
|
+
options: { mode?: "all" | "top-card" } = {},
|
|
1005
|
+
): WorkspaceCardCollectionSurfaceDescriptor<Zones> {
|
|
1006
|
+
return { kind: "cardCollection", zones, mode: options.mode };
|
|
1007
|
+
},
|
|
1008
|
+
useHand(_name: string, zoneOptions: WorkspaceHandOptions) {
|
|
1009
|
+
const handComponentOptions = {
|
|
1010
|
+
name: _name,
|
|
1011
|
+
zone: zoneOptions.zone,
|
|
1012
|
+
role: zoneOptions.role,
|
|
1013
|
+
label: zoneOptions.label,
|
|
1014
|
+
order: zoneOptions.order,
|
|
1015
|
+
};
|
|
1016
|
+
return {
|
|
1017
|
+
Hand: createHandCardsComponent(handComponentOptions),
|
|
1018
|
+
Card: createZoneCardComponent(),
|
|
1019
|
+
Staging: createStagingComponent(handComponentOptions),
|
|
1020
|
+
slot: { card: createCardInputSlot() },
|
|
1021
|
+
};
|
|
1022
|
+
},
|
|
1023
|
+
usePile(_name: string, zoneOptions: { zone: string }) {
|
|
1024
|
+
return {
|
|
1025
|
+
Pile: createZoneCardsComponent([zoneOptions.zone]),
|
|
1026
|
+
Card: createZoneCardComponent(),
|
|
1027
|
+
};
|
|
1028
|
+
},
|
|
1029
|
+
useCardCollection(
|
|
1030
|
+
_name: string,
|
|
1031
|
+
zoneOptions: { zones: readonly string[]; mode?: "all" | "top-card" },
|
|
1032
|
+
) {
|
|
1033
|
+
void zoneOptions.mode;
|
|
1034
|
+
return {
|
|
1035
|
+
Collection: createZoneCardsComponent(zoneOptions.zones),
|
|
1036
|
+
Card: createZoneCardComponent(),
|
|
1037
|
+
slot: { card: createCardInputSlot() },
|
|
1038
|
+
};
|
|
1039
|
+
},
|
|
1040
|
+
};
|
|
1041
|
+
|
|
1042
|
+
const Interaction = {
|
|
1043
|
+
State: baseUI.Interaction.State,
|
|
1044
|
+
Dialog: baseUI.Interaction.Dialog,
|
|
1045
|
+
useForm: useInteractionFormSurface,
|
|
1046
|
+
form<const Interaction extends string>(
|
|
1047
|
+
interaction: Interaction,
|
|
1048
|
+
): WorkspaceInteractionFormDescriptor<Interaction> {
|
|
1049
|
+
return { kind: "form", interaction };
|
|
1050
|
+
},
|
|
1051
|
+
forms<const Interactions extends Readonly<Record<string, string>>>(
|
|
1052
|
+
interactions: Interactions,
|
|
1053
|
+
): WorkspaceInteractionFormsDescriptor<Interactions> {
|
|
1054
|
+
return { kind: "forms", interactions };
|
|
1055
|
+
},
|
|
1056
|
+
Routes: InteractionRoutes,
|
|
1057
|
+
};
|
|
1058
|
+
|
|
1059
|
+
function isPlainObject(value: unknown): value is Record<string, unknown> {
|
|
1060
|
+
return !!value && typeof value === "object" && !Array.isArray(value);
|
|
1061
|
+
}
|
|
1062
|
+
|
|
1063
|
+
function isSurfaceDescriptor(
|
|
1064
|
+
value: unknown,
|
|
1065
|
+
): value is WorkspaceSurfaceDescriptor {
|
|
1066
|
+
return (
|
|
1067
|
+
isPlainObject(value) &&
|
|
1068
|
+
typeof (value as { kind?: unknown }).kind === "string"
|
|
1069
|
+
);
|
|
1070
|
+
}
|
|
1071
|
+
|
|
1072
|
+
function resolveSurfaceDescriptor(
|
|
1073
|
+
name: string,
|
|
1074
|
+
descriptor: WorkspaceSurfaceDescriptor,
|
|
1075
|
+
): unknown {
|
|
1076
|
+
switch (descriptor.kind) {
|
|
1077
|
+
case "board":
|
|
1078
|
+
return Board.useSurface(name);
|
|
1079
|
+
case "hand":
|
|
1080
|
+
return Zone.useHand(name, {
|
|
1081
|
+
zone: descriptor.zone,
|
|
1082
|
+
role: descriptor.role,
|
|
1083
|
+
label: descriptor.label,
|
|
1084
|
+
order: descriptor.order,
|
|
1085
|
+
});
|
|
1086
|
+
case "pile":
|
|
1087
|
+
return Zone.usePile(name, { zone: descriptor.zone });
|
|
1088
|
+
case "piles":
|
|
1089
|
+
return Object.fromEntries(
|
|
1090
|
+
descriptor.zones.map((zone) => [
|
|
1091
|
+
zone,
|
|
1092
|
+
Zone.usePile(String(zone), { zone }),
|
|
1093
|
+
]),
|
|
1094
|
+
);
|
|
1095
|
+
case "cardCollection":
|
|
1096
|
+
return Zone.useCardCollection(name, {
|
|
1097
|
+
zones: descriptor.zones,
|
|
1098
|
+
mode: descriptor.mode,
|
|
1099
|
+
});
|
|
1100
|
+
case "form":
|
|
1101
|
+
return useInteractionFormSurface(descriptor.interaction);
|
|
1102
|
+
case "forms":
|
|
1103
|
+
return Object.fromEntries(
|
|
1104
|
+
Object.entries(descriptor.interactions).map(([key, interaction]) => [
|
|
1105
|
+
key,
|
|
1106
|
+
useInteractionFormSurface(interaction),
|
|
1107
|
+
]),
|
|
1108
|
+
);
|
|
1109
|
+
}
|
|
1110
|
+
}
|
|
1111
|
+
|
|
1112
|
+
function resolveSurfaceSpec(spec: WorkspaceSurfaceSpec): unknown {
|
|
1113
|
+
return Object.fromEntries(
|
|
1114
|
+
Object.entries(spec).map(([key, value]) => [
|
|
1115
|
+
key,
|
|
1116
|
+
isSurfaceDescriptor(value)
|
|
1117
|
+
? resolveSurfaceDescriptor(key, value)
|
|
1118
|
+
: resolveSurfaceSpec(value as WorkspaceSurfaceSpec),
|
|
1119
|
+
]),
|
|
1120
|
+
);
|
|
1121
|
+
}
|
|
1122
|
+
|
|
1123
|
+
function defineSurfaces<const Spec extends WorkspaceSurfaceSpec>(spec: Spec) {
|
|
1124
|
+
return function useDefinedSurfaces() {
|
|
1125
|
+
return resolveSurfaceSpec(spec);
|
|
1126
|
+
};
|
|
1127
|
+
}
|
|
1128
|
+
|
|
1129
|
+
const UI = {
|
|
1130
|
+
...baseUI,
|
|
1131
|
+
Root: ({ children, ...props }: UIRootProps) =>
|
|
1132
|
+
createElement(ClientParamSchemaProvider, {
|
|
1133
|
+
schemas: options.clientParamSchemasByPhase,
|
|
1134
|
+
children: createElement(ToastProvider, {
|
|
1135
|
+
children: createElement(MobileHandTrayProvider, {
|
|
1136
|
+
children: createElement(baseUI.Root, { ...props, children }),
|
|
1137
|
+
}),
|
|
1138
|
+
}),
|
|
1139
|
+
}),
|
|
1140
|
+
defineSurfaces,
|
|
1141
|
+
Interaction,
|
|
1142
|
+
Board,
|
|
1143
|
+
Zone,
|
|
1144
|
+
ResourceCounter: resourceCounter,
|
|
1145
|
+
};
|
|
1146
|
+
|
|
1147
|
+
return UI as WorkspaceUI;
|
|
1148
|
+
}
|
|
1149
|
+
|
|
1150
|
+
export type {
|
|
1151
|
+
BoardHexGridProps,
|
|
1152
|
+
BoardHexViewProps,
|
|
1153
|
+
ClientParamSchemaMap,
|
|
1154
|
+
DreamboardUI,
|
|
1155
|
+
GameMeState,
|
|
1156
|
+
GamePlayersState,
|
|
1157
|
+
GameRenderState,
|
|
1158
|
+
GameTurnState,
|
|
1159
|
+
InteractionDialogProps,
|
|
1160
|
+
InteractionFormPrimitiveProps,
|
|
1161
|
+
InteractionStateProps,
|
|
1162
|
+
InteractionSubmitProps,
|
|
1163
|
+
InteractionTriggerProps,
|
|
1164
|
+
ResourceCounterComponents,
|
|
1165
|
+
TypedGame,
|
|
1166
|
+
UIContract,
|
|
1167
|
+
UIRootProps,
|
|
1168
|
+
ZoneCardRenderItem,
|
|
1169
|
+
ZoneListProps,
|
|
1170
|
+
};
|