@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.
Files changed (369) hide show
  1. package/LICENSE.md +96 -0
  2. package/README.md +12 -0
  3. package/dist/HandView-ncJIVLhN.d.ts +193 -0
  4. package/dist/ResourceCounter-CTREyF73.d.ts +102 -0
  5. package/dist/ThemeProvider-fy0_QzgO.d.ts +99 -0
  6. package/dist/bundle-TIZcw8LB.d.ts +281 -0
  7. package/dist/cards-Sl3b40Mv.d.ts +13 -0
  8. package/dist/chunk-7YAHLYBR.js +481 -0
  9. package/dist/chunk-7YAHLYBR.js.map +1 -0
  10. package/dist/chunk-FDNZTDD6.js +8085 -0
  11. package/dist/chunk-FDNZTDD6.js.map +1 -0
  12. package/dist/chunk-GKKBPPSW.js +598 -0
  13. package/dist/chunk-GKKBPPSW.js.map +1 -0
  14. package/dist/chunk-I46YJSOD.js +1 -0
  15. package/dist/chunk-I46YJSOD.js.map +1 -0
  16. package/dist/chunk-KAELH4KC.js +104 -0
  17. package/dist/chunk-KAELH4KC.js.map +1 -0
  18. package/dist/chunk-PZ5AY32C.js +10 -0
  19. package/dist/chunk-PZ5AY32C.js.map +1 -0
  20. package/dist/chunk-T3ZKNUZ7.js +1 -0
  21. package/dist/chunk-T3ZKNUZ7.js.map +1 -0
  22. package/dist/chunk-T52J5RMF.js +1 -0
  23. package/dist/chunk-T52J5RMF.js.map +1 -0
  24. package/dist/chunk-TDSWKVZ4.js +5401 -0
  25. package/dist/chunk-TDSWKVZ4.js.map +1 -0
  26. package/dist/chunk-U5C6BONG.js +34 -0
  27. package/dist/chunk-U5C6BONG.js.map +1 -0
  28. package/dist/chunk-VDXOF4FW.js +69 -0
  29. package/dist/chunk-VDXOF4FW.js.map +1 -0
  30. package/dist/chunk-VFTAA4WO.js +115 -0
  31. package/dist/chunk-VFTAA4WO.js.map +1 -0
  32. package/dist/chunk-WN74KVNY.js +17 -0
  33. package/dist/chunk-WN74KVNY.js.map +1 -0
  34. package/dist/chunk-WYPQ3GG5.js +10990 -0
  35. package/dist/chunk-WYPQ3GG5.js.map +1 -0
  36. package/dist/components-D5ZRE2Hl.d.ts +1451 -0
  37. package/dist/generated/runtime/primitives.d.ts +12 -0
  38. package/dist/generated/runtime/primitives.js +180 -0
  39. package/dist/generated/runtime/primitives.js.map +1 -0
  40. package/dist/generated/runtime-api.d.ts +3 -0
  41. package/dist/generated/runtime-api.js +2 -0
  42. package/dist/generated/runtime-api.js.map +1 -0
  43. package/dist/generated/runtime.d.ts +14 -0
  44. package/dist/generated/runtime.js +18 -0
  45. package/dist/generated/runtime.js.map +1 -0
  46. package/dist/generated/workspace-contract.d.ts +14 -0
  47. package/dist/generated/workspace-contract.js +14 -0
  48. package/dist/generated/workspace-contract.js.map +1 -0
  49. package/dist/hex-board-view-D_07hO6O.d.ts +933 -0
  50. package/dist/hex-color-MhOyuY-o.d.ts +8 -0
  51. package/dist/index-BwqPQtBu.d.ts +1433 -0
  52. package/dist/index.d.ts +1 -0
  53. package/dist/index.js +12 -0
  54. package/dist/index.js.map +1 -0
  55. package/dist/infrastructure/reducer-bundle-abi.d.ts +1083 -0
  56. package/dist/infrastructure/reducer-bundle-abi.js +14 -0
  57. package/dist/infrastructure/reducer-bundle-abi.js.map +1 -0
  58. package/dist/infrastructure/workspace-codegen.d.ts +53 -0
  59. package/dist/infrastructure/workspace-codegen.js +44 -0
  60. package/dist/infrastructure/workspace-codegen.js.map +1 -0
  61. package/dist/manifest-contract-BNHVGFtU.d.ts +9 -0
  62. package/dist/package-set.d.ts +13 -0
  63. package/dist/package-set.js +12 -0
  64. package/dist/package-set.js.map +1 -0
  65. package/dist/primitive-props-DpKs-GCr.d.ts +11 -0
  66. package/dist/reducer.d.ts +3786 -0
  67. package/dist/reducer.js +8131 -0
  68. package/dist/reducer.js.map +1 -0
  69. package/dist/runtime/primitives.d.ts +226 -0
  70. package/dist/runtime/primitives.js +180 -0
  71. package/dist/runtime/primitives.js.map +1 -0
  72. package/dist/runtime/types/runtime-api.d.ts +1 -0
  73. package/dist/runtime/types/runtime-api.js +2 -0
  74. package/dist/runtime/types/runtime-api.js.map +1 -0
  75. package/dist/runtime/workspace-contract.d.ts +172 -0
  76. package/dist/runtime/workspace-contract.js +14 -0
  77. package/dist/runtime/workspace-contract.js.map +1 -0
  78. package/dist/runtime-api-3dshj6kK.d.ts +101 -0
  79. package/dist/runtime-api-DWxvTr-O.d.ts +379 -0
  80. package/dist/runtime.d.ts +58 -0
  81. package/dist/runtime.js +13 -0
  82. package/dist/runtime.js.map +1 -0
  83. package/dist/slots-1GPGihk8.d.ts +8 -0
  84. package/dist/testing.d.ts +149 -0
  85. package/dist/testing.js +513 -0
  86. package/dist/testing.js.map +1 -0
  87. package/dist/types.d.ts +496 -0
  88. package/dist/types.js +28 -0
  89. package/dist/types.js.map +1 -0
  90. package/dist/ui/components.d.ts +16 -0
  91. package/dist/ui/components.js +192 -0
  92. package/dist/ui/components.js.map +1 -0
  93. package/dist/ui/defaults.d.ts +19 -0
  94. package/dist/ui/defaults.js +104 -0
  95. package/dist/ui/defaults.js.map +1 -0
  96. package/dist/ui/plugin-styles.css +250 -0
  97. package/dist/ui/types/player-state.d.ts +365 -0
  98. package/dist/ui/types/player-state.js +1 -0
  99. package/dist/ui/types/player-state.js.map +1 -0
  100. package/dist/ui-contract-iQfTtUSL.d.ts +1161 -0
  101. package/dist/ui.d.ts +320 -0
  102. package/dist/ui.js +253 -0
  103. package/dist/ui.js.map +1 -0
  104. package/package.json +199 -0
  105. package/src/generated/reducer-contract/builders.ts +90 -0
  106. package/src/generated/reducer-contract/version.ts +9 -0
  107. package/src/generated/reducer-contract/wire.ts +100 -0
  108. package/src/generated/reducer-contract/zod.ts +101 -0
  109. package/src/generated/runtime/primitives.ts +2 -0
  110. package/src/generated/runtime-api.ts +5 -0
  111. package/src/generated/runtime.ts +35 -0
  112. package/src/generated/workspace-contract.ts +2 -0
  113. package/src/index.ts +7 -0
  114. package/src/infrastructure/reducer-bundle-abi.ts +8 -0
  115. package/src/infrastructure/reducer-contract/bundle.ts +37 -0
  116. package/src/infrastructure/workspace-codegen/hex-geometry.ts +69 -0
  117. package/src/infrastructure/workspace-codegen/index.ts +64 -0
  118. package/src/infrastructure/workspace-codegen/manifest-contract.ts +6632 -0
  119. package/src/infrastructure/workspace-codegen/manifest-validation.ts +795 -0
  120. package/src/infrastructure/workspace-codegen/ownership.ts +131 -0
  121. package/src/infrastructure/workspace-codegen/preset-card-sets.ts +169 -0
  122. package/src/infrastructure/workspace-codegen/seeds.ts +1705 -0
  123. package/src/infrastructure/workspace-codegen.ts +1 -0
  124. package/src/package-set.ts +19 -0
  125. package/src/reducer/authoring/contract.ts +157 -0
  126. package/src/reducer/authoring/effect.ts +224 -0
  127. package/src/reducer/authoring/game.ts +23 -0
  128. package/src/reducer/authoring/interaction.ts +98 -0
  129. package/src/reducer/authoring/phase.ts +300 -0
  130. package/src/reducer/authoring/types.ts +70 -0
  131. package/src/reducer/authoring/validation.ts +382 -0
  132. package/src/reducer/authoring/view-stage.ts +68 -0
  133. package/src/reducer/authoring.ts +29 -0
  134. package/src/reducer/bundle/ingress-bundle.ts +491 -0
  135. package/src/reducer/bundle/trusted/engine-instruction-resolver.ts +254 -0
  136. package/src/reducer/bundle/trusted/flow-instruction-resolver.ts +73 -0
  137. package/src/reducer/bundle/trusted/instruction-runner.ts +414 -0
  138. package/src/reducer/bundle/trusted/interaction-authorization.ts +137 -0
  139. package/src/reducer/bundle/trusted/interaction-collectors.ts +859 -0
  140. package/src/reducer/bundle/trusted/interaction-decision.ts +747 -0
  141. package/src/reducer/bundle/trusted/interaction-resolver.ts +95 -0
  142. package/src/reducer/bundle/trusted/interaction-types.ts +171 -0
  143. package/src/reducer/bundle/trusted/lifecycle-runner.ts +427 -0
  144. package/src/reducer/bundle/trusted/projection-builder.ts +356 -0
  145. package/src/reducer/bundle/trusted/projection-context.ts +39 -0
  146. package/src/reducer/bundle/trusted/rng-sampler.ts +150 -0
  147. package/src/reducer/bundle/trusted/runtime-registry.ts +120 -0
  148. package/src/reducer/bundle/trusted/runtime-scope.ts +336 -0
  149. package/src/reducer/bundle/trusted/simultaneous-player.ts +97 -0
  150. package/src/reducer/bundle/trusted/stage-resolver.ts +87 -0
  151. package/src/reducer/bundle/trusted/static-projection.ts +116 -0
  152. package/src/reducer/bundle/trusted/trusted-runtime-args.ts +97 -0
  153. package/src/reducer/bundle/trusted/trusted-runtime-result.ts +39 -0
  154. package/src/reducer/bundle/trusted/trusted-setup-profiles.ts +43 -0
  155. package/src/reducer/bundle/trusted/trusted-state-codec.ts +48 -0
  156. package/src/reducer/bundle/trusted-bundle.ts +97 -0
  157. package/src/reducer/bundle/types.ts +171 -0
  158. package/src/reducer/bundle.ts +2 -0
  159. package/src/reducer/client-param-schemas.ts +57 -0
  160. package/src/reducer/compose.ts +34 -0
  161. package/src/reducer/core/runtime-input.ts +30 -0
  162. package/src/reducer/core/runtime-instruction.ts +59 -0
  163. package/src/reducer/core/types.ts +62 -0
  164. package/src/reducer/definition-index.ts +277 -0
  165. package/src/reducer/derived.ts +106 -0
  166. package/src/reducer/effects.ts +92 -0
  167. package/src/reducer/engine/runtime-instruction-engine.ts +155 -0
  168. package/src/reducer/ingress/decode-runtime-input.ts +7 -0
  169. package/src/reducer/ingress/decode-session-state.ts +9 -0
  170. package/src/reducer/ingress/encode-session-state.ts +6 -0
  171. package/src/reducer/ingress/input-codec.ts +18 -0
  172. package/src/reducer/ingress/phase-schemas.ts +62 -0
  173. package/src/reducer/ingress/raw-types.ts +107 -0
  174. package/src/reducer/ingress/runtime-codec.ts +14 -0
  175. package/src/reducer/ingress/runtime-payload.ts +13 -0
  176. package/src/reducer/ingress/session-codec.ts +392 -0
  177. package/src/reducer/ingress/types.ts +6 -0
  178. package/src/reducer/inputs/boardInput.ts +217 -0
  179. package/src/reducer/inputs/boardTarget.ts +190 -0
  180. package/src/reducer/inputs/cardInput.ts +86 -0
  181. package/src/reducer/inputs/cardTarget.ts +101 -0
  182. package/src/reducer/inputs/choiceTarget.ts +104 -0
  183. package/src/reducer/inputs/defineInputs.ts +71 -0
  184. package/src/reducer/inputs/formInput.ts +809 -0
  185. package/src/reducer/inputs/many.ts +120 -0
  186. package/src/reducer/inputs/promptInput.ts +87 -0
  187. package/src/reducer/inputs/rngInput.ts +58 -0
  188. package/src/reducer/inputs/targetRule.ts +123 -0
  189. package/src/reducer/inputs.ts +41 -0
  190. package/src/reducer/model/definition.ts +1072 -0
  191. package/src/reducer/model/extract.ts +745 -0
  192. package/src/reducer/model/manifest.ts +570 -0
  193. package/src/reducer/model/queries.ts +641 -0
  194. package/src/reducer/model/runtime.ts +264 -0
  195. package/src/reducer/model/spec.ts +1386 -0
  196. package/src/reducer/model/table.ts +260 -0
  197. package/src/reducer/model.ts +7 -0
  198. package/src/reducer/ops.ts +1034 -0
  199. package/src/reducer/parse-utils.ts +28 -0
  200. package/src/reducer/per-player.ts +422 -0
  201. package/src/reducer/rng.ts +69 -0
  202. package/src/reducer/schema-helpers.ts +185 -0
  203. package/src/reducer/setup-bootstrap-helpers.ts +171 -0
  204. package/src/reducer/setup-bootstrap.ts +481 -0
  205. package/src/reducer/table-ops.ts +2671 -0
  206. package/src/reducer/table-queries.ts +372 -0
  207. package/src/reducer/transaction.ts +120 -0
  208. package/src/reducer.ts +314 -0
  209. package/src/runtime/primitives.ts +1 -0
  210. package/src/runtime/types/runtime-api.ts +1 -0
  211. package/src/runtime/workspace-contract.ts +32 -0
  212. package/src/runtime-internal/components/InteractionForm.tsx +1309 -0
  213. package/src/runtime-internal/components/PluginRuntime.tsx +103 -0
  214. package/src/runtime-internal/components/board/target-layer.ts +70 -0
  215. package/src/runtime-internal/context/ClientParamSchemaContext.tsx +44 -0
  216. package/src/runtime-internal/context/InteractionDraftContext.tsx +279 -0
  217. package/src/runtime-internal/context/PluginSessionContext.tsx +47 -0
  218. package/src/runtime-internal/context/PluginStateContext.tsx +262 -0
  219. package/src/runtime-internal/context/RuntimeContext.tsx +96 -0
  220. package/src/runtime-internal/defaults/components.tsx +409 -0
  221. package/src/runtime-internal/defaults/index.ts +11 -0
  222. package/src/runtime-internal/errors/ValidationError.ts +29 -0
  223. package/src/runtime-internal/hooks/useActivePlayers.ts +33 -0
  224. package/src/runtime-internal/hooks/useBoardInteractions.ts +665 -0
  225. package/src/runtime-internal/hooks/useGameSelector.ts +105 -0
  226. package/src/runtime-internal/hooks/useGameView.ts +9 -0
  227. package/src/runtime-internal/hooks/useInteractionByKey.ts +354 -0
  228. package/src/runtime-internal/hooks/useInteractionHandle.ts +438 -0
  229. package/src/runtime-internal/hooks/useIsMyTurn.ts +20 -0
  230. package/src/runtime-internal/hooks/useLobby.ts +76 -0
  231. package/src/runtime-internal/hooks/useMe.ts +48 -0
  232. package/src/runtime-internal/hooks/usePlayerInfo.ts +28 -0
  233. package/src/runtime-internal/hooks/usePlayerTurnOrder.ts +23 -0
  234. package/src/runtime-internal/hooks/usePluginRuntime.ts +147 -0
  235. package/src/runtime-internal/hooks/useSeatInbox.ts +61 -0
  236. package/src/runtime-internal/hooks/useSimultaneousPhase.ts +10 -0
  237. package/src/runtime-internal/index.ts +42 -0
  238. package/src/runtime-internal/internal.ts +43 -0
  239. package/src/runtime-internal/plugin-styles.css +250 -0
  240. package/src/runtime-internal/primitives/board.tsx +459 -0
  241. package/src/runtime-internal/primitives/dialog-lifecycle.ts +58 -0
  242. package/src/runtime-internal/primitives/dice.tsx +79 -0
  243. package/src/runtime-internal/primitives/game-ui-provider.tsx +35 -0
  244. package/src/runtime-internal/primitives/game.tsx +387 -0
  245. package/src/runtime-internal/primitives/hand-intent-adapter.ts +147 -0
  246. package/src/runtime-internal/primitives/hand-surface.tsx +594 -0
  247. package/src/runtime-internal/primitives/index.ts +196 -0
  248. package/src/runtime-internal/primitives/interaction-form-binding.tsx +56 -0
  249. package/src/runtime-internal/primitives/interaction-submit.ts +90 -0
  250. package/src/runtime-internal/primitives/interaction.tsx +987 -0
  251. package/src/runtime-internal/primitives/phase.tsx +43 -0
  252. package/src/runtime-internal/primitives/player-roster.tsx +302 -0
  253. package/src/runtime-internal/primitives/primitive-props.tsx +101 -0
  254. package/src/runtime-internal/primitives/prompt.tsx +255 -0
  255. package/src/runtime-internal/primitives/ui.tsx +60 -0
  256. package/src/runtime-internal/primitives/zone.tsx +791 -0
  257. package/src/runtime-internal/reducer.ts +30 -0
  258. package/src/runtime-internal/runtime/createPluginRuntimeAPI.ts +605 -0
  259. package/src/runtime-internal/types/plugin-state.ts +508 -0
  260. package/src/runtime-internal/types/reducer-state.ts +24 -0
  261. package/src/runtime-internal/types/runtime-api.ts +114 -0
  262. package/src/runtime-internal/ui-contract.ts +519 -0
  263. package/src/runtime-internal/utils/card-intent-adapter.ts +546 -0
  264. package/src/runtime-internal/utils/interaction-inputs.ts +492 -0
  265. package/src/runtime-internal/utils/interaction-labels.ts +23 -0
  266. package/src/runtime-internal/utils/interaction-router.ts +273 -0
  267. package/src/runtime-internal/utils/interaction-status.ts +74 -0
  268. package/src/runtime-internal/workspace-contract.ts +1170 -0
  269. package/src/runtime.ts +34 -0
  270. package/src/testing/create-expect-api.ts +352 -0
  271. package/src/testing/create-test-runtime.ts +381 -0
  272. package/src/testing/definitions.ts +127 -0
  273. package/src/testing/index.ts +3 -0
  274. package/src/testing.ts +1 -0
  275. package/src/type-stubs/manifest-contract.d.ts +42 -0
  276. package/src/type-stubs/manifest-contract.js +72 -0
  277. package/src/type-stubs/ui-contract.d.ts +5 -0
  278. package/src/type-stubs/ui-contract.js +1 -0
  279. package/src/types/authoring-card-properties.type-test.ts +266 -0
  280. package/src/types/authoring.ts +1282 -0
  281. package/src/types/cards.ts +19 -0
  282. package/src/types/contracts.ts +1550 -0
  283. package/src/types/generated-helpers.ts +35 -0
  284. package/src/types/index.ts +147 -0
  285. package/src/types/slots.ts +11 -0
  286. package/src/types.ts +1 -0
  287. package/src/ui/components/ActionButton.tsx +97 -0
  288. package/src/ui/components/ActionPanel.tsx +315 -0
  289. package/src/ui/components/Card.tsx +378 -0
  290. package/src/ui/components/CardDragSurface.tsx +1076 -0
  291. package/src/ui/components/ChromeSuppressionContext.tsx +70 -0
  292. package/src/ui/components/CostDisplay.tsx +145 -0
  293. package/src/ui/components/DiceRoller.tsx +581 -0
  294. package/src/ui/components/Drawer.tsx +180 -0
  295. package/src/ui/components/ErrorBoundary.tsx +275 -0
  296. package/src/ui/components/GameEndDisplay.tsx +398 -0
  297. package/src/ui/components/GameSkeleton.tsx +260 -0
  298. package/src/ui/components/Hand.tsx +468 -0
  299. package/src/ui/components/HandDock.tsx +299 -0
  300. package/src/ui/components/HandView.tsx +441 -0
  301. package/src/ui/components/MobileHandTray.tsx +381 -0
  302. package/src/ui/components/MoreActions.tsx +143 -0
  303. package/src/ui/components/PhaseIndicator.tsx +341 -0
  304. package/src/ui/components/PlayArea.tsx +146 -0
  305. package/src/ui/components/PrimaryActionButton.tsx +336 -0
  306. package/src/ui/components/PrimaryButton.tsx +45 -0
  307. package/src/ui/components/ResourceCounter.tsx +270 -0
  308. package/src/ui/components/StagingZone.tsx +134 -0
  309. package/src/ui/components/ThemedButton.tsx +113 -0
  310. package/src/ui/components/Toast.tsx +264 -0
  311. package/src/ui/components/board/HexGrid.tsx +1294 -0
  312. package/src/ui/components/board/NetworkGraph.tsx +476 -0
  313. package/src/ui/components/board/SlotSystem.tsx +388 -0
  314. package/src/ui/components/board/SquareGrid.tsx +1165 -0
  315. package/src/ui/components/board/TrackBoard.tsx +496 -0
  316. package/src/ui/components/board/ZoneMap.tsx +448 -0
  317. package/src/ui/components/board/hex-board-view.ts +123 -0
  318. package/src/ui/components/board/index.ts +142 -0
  319. package/src/ui/components/board/interaction-accessibility.ts +21 -0
  320. package/src/ui/components/board/target-layer.ts +66 -0
  321. package/src/ui/components/card-render-content.type-test.ts +27 -0
  322. package/src/ui/components/hand-layout-math.ts +163 -0
  323. package/src/ui/components/hand-pointer-engine.ts +413 -0
  324. package/src/ui/components/index.ts +245 -0
  325. package/src/ui/components.ts +1 -0
  326. package/src/ui/defaults/components.tsx +106 -0
  327. package/src/ui/defaults/index.ts +8 -0
  328. package/src/ui/defaults.ts +1 -0
  329. package/src/ui/errors/ValidationError.ts +29 -0
  330. package/src/ui/helpers/cards.ts +19 -0
  331. package/src/ui/helpers/track-board.ts +211 -0
  332. package/src/ui/hooks/useBoardTopology.ts +316 -0
  333. package/src/ui/hooks/useCards.ts +10 -0
  334. package/src/ui/hooks/useHandCardPointer.ts +381 -0
  335. package/src/ui/hooks/useHandLayout.ts +378 -0
  336. package/src/ui/hooks/useHandPresentation.ts +121 -0
  337. package/src/ui/hooks/useHexBoard.ts +74 -0
  338. package/src/ui/hooks/useHexGrid.ts +185 -0
  339. package/src/ui/hooks/useIsMobile.ts +35 -0
  340. package/src/ui/hooks/usePanZoom.ts +278 -0
  341. package/src/ui/hooks/useSquareBoard.ts +124 -0
  342. package/src/ui/hooks/useSquareGrid.ts +328 -0
  343. package/src/ui/index.ts +98 -0
  344. package/src/ui/internal/ui/alert.tsx +51 -0
  345. package/src/ui/internal/ui/button.tsx +58 -0
  346. package/src/ui/internal/ui/dialog.tsx +134 -0
  347. package/src/ui/internal/ui/input.tsx +21 -0
  348. package/src/ui/internal/ui/label.tsx +21 -0
  349. package/src/ui/internal/ui/select.tsx +129 -0
  350. package/src/ui/internal/ui/tooltip.tsx +54 -0
  351. package/src/ui/internal/ui/utils.ts +5 -0
  352. package/src/ui/plugin-styles.css +250 -0
  353. package/src/ui/primitives/dialog-lifecycle.ts +58 -0
  354. package/src/ui/primitives/dice.tsx +79 -0
  355. package/src/ui/primitives/primitive-props.tsx +101 -0
  356. package/src/ui/theme/ThemeProvider.tsx +252 -0
  357. package/src/ui/theme/board.ts +61 -0
  358. package/src/ui/theme/css-vars.ts +105 -0
  359. package/src/ui/theme/derive.ts +240 -0
  360. package/src/ui/theme/index.ts +61 -0
  361. package/src/ui/theme/presets/arcade.ts +261 -0
  362. package/src/ui/theme/presets/studio.ts +261 -0
  363. package/src/ui/theme/presets/tabletop.ts +266 -0
  364. package/src/ui/theme/tokens.ts +392 -0
  365. package/src/ui/types/hex-color.ts +20 -0
  366. package/src/ui/types/player-state.ts +463 -0
  367. package/src/ui/types/tiled-board.ts +785 -0
  368. package/src/ui/types/visual-state.ts +137 -0
  369. 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
+ };