@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,859 @@
1
+ import { formatIssue } from "../../parse-utils";
2
+ import { createDerivedResolver, type DerivedResolver } from "../../derived";
3
+ import { createStateQueries } from "../../table-queries";
4
+ import type {
5
+ AnyInteractionSpec,
6
+ BoardTargetDomainDescriptor,
7
+ CardTargetDomainDescriptor,
8
+ CollectorState,
9
+ InputCollector,
10
+ InputDomainDescriptor,
11
+ InputSelectionDescriptor,
12
+ ManifestContract,
13
+ ReducerValidationResult,
14
+ TableQueriesOfState,
15
+ TableOfState,
16
+ } from "../../model";
17
+ import { makeValidationError } from "./interaction-types";
18
+ import type { InteractionInputDescriptorShape } from "./interaction-types";
19
+
20
+ const MAX_EAGER_DEPENDENT_DOMAIN_CASES = 64;
21
+
22
+ type CollectorProjectionOptions<DomainState extends CollectorState> = {
23
+ readonly queries?: TableQueriesOfState<DomainState>;
24
+ readonly derived?: DerivedResolver;
25
+ readonly eligibleTargetCache?: Map<string, string[]>;
26
+ readonly eligibleTargetCachePrefix?: string;
27
+ readonly includeEligibleTargets?: boolean;
28
+ };
29
+
30
+ function collectTargetDomainMetadata(collector: InputCollector): {
31
+ targetKind?: string;
32
+ boardId?: string;
33
+ valueKind?: "board-id" | "player-board-space";
34
+ zoneId?: string;
35
+ zoneIds?: readonly string[];
36
+ } {
37
+ switch (collector.kind) {
38
+ case "card": {
39
+ const meta = collector.meta ?? {};
40
+ return {
41
+ targetKind: meta.targetKind ?? "card",
42
+ zoneId: meta.zoneId,
43
+ zoneIds:
44
+ meta.zoneIds ??
45
+ (meta.zoneId === undefined ? undefined : [meta.zoneId]),
46
+ };
47
+ }
48
+ case "board-edge":
49
+ case "board-space":
50
+ case "board-tile":
51
+ case "board-vertex": {
52
+ const meta = collector.meta ?? {};
53
+ return {
54
+ targetKind: meta.targetKind ?? collector.kind.replace("board-", ""),
55
+ boardId: meta.boardId,
56
+ valueKind: meta.valueKind,
57
+ };
58
+ }
59
+ default:
60
+ return {};
61
+ }
62
+ }
63
+
64
+ function withSelection(
65
+ domain: InputDomainDescriptor,
66
+ collector: InputCollector,
67
+ ): InputDomainDescriptor {
68
+ if (!collector.selection) return domain;
69
+ return { ...domain, selection: collector.selection };
70
+ }
71
+
72
+ function lazyTargetDomainFromCollectorMetadata(
73
+ key: string,
74
+ collector: InputCollector,
75
+ dependencies: readonly string[],
76
+ ): CardTargetDomainDescriptor | BoardTargetDomainDescriptor | null {
77
+ const metadata = collectTargetDomainMetadata(collector);
78
+ if (collector.kind === "card") {
79
+ return withSelection(
80
+ {
81
+ type: "cardTarget",
82
+ projection: "lazy",
83
+ targetKind: "card",
84
+ zoneIds:
85
+ metadata.zoneIds ??
86
+ (metadata.zoneId === undefined ? [] : [metadata.zoneId]),
87
+ dependencies: {
88
+ mode: "lazy",
89
+ dependsOn: dependencies,
90
+ resolver: { inputKey: key },
91
+ },
92
+ },
93
+ collector,
94
+ ) as CardTargetDomainDescriptor;
95
+ }
96
+ if (
97
+ collector.kind === "board-edge" ||
98
+ collector.kind === "board-space" ||
99
+ collector.kind === "board-tile" ||
100
+ collector.kind === "board-vertex"
101
+ ) {
102
+ return withSelection(
103
+ {
104
+ type: "boardTarget",
105
+ projection: "lazy",
106
+ targetKind: (metadata.targetKind ??
107
+ collector.kind.replace("board-", "")) as
108
+ | "edge"
109
+ | "vertex"
110
+ | "space"
111
+ | "tile",
112
+ boardId: metadata.boardId ?? "",
113
+ ...(metadata.valueKind ? { valueKind: metadata.valueKind } : {}),
114
+ dependencies: {
115
+ mode: "lazy",
116
+ dependsOn: dependencies,
117
+ resolver: { inputKey: key },
118
+ },
119
+ },
120
+ collector,
121
+ ) as BoardTargetDomainDescriptor;
122
+ }
123
+ return null;
124
+ }
125
+
126
+ function finiteValuesForDependency(
127
+ key: string,
128
+ domain: InputDomainDescriptor | undefined,
129
+ ): string[] {
130
+ if (!domain) {
131
+ throw new Error(
132
+ `Interaction input '${key}' is declared as a dependency before it has a projected domain.`,
133
+ );
134
+ }
135
+ if (domain.type === "cardTarget" || domain.type === "boardTarget") {
136
+ if (domain.projection !== "resolved") {
137
+ throw new Error(
138
+ `Interaction input '${key}' cannot be used as a dependency because its target domain is not finite.`,
139
+ );
140
+ }
141
+ return [...domain.eligibleTargets];
142
+ }
143
+ if (domain.type === "choice") {
144
+ return domain.choices.flatMap((choice) =>
145
+ choice.value === null ? [] : [choice.value],
146
+ );
147
+ }
148
+ throw new Error(
149
+ `Interaction input '${key}' cannot be used as a dependency. V1 supports finite target and choice dependencies only.`,
150
+ );
151
+ }
152
+
153
+ function cartesianDependencyTuples(
154
+ dependencies: readonly string[],
155
+ domainsByKey: Record<string, InputDomainDescriptor>,
156
+ ): Array<Record<string, string>> {
157
+ return dependencies.reduce<Array<Record<string, string>>>(
158
+ (tuples, key) => {
159
+ const values = finiteValuesForDependency(key, domainsByKey[key]);
160
+ return tuples.flatMap((tuple) =>
161
+ values.map((value) => ({ ...tuple, [key]: value })),
162
+ );
163
+ },
164
+ [{}],
165
+ );
166
+ }
167
+
168
+ function withoutDependentProjection(
169
+ domain: InputDomainDescriptor,
170
+ ): InputDomainDescriptor {
171
+ const { dependencies: _dependencies, ...rest } =
172
+ domain as InputDomainDescriptor & { dependencies?: unknown };
173
+ return rest as InputDomainDescriptor;
174
+ }
175
+
176
+ function toLazyTargetDomain(
177
+ key: string,
178
+ domain: InputDomainDescriptor,
179
+ dependencies: readonly string[],
180
+ ): CardTargetDomainDescriptor | BoardTargetDomainDescriptor | null {
181
+ if (domain.type === "cardTarget") {
182
+ return {
183
+ type: "cardTarget",
184
+ projection: "lazy",
185
+ targetKind: "card",
186
+ zoneIds: domain.zoneIds,
187
+ ...(domain.selection ? { selection: domain.selection } : {}),
188
+ dependencies: {
189
+ mode: "lazy",
190
+ dependsOn: dependencies,
191
+ resolver: { inputKey: key },
192
+ },
193
+ };
194
+ }
195
+ if (domain.type === "boardTarget") {
196
+ return {
197
+ type: "boardTarget",
198
+ projection: "lazy",
199
+ targetKind: domain.targetKind,
200
+ boardId: domain.boardId,
201
+ ...(domain.valueKind ? { valueKind: domain.valueKind } : {}),
202
+ ...(domain.selection ? { selection: domain.selection } : {}),
203
+ dependencies: {
204
+ mode: "lazy",
205
+ dependsOn: dependencies,
206
+ resolver: { inputKey: key },
207
+ },
208
+ };
209
+ }
210
+ return null;
211
+ }
212
+
213
+ function dependencyCaseCount(
214
+ dependencies: readonly string[],
215
+ domainsByKey: Record<string, InputDomainDescriptor>,
216
+ ): number {
217
+ return dependencies
218
+ .map((key) => finiteValuesForDependency(key, domainsByKey[key]).length)
219
+ .reduce((total, count) => total * count, 1);
220
+ }
221
+
222
+ function unsupportedDefaultInputError(key: string, collector: InputCollector) {
223
+ return new Error(
224
+ `Interaction input '${key}' uses a '${collector.kind}' collector without a default-renderable domain or target metadata. ` +
225
+ "Use formInput.choice(...), formInput.choiceList(...), formInput.number(...), formInput.resourceMap(...), cardInput(...), or boardInput(...). " +
226
+ "For custom payloads, use a custom interaction surface with paramsSchema instead of the default InteractionForm.",
227
+ );
228
+ }
229
+
230
+ export function interactionInputsOf<
231
+ DomainState extends CollectorState,
232
+ Manifest extends ManifestContract<TableOfState<DomainState>>,
233
+ >(
234
+ interaction: AnyInteractionSpec<DomainState, Manifest>,
235
+ ): Record<string, InputCollector> {
236
+ return interaction.inputs as Record<string, InputCollector>;
237
+ }
238
+
239
+ export function collectEligibleTargets<
240
+ DomainState extends CollectorState,
241
+ Manifest extends ManifestContract<TableOfState<DomainState>>,
242
+ PlayerId extends string,
243
+ >(
244
+ interaction: AnyInteractionSpec<DomainState, Manifest>,
245
+ domainState: DomainState,
246
+ playerId: PlayerId,
247
+ options: CollectorProjectionOptions<DomainState> = {},
248
+ ): Record<string, string[]> {
249
+ const collectors = interactionInputsOf(interaction);
250
+ const result: Record<string, string[]> = {};
251
+ let queriesLazy: TableQueriesOfState<DomainState> | null =
252
+ options.queries ?? null;
253
+ const queries = () =>
254
+ (queriesLazy ??= createStateQueries(
255
+ domainState as unknown as { table: CollectorState["table"] },
256
+ ) as unknown as TableQueriesOfState<DomainState>);
257
+ for (const [key, collector] of Object.entries(collectors)) {
258
+ if (!collector.eligibleTargets) continue;
259
+ const cacheKey = options.eligibleTargetCachePrefix
260
+ ? `${options.eligibleTargetCachePrefix}:${key}`
261
+ : null;
262
+ const cached = cacheKey
263
+ ? options.eligibleTargetCache?.get(cacheKey)
264
+ : undefined;
265
+ if (cached) {
266
+ result[key] = cached;
267
+ continue;
268
+ }
269
+ const targets = collector.eligibleTargets(
270
+ domainState as unknown as Parameters<
271
+ NonNullable<typeof collector.eligibleTargets>
272
+ >[0],
273
+ playerId as unknown as Parameters<
274
+ NonNullable<typeof collector.eligibleTargets>
275
+ >[1],
276
+ queries() as unknown as Parameters<
277
+ NonNullable<typeof collector.eligibleTargets>
278
+ >[2],
279
+ );
280
+ const resolved = Array.from(targets as ReadonlyArray<unknown>).map((t) =>
281
+ String(t),
282
+ );
283
+ result[key] = resolved;
284
+ if (cacheKey) {
285
+ options.eligibleTargetCache?.set(cacheKey, resolved);
286
+ }
287
+ }
288
+ return result;
289
+ }
290
+
291
+ export function collectInputMetadata<
292
+ DomainState extends CollectorState,
293
+ Manifest extends ManifestContract<TableOfState<DomainState>>,
294
+ >(
295
+ interaction: AnyInteractionSpec<DomainState, Manifest>,
296
+ ): Record<
297
+ string,
298
+ {
299
+ kind: string;
300
+ targetKind?: string;
301
+ boardId?: string;
302
+ zoneId?: string;
303
+ zoneIds?: readonly string[];
304
+ }
305
+ > {
306
+ const collectors = interactionInputsOf(interaction);
307
+ return Object.fromEntries(
308
+ Object.entries(collectors).map(([key, collector]) => [
309
+ key,
310
+ {
311
+ kind: collector.kind,
312
+ ...collectTargetDomainMetadata(collector),
313
+ },
314
+ ]),
315
+ );
316
+ }
317
+
318
+ export function collectInputDomains<
319
+ DomainState extends CollectorState,
320
+ Manifest extends ManifestContract<TableOfState<DomainState>>,
321
+ PlayerId extends string,
322
+ >(
323
+ interaction: AnyInteractionSpec<DomainState, Manifest>,
324
+ domainState: DomainState,
325
+ playerId: PlayerId,
326
+ eligibleTargets: Record<string, string[]>,
327
+ options: CollectorProjectionOptions<DomainState> = {},
328
+ ): Record<string, InputDomainDescriptor> {
329
+ const collectors = interactionInputsOf(interaction);
330
+ let queriesLazy: TableQueriesOfState<DomainState> | null =
331
+ options.queries ?? null;
332
+ const queries = () =>
333
+ (queriesLazy ??= createStateQueries(
334
+ domainState as unknown as { table: CollectorState["table"] },
335
+ ) as unknown as TableQueriesOfState<DomainState>);
336
+ let derivedLazy: DerivedResolver | null = options.derived ?? null;
337
+ const derived = () =>
338
+ (derivedLazy ??= createDerivedResolver(domainState, { q: queries() }));
339
+ const result: Record<string, InputDomainDescriptor> = {};
340
+ for (const [key, collector] of Object.entries(collectors)) {
341
+ if (collector.kind === "rng") {
342
+ continue;
343
+ }
344
+ if (collector.kind === "prompt") {
345
+ const optionFactory = collector.meta?.options;
346
+ if (!optionFactory) continue;
347
+ result[key] = {
348
+ type: "choice",
349
+ choices: optionFactory(domainState, playerId, queries() as unknown).map(
350
+ (option) => ({
351
+ value: String(option.id),
352
+ label: option.label ?? String(option.id),
353
+ ...(() => {
354
+ const issue = collector.validateTarget?.(
355
+ domainState,
356
+ playerId,
357
+ queries() as unknown,
358
+ String(option.id),
359
+ );
360
+ return issue
361
+ ? {
362
+ disabled: true,
363
+ disabledReason: issue.message ?? issue.errorCode,
364
+ }
365
+ : {};
366
+ })(),
367
+ }),
368
+ ),
369
+ };
370
+ continue;
371
+ }
372
+ if (collector.domain) {
373
+ const domainProjector = collector.domain;
374
+ const dependencies = collector.dependsOn ?? [];
375
+ if (
376
+ options.includeEligibleTargets === false &&
377
+ !Object.prototype.hasOwnProperty.call(eligibleTargets, key)
378
+ ) {
379
+ const lazyDomain = lazyTargetDomainFromCollectorMetadata(
380
+ key,
381
+ collector,
382
+ dependencies,
383
+ );
384
+ if (lazyDomain) {
385
+ result[key] = lazyDomain;
386
+ continue;
387
+ }
388
+ }
389
+ const baseDomain = withSelection(
390
+ domainProjector(
391
+ domainState,
392
+ playerId as string,
393
+ queries() as unknown,
394
+ derived(),
395
+ {},
396
+ ),
397
+ collector,
398
+ );
399
+ if (dependencies.length === 0) {
400
+ result[key] = baseDomain;
401
+ continue;
402
+ }
403
+ const caseCount = dependencyCaseCount(dependencies, result);
404
+ const lazyDomain =
405
+ caseCount > MAX_EAGER_DEPENDENT_DOMAIN_CASES
406
+ ? toLazyTargetDomain(key, baseDomain, dependencies)
407
+ : null;
408
+ if (lazyDomain) {
409
+ result[key] = lazyDomain;
410
+ continue;
411
+ }
412
+ const dependentCases = cartesianDependencyTuples(
413
+ dependencies,
414
+ result,
415
+ ).map((values) => ({
416
+ when: values,
417
+ domain: withoutDependentProjection(
418
+ withSelection(
419
+ domainProjector(
420
+ domainState,
421
+ playerId as string,
422
+ queries() as unknown,
423
+ derived(),
424
+ values,
425
+ ),
426
+ collector,
427
+ ),
428
+ ),
429
+ }));
430
+ result[key] = {
431
+ ...baseDomain,
432
+ dependencies: {
433
+ mode: "eager",
434
+ dependentCases,
435
+ },
436
+ } as InputDomainDescriptor;
437
+ continue;
438
+ }
439
+ throw unsupportedDefaultInputError(key, collector);
440
+ }
441
+ return result;
442
+ }
443
+
444
+ export function collectInteractionInputs<
445
+ DomainState extends CollectorState,
446
+ Manifest extends ManifestContract<TableOfState<DomainState>>,
447
+ PlayerId extends string,
448
+ >(
449
+ interaction: AnyInteractionSpec<DomainState, Manifest>,
450
+ domainState: DomainState,
451
+ playerId: PlayerId,
452
+ options: CollectorProjectionOptions<DomainState> = {},
453
+ ): InteractionInputDescriptorShape[] {
454
+ const collectors = interactionInputsOf(interaction);
455
+ const dependencyKeys = new Set(
456
+ Object.values(collectors).flatMap((collector) => [
457
+ ...(collector.dependsOn ?? []),
458
+ ]),
459
+ );
460
+ const collectedEligibleTargets =
461
+ options.includeEligibleTargets === false && dependencyKeys.size === 0
462
+ ? {}
463
+ : collectEligibleTargets(interaction, domainState, playerId, options);
464
+ const eligibleTargets =
465
+ options.includeEligibleTargets === false
466
+ ? Object.fromEntries(
467
+ Object.entries(collectedEligibleTargets).filter(([key]) =>
468
+ dependencyKeys.has(key),
469
+ ),
470
+ )
471
+ : collectedEligibleTargets;
472
+ const domainsByKey = collectInputDomains(
473
+ interaction,
474
+ domainState,
475
+ playerId,
476
+ eligibleTargets,
477
+ options,
478
+ );
479
+ let queriesLazy: TableQueriesOfState<DomainState> | null =
480
+ options.queries ?? null;
481
+ const queries = () =>
482
+ (queriesLazy ??= createStateQueries(
483
+ domainState as unknown as { table: CollectorState["table"] },
484
+ ) as unknown as TableQueriesOfState<DomainState>);
485
+ let derivedLazy: DerivedResolver | null = options.derived ?? null;
486
+ const derived = () =>
487
+ (derivedLazy ??= createDerivedResolver(domainState, { q: queries() }));
488
+ return Object.entries(collectors).flatMap(([key, collector]) => {
489
+ if (collector.kind === "rng") {
490
+ return [];
491
+ }
492
+ const domain = domainsByKey[key];
493
+ if (!domain && collector.kind === "prompt") {
494
+ return [];
495
+ }
496
+ if (!domain) {
497
+ throw unsupportedDefaultInputError(key, collector);
498
+ }
499
+ const dynamicDefault = collector.resolveDefaultValue?.(
500
+ domainState,
501
+ playerId as string,
502
+ queries() as unknown,
503
+ derived(),
504
+ domain,
505
+ );
506
+ const defaultValue =
507
+ "defaultValue" in collector ? collector.defaultValue : dynamicDefault;
508
+ warnConcreteDependentChoiceDefault({
509
+ inputKey: key,
510
+ collector,
511
+ domain,
512
+ defaultValue,
513
+ dependencyKeys,
514
+ });
515
+ return [
516
+ {
517
+ key,
518
+ kind: collector.kind,
519
+ domain,
520
+ ...(defaultValue !== undefined ? { defaultValue } : {}),
521
+ },
522
+ ];
523
+ });
524
+ }
525
+
526
+ const concreteDependentChoiceDefaultWarnings = new Set<string>();
527
+
528
+ function shouldEmitAuthoringWarning(): boolean {
529
+ const env = (
530
+ globalThis as typeof globalThis & {
531
+ process?: { env?: Record<string, string | undefined> };
532
+ }
533
+ ).process?.env;
534
+ return (
535
+ env?.NODE_ENV !== "production" &&
536
+ env?.DREAMBOARD_SUPPRESS_AUTHORING_WARNINGS !== "1"
537
+ );
538
+ }
539
+
540
+ function warnConcreteDependentChoiceDefault({
541
+ inputKey,
542
+ collector,
543
+ domain,
544
+ defaultValue,
545
+ dependencyKeys,
546
+ }: {
547
+ inputKey: string;
548
+ collector: InputCollector;
549
+ domain: InputDomainDescriptor;
550
+ defaultValue: unknown;
551
+ dependencyKeys: ReadonlySet<string>;
552
+ }): void {
553
+ if (
554
+ !shouldEmitAuthoringWarning() ||
555
+ !dependencyKeys.has(inputKey) ||
556
+ collector.kind !== "form" ||
557
+ domain.type !== "choice" ||
558
+ defaultValue === undefined ||
559
+ defaultValue === null
560
+ ) {
561
+ return;
562
+ }
563
+
564
+ const warningKey = `${inputKey}:${String(defaultValue)}`;
565
+ if (concreteDependentChoiceDefaultWarnings.has(warningKey)) {
566
+ return;
567
+ }
568
+ concreteDependentChoiceDefaultWarnings.add(warningKey);
569
+ console.warn(
570
+ `[dreamboard] Form choice input "${inputKey}" feeds another collector and defaulted to "${String(
571
+ defaultValue,
572
+ )}". Dependent choices that select the object an action applies to should usually use defaultValue: () => undefined so collection stays explicit.`,
573
+ );
574
+ }
575
+
576
+ export function collectFirstCardZoneId<
577
+ DomainState extends CollectorState,
578
+ Manifest extends ManifestContract<TableOfState<DomainState>>,
579
+ >(interaction: AnyInteractionSpec<DomainState, Manifest>): string | undefined {
580
+ const collectors = interactionInputsOf(interaction);
581
+ for (const collector of Object.values(collectors)) {
582
+ if (collector.kind === "card") {
583
+ if (collector.meta.zoneId.length > 0) {
584
+ return collector.meta.zoneId;
585
+ }
586
+ }
587
+ }
588
+ return undefined;
589
+ }
590
+
591
+ export function collectCardZoneIds<
592
+ DomainState extends CollectorState,
593
+ Manifest extends ManifestContract<TableOfState<DomainState>>,
594
+ >(interaction: AnyInteractionSpec<DomainState, Manifest>): readonly string[] {
595
+ const collectors = interactionInputsOf(interaction);
596
+ const zoneIds = new Set<string>();
597
+ for (const collector of Object.values(collectors)) {
598
+ if (collector.kind === "card") {
599
+ for (const zoneId of collector.meta.zoneIds ?? [collector.meta.zoneId]) {
600
+ if (zoneId.length > 0) zoneIds.add(zoneId);
601
+ }
602
+ }
603
+ }
604
+ return [...zoneIds];
605
+ }
606
+
607
+ export function collectPromptOptions(
608
+ interaction: {
609
+ inputs: Record<string, InputCollector>;
610
+ },
611
+ domainState: unknown,
612
+ playerId: string,
613
+ queries: unknown,
614
+ ): Array<{ id: string; label?: string }> | undefined {
615
+ for (const collector of Object.values(interaction.inputs)) {
616
+ if (collector.kind !== "prompt") continue;
617
+ const factory = collector.meta?.options;
618
+ if (!factory) continue;
619
+ return factory(domainState, playerId, queries).map((option) => ({
620
+ id: String(option.id),
621
+ ...(option.label !== undefined ? { label: option.label } : {}),
622
+ }));
623
+ }
624
+ return undefined;
625
+ }
626
+
627
+ export function parseInteractionParams<
628
+ DomainState extends CollectorState,
629
+ Manifest extends ManifestContract<TableOfState<DomainState>>,
630
+ >(
631
+ interaction: AnyInteractionSpec<DomainState, Manifest>,
632
+ rawParams: unknown,
633
+ options: { skipRng?: boolean; playerId?: string } = {},
634
+ ):
635
+ | { ok: true; params: Record<string, unknown> }
636
+ | { ok: false; message: string } {
637
+ const collectors = interactionInputsOf(interaction);
638
+ const record = (rawParams ?? {}) as Record<string, unknown>;
639
+ if (options.skipRng && interaction.paramsSchema) {
640
+ const result = interaction.paramsSchema.safeParse(record);
641
+ if (!result.success) {
642
+ return {
643
+ ok: false,
644
+ message: result.error.issues
645
+ .map((issue) => formatIssue("params", issue))
646
+ .join("; "),
647
+ };
648
+ }
649
+ return { ok: true, params: result.data as Record<string, unknown> };
650
+ }
651
+ const parsed: Record<string, unknown> = {};
652
+ const issues: string[] = [];
653
+ for (const [key, collector] of Object.entries(collectors)) {
654
+ if (collector.kind === "rng" && options.skipRng) continue;
655
+ const rawValueBase =
656
+ record[key] === undefined && "defaultValue" in collector
657
+ ? collector.defaultValue
658
+ : record[key];
659
+ const rawValue =
660
+ collector.kind === "board-space" &&
661
+ collector.meta?.valueKind === "player-board-space" &&
662
+ typeof rawValueBase === "string" &&
663
+ options.playerId
664
+ ? {
665
+ boardId: collector.meta.boardId,
666
+ playerId: options.playerId,
667
+ spaceId: rawValueBase,
668
+ }
669
+ : rawValueBase;
670
+ const result = collector.schema.safeParse(rawValue);
671
+ if (!result.success) {
672
+ for (const issue of result.error.issues) {
673
+ issues.push(formatIssue(`params.${key}`, issue));
674
+ }
675
+ continue;
676
+ }
677
+ parsed[key] = result.data;
678
+ }
679
+ if (issues.length > 0) {
680
+ return { ok: false, message: issues.join("; ") };
681
+ }
682
+ return { ok: true, params: parsed };
683
+ }
684
+
685
+ export function prepareInteractionProjectionParams<
686
+ DomainState extends CollectorState,
687
+ Manifest extends ManifestContract<TableOfState<DomainState>>,
688
+ >(
689
+ interaction: AnyInteractionSpec<DomainState, Manifest>,
690
+ rawParams: unknown,
691
+ ): Record<string, unknown> {
692
+ const collectors = interactionInputsOf(interaction);
693
+ const record = (rawParams ?? {}) as Record<string, unknown>;
694
+ const prepared: Record<string, unknown> = { ...record };
695
+ for (const [key, collector] of Object.entries(collectors)) {
696
+ if (collector.kind === "rng") continue;
697
+ if (record[key] !== undefined) {
698
+ const result = collector.schema.safeParse(record[key]);
699
+ if (result.success) {
700
+ prepared[key] = result.data;
701
+ }
702
+ continue;
703
+ }
704
+ if ("defaultValue" in collector) {
705
+ prepared[key] = collector.defaultValue;
706
+ continue;
707
+ }
708
+ const defaultResult = collector.schema.safeParse(undefined);
709
+ if (defaultResult.success) {
710
+ prepared[key] = defaultResult.data;
711
+ }
712
+ }
713
+ return prepared;
714
+ }
715
+
716
+ export function validateCollectorTargets<
717
+ DomainState extends CollectorState,
718
+ Manifest extends ManifestContract<TableOfState<DomainState>>,
719
+ PlayerId extends string,
720
+ >(
721
+ interaction: AnyInteractionSpec<DomainState, Manifest>,
722
+ domainState: DomainState,
723
+ playerId: PlayerId,
724
+ params: Record<string, unknown>,
725
+ ): ReducerValidationResult {
726
+ const collectors = interactionInputsOf(interaction);
727
+ let queriesLazy: ReturnType<typeof createStateQueries> | null = null;
728
+ const queries = () =>
729
+ (queriesLazy ??= createStateQueries(
730
+ domainState as unknown as { table: CollectorState["table"] },
731
+ ));
732
+ for (const [key, collector] of Object.entries(collectors)) {
733
+ const selectionIssue = validateCollectorSelection(collector, params[key]);
734
+ if (selectionIssue) {
735
+ return makeValidationError(
736
+ selectionIssue.errorCode,
737
+ selectionIssue.message,
738
+ );
739
+ }
740
+ if (!collector.validateTarget) continue;
741
+ const rawTarget = params[key];
742
+ if (rawTarget === null || rawTarget === undefined) continue;
743
+ const dependencyValues = dependencyValuesForCollector(collector, params);
744
+ const targetValues = valuesForCollectorValidation(
745
+ collector.selection,
746
+ rawTarget,
747
+ );
748
+ for (const value of targetValues) {
749
+ const issue = collector.validateTarget(
750
+ domainState as unknown as Parameters<
751
+ typeof collector.validateTarget
752
+ >[0],
753
+ playerId as unknown as Parameters<typeof collector.validateTarget>[1],
754
+ queries() as unknown as Parameters<typeof collector.validateTarget>[2],
755
+ value,
756
+ dependencyValues,
757
+ );
758
+ if (issue) {
759
+ return makeValidationError(issue.errorCode, issue.message);
760
+ }
761
+ }
762
+ }
763
+ return { valid: true };
764
+ }
765
+
766
+ function dependencyValuesForCollector(
767
+ collector: InputCollector,
768
+ params: Record<string, unknown>,
769
+ ): Readonly<Record<string, unknown>> | undefined {
770
+ const dependencies = collector.dependsOn ?? [];
771
+ if (dependencies.length === 0) return undefined;
772
+ return Object.fromEntries(
773
+ dependencies.map((dependencyKey) => [dependencyKey, params[dependencyKey]]),
774
+ );
775
+ }
776
+
777
+ function valuesForCollectorValidation(
778
+ selection: InputSelectionDescriptor | undefined,
779
+ value: unknown,
780
+ ): readonly unknown[] {
781
+ if (selection?.mode === "many") {
782
+ return Array.isArray(value) ? value : [];
783
+ }
784
+ return [value];
785
+ }
786
+
787
+ function validateCollectorSelection(
788
+ collector: InputCollector,
789
+ value: unknown,
790
+ ): { errorCode: string; message?: string } | null {
791
+ const selection = collector.selection;
792
+ if (selection?.mode !== "many") return null;
793
+ if (!Array.isArray(value)) return null;
794
+ if (value.length < selection.min) {
795
+ return {
796
+ errorCode: "INVALID_INPUT_COUNT",
797
+ message: `Input expected at least ${selection.min} value${selection.min === 1 ? "" : "s"}.`,
798
+ };
799
+ }
800
+ if (selection.max !== undefined && value.length > selection.max) {
801
+ return {
802
+ errorCode: "INVALID_INPUT_COUNT",
803
+ message: `Input expected at most ${selection.max} value${selection.max === 1 ? "" : "s"}.`,
804
+ };
805
+ }
806
+ if (selection.distinct) {
807
+ const seen = new Set<string>();
808
+ for (const item of value) {
809
+ const key = stableValueKey(item);
810
+ if (seen.has(key)) {
811
+ return {
812
+ errorCode: "DUPLICATE_INPUT_VALUE",
813
+ message: "Input values must be distinct.",
814
+ };
815
+ }
816
+ seen.add(key);
817
+ }
818
+ }
819
+ return null;
820
+ }
821
+
822
+ function stableValueKey(value: unknown): string {
823
+ if (value === null) return "null";
824
+ switch (typeof value) {
825
+ case "string":
826
+ return `string:${value}`;
827
+ case "number":
828
+ case "boolean":
829
+ case "undefined":
830
+ return `${typeof value}:${String(value)}`;
831
+ default:
832
+ return `json:${JSON.stringify(value)}`;
833
+ }
834
+ }
835
+
836
+ export function findCardInputKey<
837
+ DomainState extends CollectorState,
838
+ Manifest extends ManifestContract<TableOfState<DomainState>>,
839
+ >(interaction: AnyInteractionSpec<DomainState, Manifest>): string | undefined {
840
+ return Object.entries(interactionInputsOf(interaction)).find(
841
+ ([, collector]) => collector.kind === "card",
842
+ )?.[0];
843
+ }
844
+
845
+ export function findCardInputKeyForZone<
846
+ DomainState extends CollectorState,
847
+ Manifest extends ManifestContract<TableOfState<DomainState>>,
848
+ >(
849
+ interaction: AnyInteractionSpec<DomainState, Manifest>,
850
+ zoneId: string,
851
+ ): string | undefined {
852
+ return Object.entries(interactionInputsOf(interaction)).find(
853
+ ([, collector]) =>
854
+ collector.kind === "card" &&
855
+ (collector.meta.zoneIds ?? [collector.meta.zoneId])
856
+ .map(String)
857
+ .includes(zoneId),
858
+ )?.[0];
859
+ }