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