@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,791 @@
|
|
|
1
|
+
import {
|
|
2
|
+
createContext,
|
|
3
|
+
useContext,
|
|
4
|
+
useMemo,
|
|
5
|
+
type ButtonHTMLAttributes,
|
|
6
|
+
type HTMLAttributes,
|
|
7
|
+
type ReactNode,
|
|
8
|
+
} from "react";
|
|
9
|
+
import type { ViewCard } from "../../types/index.js";
|
|
10
|
+
import { useInteractionUiStore } from "../context/InteractionDraftContext.js";
|
|
11
|
+
import { usePluginSession } from "../context/PluginSessionContext.js";
|
|
12
|
+
import { usePluginState } from "../context/PluginStateContext.js";
|
|
13
|
+
import { useRuntimeContext } from "../context/RuntimeContext.js";
|
|
14
|
+
import type {
|
|
15
|
+
InteractionDescriptor,
|
|
16
|
+
ZoneHandlesSnapshot,
|
|
17
|
+
} from "../types/plugin-state.js";
|
|
18
|
+
import type { ZoneKey } from "../ui-contract.js";
|
|
19
|
+
import {
|
|
20
|
+
inputByTarget,
|
|
21
|
+
interactionInputKeys,
|
|
22
|
+
isResolvedTargetDomain,
|
|
23
|
+
} from "../utils/interaction-inputs.js";
|
|
24
|
+
import {
|
|
25
|
+
claimInteractionSubmit,
|
|
26
|
+
clearInteractionRoute,
|
|
27
|
+
markInteractionPending,
|
|
28
|
+
routeInteractionTarget,
|
|
29
|
+
shouldRouteInteractionPending,
|
|
30
|
+
} from "../utils/interaction-router.js";
|
|
31
|
+
import { useGameActionError } from "./game.js";
|
|
32
|
+
import { runInteractionAction } from "./interaction-submit.js";
|
|
33
|
+
import { isInteractionAvailable } from "../utils/interaction-status.js";
|
|
34
|
+
import {
|
|
35
|
+
composeEventHandlers,
|
|
36
|
+
renderPrimitive,
|
|
37
|
+
type PrimitiveCommonProps,
|
|
38
|
+
} from "./primitive-props.js";
|
|
39
|
+
|
|
40
|
+
interface ZoneContextValue {
|
|
41
|
+
zone: string;
|
|
42
|
+
snapshot: ZoneHandlesSnapshot | null;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
interface ZoneCardContextValue {
|
|
46
|
+
zone: string;
|
|
47
|
+
cardId: string;
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
/**
|
|
51
|
+
* An item rendered by a zone primitive. Discriminated by `hidden`:
|
|
52
|
+
*
|
|
53
|
+
* - `hidden: false` — fully hydrated from the zone projection. Carries the
|
|
54
|
+
* authored card type, properties, and reducer-projected interaction state.
|
|
55
|
+
* - `hidden: true` — the zone snapshot exposes the card id but withholds its
|
|
56
|
+
* contents (e.g. opponent zones, or a zone that is projected count-only).
|
|
57
|
+
* The render contract is honest about not knowing the type; authors must
|
|
58
|
+
* narrow on `hidden` before reading `cardType` / `properties`.
|
|
59
|
+
*
|
|
60
|
+
* Replaces the previous silent fallback that widened `cardType` to the
|
|
61
|
+
* untyped string `"unknown"` — see SDK Design Principles §2 (strong contracts
|
|
62
|
+
* over comments).
|
|
63
|
+
*/
|
|
64
|
+
export type ZoneCardRenderItem<
|
|
65
|
+
CardIdValue extends string = string,
|
|
66
|
+
CardTypeValue extends string = string,
|
|
67
|
+
Properties extends Record<string, unknown> = Record<string, unknown>,
|
|
68
|
+
> =
|
|
69
|
+
| HydratedZoneCardRenderItem<CardIdValue, CardTypeValue, Properties>
|
|
70
|
+
| HiddenZoneCardRenderItem<CardIdValue>;
|
|
71
|
+
|
|
72
|
+
export interface HydratedZoneCardRenderItem<
|
|
73
|
+
CardIdValue extends string = string,
|
|
74
|
+
CardTypeValue extends string = string,
|
|
75
|
+
Properties extends Record<string, unknown> = Record<string, unknown>,
|
|
76
|
+
> extends ViewCard<CardIdValue, CardTypeValue, Properties> {
|
|
77
|
+
zone: string;
|
|
78
|
+
index: number;
|
|
79
|
+
hidden: false;
|
|
80
|
+
playable: boolean;
|
|
81
|
+
interactions: readonly InteractionDescriptor[];
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
export interface HiddenZoneCardRenderItem<CardIdValue extends string = string> {
|
|
85
|
+
id: CardIdValue;
|
|
86
|
+
zone: string;
|
|
87
|
+
index: number;
|
|
88
|
+
hidden: true;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
export interface ZonePileContextValue {
|
|
92
|
+
zone: string;
|
|
93
|
+
label: string;
|
|
94
|
+
count: number;
|
|
95
|
+
cards: readonly string[];
|
|
96
|
+
items: readonly ZoneCardRenderItem[];
|
|
97
|
+
hasVisibleCards: boolean;
|
|
98
|
+
isHidden: boolean;
|
|
99
|
+
description: string | null;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
const ZoneContext = createContext<ZoneContextValue | null>(null);
|
|
103
|
+
const ZoneCardContext = createContext<ZoneCardContextValue | null>(null);
|
|
104
|
+
const ZonePileContext = createContext<ZonePileContextValue | null>(null);
|
|
105
|
+
|
|
106
|
+
export function useZonePrimitiveContext(): ZoneContextValue {
|
|
107
|
+
const value = useContext(ZoneContext);
|
|
108
|
+
if (!value) {
|
|
109
|
+
throw new Error("Zone primitives must be rendered inside <Zone.Root>.");
|
|
110
|
+
}
|
|
111
|
+
return value;
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
export function useOptionalZonePrimitiveContext(): ZoneContextValue | null {
|
|
115
|
+
return useContext(ZoneContext);
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
export function useZoneCardContext(): ZoneCardContextValue | null {
|
|
119
|
+
return useContext(ZoneCardContext);
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
export function useZonePileContext(): ZonePileContextValue {
|
|
123
|
+
const value = useContext(ZonePileContext);
|
|
124
|
+
if (!value) {
|
|
125
|
+
throw new Error(
|
|
126
|
+
"Zone pile primitives must be rendered inside <Zone.PileRoot>.",
|
|
127
|
+
);
|
|
128
|
+
}
|
|
129
|
+
return value;
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
export interface ZoneRootProps<Zone extends string = ZoneKey>
|
|
133
|
+
extends PrimitiveCommonProps, HTMLAttributes<HTMLElement> {
|
|
134
|
+
zone: Zone;
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
export function ZoneRoot<Zone extends string = ZoneKey>({
|
|
138
|
+
zone,
|
|
139
|
+
children,
|
|
140
|
+
...props
|
|
141
|
+
}: ZoneRootProps<Zone>) {
|
|
142
|
+
const snapshot =
|
|
143
|
+
usePluginState((state) => state.gameplay.zones[zone]) ?? null;
|
|
144
|
+
const value = useMemo<ZoneContextValue>(
|
|
145
|
+
() => ({ zone, snapshot }),
|
|
146
|
+
[snapshot, zone],
|
|
147
|
+
);
|
|
148
|
+
return (
|
|
149
|
+
<ZoneContext.Provider value={value}>
|
|
150
|
+
{renderPrimitive("div", {
|
|
151
|
+
...props,
|
|
152
|
+
"data-dreamboard-zone-root": "",
|
|
153
|
+
"data-zone": zone,
|
|
154
|
+
"data-card-count": snapshot?.cardIds.length ?? 0,
|
|
155
|
+
children,
|
|
156
|
+
})}
|
|
157
|
+
</ZoneContext.Provider>
|
|
158
|
+
);
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
export interface ZoneListProps
|
|
162
|
+
extends
|
|
163
|
+
Omit<PrimitiveCommonProps, "children">,
|
|
164
|
+
Omit<HTMLAttributes<HTMLElement>, "children"> {
|
|
165
|
+
children?: ReactNode | ((card: ZoneCardRenderItem) => ReactNode);
|
|
166
|
+
empty?: ReactNode;
|
|
167
|
+
sort?: (a: ZoneCardRenderItem, b: ZoneCardRenderItem) => number;
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
export function ZoneList({ children, empty, sort, ...props }: ZoneListProps) {
|
|
171
|
+
const { zone, items, isEmpty } = useZoneCards({ sort });
|
|
172
|
+
const renderedChildren =
|
|
173
|
+
typeof children === "function"
|
|
174
|
+
? isEmpty
|
|
175
|
+
? empty
|
|
176
|
+
: items.map((item) => (
|
|
177
|
+
<ZoneItem key={item.id} card={item}>
|
|
178
|
+
{children(item)}
|
|
179
|
+
</ZoneItem>
|
|
180
|
+
))
|
|
181
|
+
: isEmpty && empty !== undefined
|
|
182
|
+
? empty
|
|
183
|
+
: children;
|
|
184
|
+
return renderPrimitive("div", {
|
|
185
|
+
...props,
|
|
186
|
+
role: props.role ?? "list",
|
|
187
|
+
"data-dreamboard-zone-list": "",
|
|
188
|
+
"data-zone": zone,
|
|
189
|
+
"data-empty": isEmpty || undefined,
|
|
190
|
+
children: renderedChildren,
|
|
191
|
+
});
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
export interface UseZoneCardsOptions {
|
|
195
|
+
sort?: (a: ZoneCardRenderItem, b: ZoneCardRenderItem) => number;
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
export interface UseZoneCardsResult {
|
|
199
|
+
zone: string;
|
|
200
|
+
items: readonly ZoneCardRenderItem[];
|
|
201
|
+
count: number;
|
|
202
|
+
isEmpty: boolean;
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
export function useZoneCards({
|
|
206
|
+
sort,
|
|
207
|
+
}: UseZoneCardsOptions = {}): UseZoneCardsResult {
|
|
208
|
+
const { zone, snapshot } = useZonePrimitiveContext();
|
|
209
|
+
return useMemo(() => {
|
|
210
|
+
const cards = (snapshot?.cardIds ?? []).map((cardId, index) =>
|
|
211
|
+
createZoneCardRenderItem(zone, snapshot, cardId, index),
|
|
212
|
+
);
|
|
213
|
+
const items = sort ? [...cards].sort(sort) : cards;
|
|
214
|
+
const count = snapshot?.cardIds.length ?? 0;
|
|
215
|
+
return {
|
|
216
|
+
zone,
|
|
217
|
+
items,
|
|
218
|
+
count,
|
|
219
|
+
isEmpty: count === 0,
|
|
220
|
+
};
|
|
221
|
+
}, [snapshot, sort, zone]);
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
export interface ZoneItemProps
|
|
225
|
+
extends PrimitiveCommonProps, HTMLAttributes<HTMLElement> {
|
|
226
|
+
card: string | ZoneCardRenderItem;
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
export function ZoneItem({ card, children, ...props }: ZoneItemProps) {
|
|
230
|
+
const { zone, snapshot } = useZonePrimitiveContext();
|
|
231
|
+
const item =
|
|
232
|
+
typeof card === "string"
|
|
233
|
+
? createZoneCardRenderItem(
|
|
234
|
+
zone,
|
|
235
|
+
snapshot,
|
|
236
|
+
card,
|
|
237
|
+
indexOfCard(snapshot, card),
|
|
238
|
+
)
|
|
239
|
+
: card;
|
|
240
|
+
const cardContext = useMemo<ZoneCardContextValue>(
|
|
241
|
+
() => ({ zone, cardId: item.id }),
|
|
242
|
+
[item.id, zone],
|
|
243
|
+
);
|
|
244
|
+
return (
|
|
245
|
+
<ZoneCardContext.Provider value={cardContext}>
|
|
246
|
+
{renderPrimitive("div", {
|
|
247
|
+
...props,
|
|
248
|
+
role: props.role ?? "listitem",
|
|
249
|
+
"data-dreamboard-zone-item": "",
|
|
250
|
+
"data-zone": zone,
|
|
251
|
+
"data-card-id": item.id,
|
|
252
|
+
"data-card-type": item.hidden ? undefined : item.cardType,
|
|
253
|
+
"data-card-index": item.index,
|
|
254
|
+
"data-card-hidden": item.hidden || undefined,
|
|
255
|
+
"data-playable": item.hidden ? undefined : item.playable || undefined,
|
|
256
|
+
children,
|
|
257
|
+
})}
|
|
258
|
+
</ZoneCardContext.Provider>
|
|
259
|
+
);
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
export interface ZoneCardAtProps<Zone extends string = ZoneKey> extends Omit<
|
|
263
|
+
ZoneItemProps,
|
|
264
|
+
"card" | "children"
|
|
265
|
+
> {
|
|
266
|
+
zone?: Zone;
|
|
267
|
+
index: number;
|
|
268
|
+
children?: ReactNode | ((card: ZoneCardRenderItem) => ReactNode);
|
|
269
|
+
empty?: ReactNode;
|
|
270
|
+
}
|
|
271
|
+
|
|
272
|
+
function ZoneCardAtContent<Zone extends string = ZoneKey>({
|
|
273
|
+
index,
|
|
274
|
+
children,
|
|
275
|
+
empty = null,
|
|
276
|
+
...props
|
|
277
|
+
}: Omit<ZoneCardAtProps<Zone>, "zone">) {
|
|
278
|
+
const { zone, snapshot } = useZonePrimitiveContext();
|
|
279
|
+
const cardIndex = resolveZoneCardIndex(snapshot, index);
|
|
280
|
+
if (cardIndex === null) return <>{empty}</>;
|
|
281
|
+
|
|
282
|
+
const cardId = snapshot?.cardIds[cardIndex];
|
|
283
|
+
if (!cardId) return <>{empty}</>;
|
|
284
|
+
|
|
285
|
+
const card = createZoneCardRenderItem(zone, snapshot, cardId, cardIndex);
|
|
286
|
+
return (
|
|
287
|
+
<ZoneItem card={card} {...props}>
|
|
288
|
+
{typeof children === "function" ? children(card) : children}
|
|
289
|
+
</ZoneItem>
|
|
290
|
+
);
|
|
291
|
+
}
|
|
292
|
+
|
|
293
|
+
export function ZoneCardAt<Zone extends string = ZoneKey>({
|
|
294
|
+
zone,
|
|
295
|
+
...props
|
|
296
|
+
}: ZoneCardAtProps<Zone>) {
|
|
297
|
+
if (zone) {
|
|
298
|
+
return (
|
|
299
|
+
<ZoneRoot zone={zone}>
|
|
300
|
+
<ZoneCardAtContent {...props} />
|
|
301
|
+
</ZoneRoot>
|
|
302
|
+
);
|
|
303
|
+
}
|
|
304
|
+
return <ZoneCardAtContent {...props} />;
|
|
305
|
+
}
|
|
306
|
+
|
|
307
|
+
export type ZoneTopCardProps<Zone extends string = ZoneKey> = Omit<
|
|
308
|
+
ZoneCardAtProps<Zone>,
|
|
309
|
+
"index"
|
|
310
|
+
>;
|
|
311
|
+
|
|
312
|
+
export function ZoneTopCard<Zone extends string = ZoneKey>(
|
|
313
|
+
props: ZoneTopCardProps<Zone>,
|
|
314
|
+
) {
|
|
315
|
+
return <ZoneCardAt {...props} index={0} />;
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
export type ZoneCardActionExtraInputs =
|
|
319
|
+
| Record<string, unknown>
|
|
320
|
+
| ((cardId: string) => Record<string, unknown>);
|
|
321
|
+
|
|
322
|
+
export interface ZoneCardActionProps<Card extends string = string>
|
|
323
|
+
extends
|
|
324
|
+
PrimitiveCommonProps,
|
|
325
|
+
Omit<ButtonHTMLAttributes<HTMLButtonElement>, "onSelect"> {
|
|
326
|
+
card?: Card;
|
|
327
|
+
interaction?: string;
|
|
328
|
+
input?: string;
|
|
329
|
+
extraInputs?: ZoneCardActionExtraInputs;
|
|
330
|
+
onSelect?: (result: ZoneCardActionResult) => void;
|
|
331
|
+
onSelectError?: (error: unknown) => void;
|
|
332
|
+
}
|
|
333
|
+
|
|
334
|
+
export type ZoneCardActionResult =
|
|
335
|
+
| { status: "none" }
|
|
336
|
+
| {
|
|
337
|
+
status: "pending";
|
|
338
|
+
interactionKey: string;
|
|
339
|
+
descriptor: InteractionDescriptor;
|
|
340
|
+
missingInputs: readonly string[];
|
|
341
|
+
}
|
|
342
|
+
| {
|
|
343
|
+
status: "submitted";
|
|
344
|
+
interactionKey: string;
|
|
345
|
+
descriptor: InteractionDescriptor;
|
|
346
|
+
};
|
|
347
|
+
|
|
348
|
+
export function ZoneCardAction<Card extends string = string>({
|
|
349
|
+
card,
|
|
350
|
+
interaction,
|
|
351
|
+
input,
|
|
352
|
+
extraInputs,
|
|
353
|
+
onSelect,
|
|
354
|
+
onSelectError,
|
|
355
|
+
disabled,
|
|
356
|
+
onClick,
|
|
357
|
+
children,
|
|
358
|
+
...props
|
|
359
|
+
}: ZoneCardActionProps<Card>) {
|
|
360
|
+
const { controllingPlayerId } = usePluginSession();
|
|
361
|
+
const runtime = useRuntimeContext();
|
|
362
|
+
const store = useInteractionUiStore();
|
|
363
|
+
const contextCard = useZoneCardContext();
|
|
364
|
+
const { snapshot } = useZonePrimitiveContext();
|
|
365
|
+
const gameActionError = useGameActionError();
|
|
366
|
+
const cardId = card ?? (contextCard?.cardId as Card | undefined);
|
|
367
|
+
const route = useZoneCardActionRoute(cardId, snapshot, interaction, input);
|
|
368
|
+
const isDisabled =
|
|
369
|
+
disabled ??
|
|
370
|
+
(!cardId ||
|
|
371
|
+
!route.descriptor ||
|
|
372
|
+
!route.inputKey ||
|
|
373
|
+
route.ambiguous ||
|
|
374
|
+
!isInteractionAvailable(route.descriptor));
|
|
375
|
+
|
|
376
|
+
return renderPrimitive("button", {
|
|
377
|
+
type: "button",
|
|
378
|
+
...props,
|
|
379
|
+
children,
|
|
380
|
+
disabled: isDisabled,
|
|
381
|
+
"aria-disabled": isDisabled,
|
|
382
|
+
"data-dreamboard-zone-card-action": "",
|
|
383
|
+
"data-card-id": cardId,
|
|
384
|
+
"data-interaction-id": route.descriptor?.interactionId,
|
|
385
|
+
"data-interaction-key": route.descriptor?.interactionKey ?? interaction,
|
|
386
|
+
"data-input-name": route.inputKey ?? input,
|
|
387
|
+
"data-eligible": Boolean(route.descriptor && route.inputKey),
|
|
388
|
+
"data-ambiguous": route.ambiguous || undefined,
|
|
389
|
+
"data-disabled": isDisabled || undefined,
|
|
390
|
+
onClick: composeEventHandlers(onClick, () => {
|
|
391
|
+
if (
|
|
392
|
+
isDisabled ||
|
|
393
|
+
!cardId ||
|
|
394
|
+
!route.descriptor ||
|
|
395
|
+
!route.inputKey ||
|
|
396
|
+
!controllingPlayerId
|
|
397
|
+
) {
|
|
398
|
+
return;
|
|
399
|
+
}
|
|
400
|
+
const descriptor = route.descriptor;
|
|
401
|
+
const inputKey = route.inputKey;
|
|
402
|
+
void runInteractionAction(
|
|
403
|
+
async (): Promise<ZoneCardActionResult> => {
|
|
404
|
+
const { params, readiness } = routeInteractionTarget(
|
|
405
|
+
store,
|
|
406
|
+
descriptor,
|
|
407
|
+
{
|
|
408
|
+
inputKey,
|
|
409
|
+
value: cardId,
|
|
410
|
+
extraInputs: resolveCardActionExtraInputs(extraInputs, cardId),
|
|
411
|
+
},
|
|
412
|
+
);
|
|
413
|
+
if (shouldRouteInteractionPending(descriptor, readiness)) {
|
|
414
|
+
markInteractionPending(store, descriptor);
|
|
415
|
+
return {
|
|
416
|
+
status: "pending",
|
|
417
|
+
interactionKey: descriptor.interactionKey,
|
|
418
|
+
descriptor,
|
|
419
|
+
missingInputs: readiness.missingInputs,
|
|
420
|
+
};
|
|
421
|
+
}
|
|
422
|
+
if (!claimInteractionSubmit(store, descriptor)) {
|
|
423
|
+
return {
|
|
424
|
+
status: "submitted",
|
|
425
|
+
interactionKey: descriptor.interactionKey,
|
|
426
|
+
descriptor,
|
|
427
|
+
};
|
|
428
|
+
}
|
|
429
|
+
try {
|
|
430
|
+
await runtime.submitInteraction(
|
|
431
|
+
controllingPlayerId,
|
|
432
|
+
descriptor.interactionId,
|
|
433
|
+
params,
|
|
434
|
+
);
|
|
435
|
+
clearInteractionRoute(store, descriptor);
|
|
436
|
+
return {
|
|
437
|
+
status: "submitted",
|
|
438
|
+
interactionKey: descriptor.interactionKey,
|
|
439
|
+
descriptor,
|
|
440
|
+
};
|
|
441
|
+
} finally {
|
|
442
|
+
store.setSubmitting(descriptor.interactionKey, false);
|
|
443
|
+
}
|
|
444
|
+
},
|
|
445
|
+
{
|
|
446
|
+
onSuccess: onSelect,
|
|
447
|
+
onError: onSelectError ?? gameActionError ?? undefined,
|
|
448
|
+
},
|
|
449
|
+
);
|
|
450
|
+
}),
|
|
451
|
+
});
|
|
452
|
+
}
|
|
453
|
+
|
|
454
|
+
export interface ZonePileRootProps<Zone extends string = ZoneKey> extends Omit<
|
|
455
|
+
ZoneRootProps<Zone>,
|
|
456
|
+
"children"
|
|
457
|
+
> {
|
|
458
|
+
label: string;
|
|
459
|
+
children?: ReactNode;
|
|
460
|
+
hiddenDescription?: string | null;
|
|
461
|
+
emptyDescription?: string | null;
|
|
462
|
+
visibleDescription?: ((count: number) => string) | null;
|
|
463
|
+
}
|
|
464
|
+
|
|
465
|
+
export function ZonePileRoot<Zone extends string = ZoneKey>({
|
|
466
|
+
zone,
|
|
467
|
+
label,
|
|
468
|
+
hiddenDescription = null,
|
|
469
|
+
emptyDescription = null,
|
|
470
|
+
visibleDescription = null,
|
|
471
|
+
children,
|
|
472
|
+
...props
|
|
473
|
+
}: ZonePileRootProps<Zone>) {
|
|
474
|
+
return (
|
|
475
|
+
<ZoneRoot zone={zone} {...props}>
|
|
476
|
+
<ZonePileProvider
|
|
477
|
+
emptyDescription={emptyDescription}
|
|
478
|
+
hiddenDescription={hiddenDescription}
|
|
479
|
+
label={label}
|
|
480
|
+
visibleDescription={visibleDescription}
|
|
481
|
+
>
|
|
482
|
+
{children}
|
|
483
|
+
</ZonePileProvider>
|
|
484
|
+
</ZoneRoot>
|
|
485
|
+
);
|
|
486
|
+
}
|
|
487
|
+
|
|
488
|
+
interface ZonePileProviderProps {
|
|
489
|
+
label: string;
|
|
490
|
+
children?: ReactNode;
|
|
491
|
+
hiddenDescription: string | null;
|
|
492
|
+
emptyDescription: string | null;
|
|
493
|
+
visibleDescription: ((count: number) => string) | null;
|
|
494
|
+
}
|
|
495
|
+
|
|
496
|
+
function ZonePileProvider({
|
|
497
|
+
label,
|
|
498
|
+
hiddenDescription,
|
|
499
|
+
emptyDescription,
|
|
500
|
+
visibleDescription,
|
|
501
|
+
children,
|
|
502
|
+
}: ZonePileProviderProps) {
|
|
503
|
+
const { zone, snapshot } = useZonePrimitiveContext();
|
|
504
|
+
// Snapshot is the single source of truth for what's in a pile. A zone that
|
|
505
|
+
// isn't in the current phase's projection scope (snapshot === null) is
|
|
506
|
+
// treated as hidden — author should change the reducer projection rather
|
|
507
|
+
// than inject ids in the UI.
|
|
508
|
+
const cards = snapshot?.cardIds ?? [];
|
|
509
|
+
const items = cards.map((cardId, index) =>
|
|
510
|
+
createZoneCardRenderItem(zone, snapshot, cardId, index),
|
|
511
|
+
);
|
|
512
|
+
const count = cards.length;
|
|
513
|
+
const isHidden = snapshot === null;
|
|
514
|
+
// PileCards iterates whatever the snapshot exposes. Items are tagged
|
|
515
|
+
// `hidden: true | false` so the author's `renderCard` discriminates on
|
|
516
|
+
// honest data — including the "id without contents" case — rather than
|
|
517
|
+
// receiving a lying ViewCard with `cardType: "unknown"`.
|
|
518
|
+
const hasVisibleCards = items.length > 0;
|
|
519
|
+
const description = isHidden
|
|
520
|
+
? hiddenDescription
|
|
521
|
+
: hasVisibleCards
|
|
522
|
+
? (visibleDescription?.(count) ?? null)
|
|
523
|
+
: emptyDescription;
|
|
524
|
+
const value = useMemo<ZonePileContextValue>(
|
|
525
|
+
() => ({
|
|
526
|
+
zone,
|
|
527
|
+
label,
|
|
528
|
+
count,
|
|
529
|
+
cards,
|
|
530
|
+
items,
|
|
531
|
+
hasVisibleCards,
|
|
532
|
+
isHidden,
|
|
533
|
+
description,
|
|
534
|
+
}),
|
|
535
|
+
[cards, count, description, hasVisibleCards, isHidden, items, label, zone],
|
|
536
|
+
);
|
|
537
|
+
|
|
538
|
+
return (
|
|
539
|
+
<ZonePileContext.Provider value={value}>
|
|
540
|
+
{children}
|
|
541
|
+
</ZonePileContext.Provider>
|
|
542
|
+
);
|
|
543
|
+
}
|
|
544
|
+
|
|
545
|
+
export interface ZonePileTriggerProps
|
|
546
|
+
extends PrimitiveCommonProps, ButtonHTMLAttributes<HTMLButtonElement> {}
|
|
547
|
+
|
|
548
|
+
export function ZonePileTrigger({ children, ...props }: ZonePileTriggerProps) {
|
|
549
|
+
const pile = useZonePileContext();
|
|
550
|
+
return renderPrimitive("button", {
|
|
551
|
+
type: "button",
|
|
552
|
+
...props,
|
|
553
|
+
"aria-label": props["aria-label"] ?? `Show ${pile.label} pile`,
|
|
554
|
+
"data-dreamboard-zone-pile-trigger": "",
|
|
555
|
+
"data-zone": pile.zone,
|
|
556
|
+
"data-card-count": pile.count,
|
|
557
|
+
"data-hidden": pile.isHidden || undefined,
|
|
558
|
+
children,
|
|
559
|
+
});
|
|
560
|
+
}
|
|
561
|
+
|
|
562
|
+
export type ZonePileLabelProps = PrimitiveCommonProps &
|
|
563
|
+
HTMLAttributes<HTMLElement>;
|
|
564
|
+
|
|
565
|
+
export function ZonePileLabel({ children, ...props }: ZonePileLabelProps) {
|
|
566
|
+
const pile = useZonePileContext();
|
|
567
|
+
return renderPrimitive("span", {
|
|
568
|
+
...props,
|
|
569
|
+
"data-dreamboard-zone-pile-label": "",
|
|
570
|
+
"data-zone": pile.zone,
|
|
571
|
+
children: children ?? pile.label,
|
|
572
|
+
});
|
|
573
|
+
}
|
|
574
|
+
|
|
575
|
+
export type ZonePileCountProps = PrimitiveCommonProps &
|
|
576
|
+
HTMLAttributes<HTMLElement>;
|
|
577
|
+
|
|
578
|
+
export function ZonePileCount({ children, ...props }: ZonePileCountProps) {
|
|
579
|
+
const pile = useZonePileContext();
|
|
580
|
+
return renderPrimitive("span", {
|
|
581
|
+
...props,
|
|
582
|
+
"data-dreamboard-zone-pile-count": "",
|
|
583
|
+
"data-zone": pile.zone,
|
|
584
|
+
"data-card-count": pile.count,
|
|
585
|
+
children: children ?? `${pile.count} cards`,
|
|
586
|
+
});
|
|
587
|
+
}
|
|
588
|
+
|
|
589
|
+
export type ZonePileDescriptionProps = PrimitiveCommonProps &
|
|
590
|
+
HTMLAttributes<HTMLElement>;
|
|
591
|
+
|
|
592
|
+
export function ZonePileDescription({
|
|
593
|
+
children,
|
|
594
|
+
...props
|
|
595
|
+
}: ZonePileDescriptionProps) {
|
|
596
|
+
const pile = useZonePileContext();
|
|
597
|
+
const description = children ?? pile.description;
|
|
598
|
+
if (description === null || description === undefined) return null;
|
|
599
|
+
|
|
600
|
+
return renderPrimitive("span", {
|
|
601
|
+
...props,
|
|
602
|
+
"data-dreamboard-zone-pile-description": "",
|
|
603
|
+
"data-zone": pile.zone,
|
|
604
|
+
children: description,
|
|
605
|
+
});
|
|
606
|
+
}
|
|
607
|
+
|
|
608
|
+
export interface ZonePileCardsProps extends Omit<ZoneListProps, "children"> {
|
|
609
|
+
renderCard: (card: ZoneCardRenderItem) => ReactNode;
|
|
610
|
+
}
|
|
611
|
+
|
|
612
|
+
export function ZonePileCards({ renderCard, ...props }: ZonePileCardsProps) {
|
|
613
|
+
const pile = useZonePileContext();
|
|
614
|
+
if (!pile.hasVisibleCards) return null;
|
|
615
|
+
|
|
616
|
+
return (
|
|
617
|
+
<ZoneList {...props}>
|
|
618
|
+
{pile.items.map((card) => (
|
|
619
|
+
<ZoneItem key={card.id} card={card}>
|
|
620
|
+
{renderCard(card)}
|
|
621
|
+
</ZoneItem>
|
|
622
|
+
))}
|
|
623
|
+
</ZoneList>
|
|
624
|
+
);
|
|
625
|
+
}
|
|
626
|
+
|
|
627
|
+
function indexOfCard(
|
|
628
|
+
snapshot: ZoneHandlesSnapshot | null,
|
|
629
|
+
cardId: string,
|
|
630
|
+
): number {
|
|
631
|
+
return snapshot?.cardIds.indexOf(cardId) ?? -1;
|
|
632
|
+
}
|
|
633
|
+
|
|
634
|
+
function resolveZoneCardIndex(
|
|
635
|
+
snapshot: ZoneHandlesSnapshot | null,
|
|
636
|
+
index: number,
|
|
637
|
+
): number | null {
|
|
638
|
+
const count = snapshot?.cardIds.length ?? 0;
|
|
639
|
+
const resolved = index < 0 ? count + index : index;
|
|
640
|
+
return resolved >= 0 && resolved < count ? resolved : null;
|
|
641
|
+
}
|
|
642
|
+
|
|
643
|
+
export function createZoneCardRenderItem(
|
|
644
|
+
zone: string,
|
|
645
|
+
snapshot: ZoneHandlesSnapshot | null,
|
|
646
|
+
cardId: string,
|
|
647
|
+
index: number,
|
|
648
|
+
): ZoneCardRenderItem {
|
|
649
|
+
const card = parseViewCard(snapshot?.cardViewsById[cardId]);
|
|
650
|
+
if (card === null) {
|
|
651
|
+
// The snapshot exposes this card id but not its contents. Surface that
|
|
652
|
+
// honestly via the `hidden: true` variant instead of fabricating a
|
|
653
|
+
// ViewCard with a fake `cardType: "unknown"`.
|
|
654
|
+
return { id: cardId, zone, index, hidden: true };
|
|
655
|
+
}
|
|
656
|
+
const interactions = snapshot?.playableByCardId[cardId] ?? [];
|
|
657
|
+
return {
|
|
658
|
+
...card,
|
|
659
|
+
id: cardId,
|
|
660
|
+
zone,
|
|
661
|
+
index,
|
|
662
|
+
hidden: false,
|
|
663
|
+
playable: interactions.some(isInteractionAvailable),
|
|
664
|
+
interactions,
|
|
665
|
+
};
|
|
666
|
+
}
|
|
667
|
+
|
|
668
|
+
function useZoneCardActionRoute(
|
|
669
|
+
cardId: string | undefined,
|
|
670
|
+
snapshot: ZoneHandlesSnapshot | null,
|
|
671
|
+
interaction: string | undefined,
|
|
672
|
+
input: string | undefined,
|
|
673
|
+
): {
|
|
674
|
+
descriptor: InteractionDescriptor | null;
|
|
675
|
+
inputKey: string | null;
|
|
676
|
+
ambiguous: boolean;
|
|
677
|
+
} {
|
|
678
|
+
return useMemo(() => {
|
|
679
|
+
if (!cardId || !snapshot) {
|
|
680
|
+
return { descriptor: null, inputKey: null, ambiguous: false };
|
|
681
|
+
}
|
|
682
|
+
const interactions = snapshot.playableByCardId[cardId] ?? [];
|
|
683
|
+
if (interaction) {
|
|
684
|
+
const descriptor =
|
|
685
|
+
interactions.find(
|
|
686
|
+
(candidate) =>
|
|
687
|
+
candidate.interactionKey === interaction ||
|
|
688
|
+
candidate.interactionId === interaction,
|
|
689
|
+
) ?? null;
|
|
690
|
+
return {
|
|
691
|
+
descriptor,
|
|
692
|
+
inputKey: descriptor
|
|
693
|
+
? inputKeyForCardAction(descriptor, cardId, input)
|
|
694
|
+
: null,
|
|
695
|
+
ambiguous: false,
|
|
696
|
+
};
|
|
697
|
+
}
|
|
698
|
+
const matches = interactions.flatMap((descriptor) => {
|
|
699
|
+
if (!isInteractionAvailable(descriptor)) return [];
|
|
700
|
+
const inputKey = inputKeyForCardAction(descriptor, cardId, input);
|
|
701
|
+
return inputKey ? [{ descriptor, inputKey }] : [];
|
|
702
|
+
});
|
|
703
|
+
if (matches.length !== 1) {
|
|
704
|
+
return {
|
|
705
|
+
descriptor: matches[0]?.descriptor ?? null,
|
|
706
|
+
inputKey: matches[0]?.inputKey ?? null,
|
|
707
|
+
ambiguous: matches.length > 1,
|
|
708
|
+
};
|
|
709
|
+
}
|
|
710
|
+
const match = matches[0];
|
|
711
|
+
if (!match) {
|
|
712
|
+
return { descriptor: null, inputKey: null, ambiguous: false };
|
|
713
|
+
}
|
|
714
|
+
return {
|
|
715
|
+
descriptor: match.descriptor,
|
|
716
|
+
inputKey: match.inputKey,
|
|
717
|
+
ambiguous: false,
|
|
718
|
+
};
|
|
719
|
+
}, [cardId, input, interaction, snapshot]);
|
|
720
|
+
}
|
|
721
|
+
|
|
722
|
+
function inputKeyForCardAction(
|
|
723
|
+
descriptor: InteractionDescriptor,
|
|
724
|
+
cardId: string,
|
|
725
|
+
explicitInput?: string,
|
|
726
|
+
): string | null {
|
|
727
|
+
if (explicitInput) {
|
|
728
|
+
const input = descriptor.inputs.find(
|
|
729
|
+
(candidate) => candidate.key === explicitInput,
|
|
730
|
+
);
|
|
731
|
+
return input?.domain.type === "cardTarget" &&
|
|
732
|
+
isResolvedTargetDomain(input.domain) &&
|
|
733
|
+
input.domain.eligibleTargets.includes(cardId)
|
|
734
|
+
? input.key
|
|
735
|
+
: null;
|
|
736
|
+
}
|
|
737
|
+
const targetInput = inputByTarget(descriptor, "card", cardId);
|
|
738
|
+
if (targetInput) return targetInput.key;
|
|
739
|
+
if (interactionInputKeys(descriptor).includes("cardId")) {
|
|
740
|
+
return descriptor.inputs.find((candidate) => candidate.key === "cardId")
|
|
741
|
+
? "cardId"
|
|
742
|
+
: null;
|
|
743
|
+
}
|
|
744
|
+
return null;
|
|
745
|
+
}
|
|
746
|
+
|
|
747
|
+
function resolveCardActionExtraInputs(
|
|
748
|
+
extraInputs: ZoneCardActionExtraInputs | undefined,
|
|
749
|
+
cardId: string,
|
|
750
|
+
): Record<string, unknown> {
|
|
751
|
+
return typeof extraInputs === "function"
|
|
752
|
+
? extraInputs(cardId)
|
|
753
|
+
: (extraInputs ?? {});
|
|
754
|
+
}
|
|
755
|
+
|
|
756
|
+
function parseViewCard(serialized: string | undefined): ViewCard | null {
|
|
757
|
+
if (!serialized) return null;
|
|
758
|
+
try {
|
|
759
|
+
const parsed = JSON.parse(serialized) as Partial<ViewCard>;
|
|
760
|
+
if (typeof parsed.id !== "string" || typeof parsed.cardType !== "string") {
|
|
761
|
+
return null;
|
|
762
|
+
}
|
|
763
|
+
return {
|
|
764
|
+
...parsed,
|
|
765
|
+
id: parsed.id,
|
|
766
|
+
cardType: parsed.cardType,
|
|
767
|
+
properties:
|
|
768
|
+
parsed.properties && typeof parsed.properties === "object"
|
|
769
|
+
? parsed.properties
|
|
770
|
+
: {},
|
|
771
|
+
};
|
|
772
|
+
} catch {
|
|
773
|
+
return null;
|
|
774
|
+
}
|
|
775
|
+
}
|
|
776
|
+
|
|
777
|
+
export const Zone = {
|
|
778
|
+
Root: ZoneRoot,
|
|
779
|
+
List: ZoneList,
|
|
780
|
+
Item: ZoneItem,
|
|
781
|
+
CardAt: ZoneCardAt,
|
|
782
|
+
TopCard: ZoneTopCard,
|
|
783
|
+
CardAction: ZoneCardAction,
|
|
784
|
+
PileRoot: ZonePileRoot,
|
|
785
|
+
PileTrigger: ZonePileTrigger,
|
|
786
|
+
PileLabel: ZonePileLabel,
|
|
787
|
+
PileCount: ZonePileCount,
|
|
788
|
+
PileDescription: ZonePileDescription,
|
|
789
|
+
PileCards: ZonePileCards,
|
|
790
|
+
useZoneCards,
|
|
791
|
+
};
|