@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,441 @@
1
+ /**
2
+ * Controlled, presentational hand view.
3
+ *
4
+ * `HandView` does not consume any Dreamboard descriptor or runtime state.
5
+ * Callers supply:
6
+ *
7
+ * - the cards to render
8
+ * - a `stateForCard(card)` function returning a controlled
9
+ * {@link InteractionVisualState}
10
+ * - a `renderCard(card, state)` slot for the visual surface (typically
11
+ * a {@link CardFace})
12
+ *
13
+ * The component emits generic {@link CardIntent} events. A runtime
14
+ * adapter is responsible for turning an `activate`/`drop` intent into a
15
+ * Dreamboard submission.
16
+ *
17
+ * `mobileInteraction` selects the gesture policy:
18
+ * - `direct-activate` — desktop tap and keyboard Enter/Space emit a single
19
+ * `activate` intent.
20
+ * - `drag-to-target` — pointer/keyboard lift initiates a drag against a
21
+ * surrounding `CardDragSurface`. Tap collapses to an `inspecting` state
22
+ * that does not commit. Drop emits an opaque `drop` intent.
23
+ *
24
+ * Layout, gesture, scroll arbitration and tray presentation are documented in
25
+ * `docs/references/ui-sdk-mobile-hand-and-card-interactions.md`.
26
+ */
27
+
28
+ import {
29
+ useCallback,
30
+ useId,
31
+ useMemo,
32
+ type CSSProperties,
33
+ type ReactNode,
34
+ } from "react";
35
+ import { clsx } from "clsx";
36
+ import type { ViewCard } from "../../types/index.js";
37
+ import type {
38
+ CardIntent,
39
+ InteractionVisualState,
40
+ } from "../types/visual-state.js";
41
+ import type {
42
+ HandPresentationMode,
43
+ FanCardPosition,
44
+ } from "./hand-layout-math.js";
45
+ import { useHandPresentation } from "../hooks/useHandPresentation.js";
46
+ import { useHandCardPointer } from "../hooks/useHandCardPointer.js";
47
+ import { useTheme } from "../theme/ThemeProvider.js";
48
+ import { useCardDragSurface } from "./CardDragSurface.js";
49
+ import type { HandInteractionPolicy } from "./hand-pointer-engine.js";
50
+
51
+ export type HandLayoutKind = HandPresentationMode | "stack";
52
+
53
+ export interface HandLayoutPolicy {
54
+ desktop?: HandLayoutKind;
55
+ mobile?: HandLayoutKind;
56
+ }
57
+
58
+ export interface HandViewProps<CardData extends ViewCard = ViewCard> {
59
+ cards: readonly CardData[];
60
+ /**
61
+ * Layout policy. Pass a single layout kind to use the same layout in every
62
+ * viewport, or pass `{ desktop, mobile }` for a measured-width decision.
63
+ * Defaults to `{ desktop: "fan", mobile: "tray" }`.
64
+ */
65
+ layout?: HandLayoutKind | HandLayoutPolicy;
66
+ /**
67
+ * Mobile interaction policy. Defaults to `direct-activate`. Use
68
+ * `drag-to-target` together with a surrounding `CardDragSurface` and one
69
+ * or more `CardDropTargetView` children.
70
+ */
71
+ mobileInteraction?: HandInteractionPolicy;
72
+ /** Returns the controlled visual state for a card. */
73
+ stateForCard?: (
74
+ card: CardData,
75
+ index: number,
76
+ ) => InteractionVisualState | undefined;
77
+ /** Visual surface renderer. */
78
+ renderCard: (
79
+ card: CardData,
80
+ state: InteractionVisualState,
81
+ index: number,
82
+ ) => ReactNode;
83
+ /** Generic UI intent callback. */
84
+ onCardIntent?: (intent: CardIntent<CardData["id"] & string>) => void;
85
+ /** Card width hint for fan geometry. */
86
+ cardSize?: "sm" | "md" | "lg";
87
+ /** Slot for the empty state. Defaults to nothing rendered. */
88
+ renderEmpty?: () => ReactNode;
89
+ /** ARIA label for the hand region. */
90
+ "aria-label"?: string;
91
+ className?: string;
92
+ style?: CSSProperties;
93
+ }
94
+
95
+ const CARD_DIMENSIONS = {
96
+ sm: { width: 80, height: 112 },
97
+ md: { width: 96, height: 144 },
98
+ lg: { width: 128, height: 192 },
99
+ } as const;
100
+
101
+ const EMPTY_STATE: InteractionVisualState = {};
102
+
103
+ interface ResolvedLayoutPolicy {
104
+ desktop: HandPresentationMode;
105
+ mobile: HandPresentationMode;
106
+ /** What we report on `data-layout` for stories/tests. */
107
+ reported: HandLayoutKind;
108
+ }
109
+
110
+ function resolveLayoutPolicy(
111
+ layout: HandViewProps["layout"],
112
+ ): ResolvedLayoutPolicy {
113
+ if (!layout) {
114
+ return { desktop: "fan", mobile: "tray", reported: "fan" };
115
+ }
116
+ if (typeof layout === "string") {
117
+ if (layout === "stack") {
118
+ return { desktop: "strip", mobile: "strip", reported: "stack" };
119
+ }
120
+ return { desktop: layout, mobile: layout, reported: layout };
121
+ }
122
+ const desktop =
123
+ layout.desktop === "stack" ? "strip" : (layout.desktop ?? "fan");
124
+ const mobile =
125
+ layout.mobile === "stack" ? "strip" : (layout.mobile ?? "tray");
126
+ return {
127
+ desktop,
128
+ mobile,
129
+ reported: (layout.desktop ?? layout.mobile ?? "fan") as HandLayoutKind,
130
+ };
131
+ }
132
+
133
+ export function HandView<CardData extends ViewCard = ViewCard>({
134
+ cards,
135
+ layout,
136
+ mobileInteraction = "direct-activate",
137
+ stateForCard,
138
+ renderCard,
139
+ onCardIntent,
140
+ cardSize = "md",
141
+ renderEmpty,
142
+ "aria-label": ariaLabel = "Your hand",
143
+ className,
144
+ style,
145
+ }: HandViewProps<CardData>) {
146
+ const policy = useMemo(() => resolveLayoutPolicy(layout), [layout]);
147
+ const dims = CARD_DIMENSIONS[cardSize];
148
+ const regionId = useId();
149
+ const theme = useTheme();
150
+ const reducedMotion = theme.motion.reducedMotion === "true";
151
+
152
+ const presentation = useHandPresentation({
153
+ cardCount: cards.length,
154
+ cardWidth: dims.width,
155
+ cardHeight: dims.height,
156
+ desktop: policy.desktop,
157
+ mobile: policy.mobile,
158
+ });
159
+
160
+ const surface = useCardDragSurface();
161
+ const surfaceController = surface?.controller ?? null;
162
+
163
+ const emit = useCallback(
164
+ (intent: CardIntent<CardData["id"] & string>) => {
165
+ onCardIntent?.(intent);
166
+ },
167
+ [onCardIntent],
168
+ );
169
+
170
+ const items = useMemo(
171
+ () =>
172
+ cards.map((card, index) => {
173
+ const baseState = stateForCard?.(card, index) ?? EMPTY_STATE;
174
+ return { card, index, baseState };
175
+ }),
176
+ [cards, stateForCard],
177
+ );
178
+
179
+ const renderLiftedCard = useCallback(
180
+ (cardId: string): ReactNode => {
181
+ const found = items.find((item) => (item.card.id as string) === cardId);
182
+ if (!found) return null;
183
+ return renderCard(found.card, found.baseState, found.index);
184
+ },
185
+ [items, renderCard],
186
+ );
187
+
188
+ const resolveCardLabel = useCallback(
189
+ (cardId: string): string | null => {
190
+ const found = items.find((item) => (item.card.id as string) === cardId);
191
+ if (!found) return null;
192
+ const card = found.card as ViewCard;
193
+ return (
194
+ card.name ??
195
+ (card.properties as { title?: string } | undefined)?.title ??
196
+ null
197
+ );
198
+ },
199
+ [items],
200
+ );
201
+
202
+ const pointer = useHandCardPointer({
203
+ onIntent: (intent) => emit(intent as CardIntent<CardData["id"] & string>),
204
+ policy: mobileInteraction,
205
+ surface: surfaceController,
206
+ renderLiftedCard,
207
+ resolveCardLabel,
208
+ });
209
+
210
+ if (cards.length === 0) {
211
+ return (
212
+ <div
213
+ ref={presentation.containerRef}
214
+ data-dreamboard-hand-view=""
215
+ data-layout={policy.reported}
216
+ data-mode="empty"
217
+ data-empty="true"
218
+ data-mobile-interaction={mobileInteraction}
219
+ role="group"
220
+ aria-label={`${ariaLabel} - empty`}
221
+ className={clsx(
222
+ "relative w-full min-w-0 flex items-end justify-center py-4 sm:py-6",
223
+ className,
224
+ )}
225
+ style={style}
226
+ >
227
+ {renderEmpty ? renderEmpty() : null}
228
+ </div>
229
+ );
230
+ }
231
+
232
+ const ariaCount = `${ariaLabel} - ${cards.length} card${cards.length === 1 ? "" : "s"}`;
233
+
234
+ const surfacePhase = surfaceController?.phase ?? "idle";
235
+ const surfaceActiveCardId = surfaceController?.activeCardId ?? null;
236
+
237
+ const renderItem = (
238
+ {
239
+ card,
240
+ index,
241
+ baseState,
242
+ }: { card: CardData; index: number; baseState: InteractionVisualState },
243
+ extra: { positioned?: FanCardPosition } = {},
244
+ ) => {
245
+ const cardId = card.id as string;
246
+ const isPointerActive = pointer.activeCardId === cardId;
247
+ const isSurfaceActive = surfaceActiveCardId === cardId;
248
+ const previewing =
249
+ isPointerActive && pointer.recognitionState.kind === "preview"
250
+ ? true
251
+ : baseState.previewing;
252
+ const isInspecting = isSurfaceActive && surfacePhase === "inspecting";
253
+ const isDragging = isSurfaceActive && surfacePhase === "dragging";
254
+ const isSettling = isSurfaceActive && surfacePhase === "settling";
255
+ const isReturning = isSurfaceActive && surfacePhase === "returning";
256
+ // Hide the source slot whenever the portal overlay is showing this card
257
+ // (drag, settle, or return). The overlay is the visible card during
258
+ // those phases — leaving the source visible duplicates it.
259
+ const isLifted = isDragging || isSettling || isReturning;
260
+ const state: InteractionVisualState = {
261
+ ...baseState,
262
+ previewing,
263
+ };
264
+ const eligible = state.eligible ?? false;
265
+ const disabled = state.disabled ?? false;
266
+ const bindings = pointer.bindCard({
267
+ cardId,
268
+ eligible,
269
+ disabled,
270
+ });
271
+ const positioned = extra.positioned;
272
+ const transformParts: string[] = [];
273
+ if (positioned) {
274
+ transformParts.push(`translateY(${positioned.translateY}px)`);
275
+ transformParts.push(`rotate(${positioned.rotate}deg)`);
276
+ }
277
+ if (isInspecting && !reducedMotion) {
278
+ transformParts.push("translateY(-12px)", "scale(1.04)");
279
+ }
280
+ const transform = transformParts.length
281
+ ? transformParts.join(" ")
282
+ : undefined;
283
+ const liftVisualOpacity = isLifted ? 0.25 : 1;
284
+ const inspectShadow =
285
+ isInspecting && !reducedMotion
286
+ ? "0 14px 28px rgba(0,0,0,0.18)"
287
+ : undefined;
288
+ return (
289
+ <div
290
+ key={card.id}
291
+ role="gridcell"
292
+ aria-rowindex={1}
293
+ aria-colindex={index + 1}
294
+ id={`${regionId}-${index}`}
295
+ tabIndex={disabled ? -1 : 0}
296
+ data-lifted={isLifted ? "true" : undefined}
297
+ data-returning={isReturning ? "true" : undefined}
298
+ data-inspecting={isInspecting ? "true" : undefined}
299
+ {...bindings}
300
+ style={{
301
+ ...bindings.style,
302
+ opacity: liftVisualOpacity,
303
+ boxShadow: inspectShadow,
304
+ ...(positioned
305
+ ? {
306
+ position: "absolute",
307
+ left: positioned.left,
308
+ bottom: 0,
309
+ zIndex: isInspecting ? 50 : positioned.zIndex,
310
+ transform,
311
+ transformOrigin: "bottom center",
312
+ transition: reducedMotion
313
+ ? "none"
314
+ : "transform 180ms ease-out, box-shadow 180ms ease-out",
315
+ }
316
+ : transform
317
+ ? {
318
+ transform,
319
+ transformOrigin: "bottom center",
320
+ transition: reducedMotion
321
+ ? "none"
322
+ : "transform 180ms ease-out, box-shadow 180ms ease-out",
323
+ }
324
+ : null),
325
+ }}
326
+ >
327
+ {renderCard(card, state, index)}
328
+ </div>
329
+ );
330
+ };
331
+
332
+ if (presentation.mode === "fan" || presentation.mode === "compressed-fan") {
333
+ const totalHeight = dims.height + 24;
334
+ return (
335
+ <div
336
+ ref={presentation.containerRef}
337
+ data-dreamboard-hand-view=""
338
+ data-layout={policy.reported}
339
+ data-mode={presentation.mode}
340
+ data-mobile-interaction={mobileInteraction}
341
+ data-reduced-motion={reducedMotion ? "true" : undefined}
342
+ role="grid"
343
+ aria-label={ariaCount}
344
+ aria-rowcount={1}
345
+ aria-colcount={cards.length}
346
+ className={clsx(
347
+ "relative w-full min-w-0 flex items-end justify-center py-4 sm:py-6 overflow-visible",
348
+ className,
349
+ )}
350
+ style={{
351
+ touchAction: "pan-x",
352
+ ...style,
353
+ }}
354
+ >
355
+ <div
356
+ role="row"
357
+ className="relative"
358
+ style={{
359
+ width: presentation.totalWidth,
360
+ height: totalHeight,
361
+ }}
362
+ >
363
+ {items.map((item, index) =>
364
+ renderItem(item, {
365
+ positioned: presentation.fanPositions[index],
366
+ }),
367
+ )}
368
+ </div>
369
+ </div>
370
+ );
371
+ }
372
+
373
+ if (presentation.mode === "tray") {
374
+ return (
375
+ <div
376
+ ref={presentation.containerRef}
377
+ data-dreamboard-hand-view=""
378
+ data-layout={policy.reported}
379
+ data-mode="tray"
380
+ data-mobile-interaction={mobileInteraction}
381
+ data-reduced-motion={reducedMotion ? "true" : undefined}
382
+ role="grid"
383
+ aria-label={ariaCount}
384
+ aria-rowcount={1}
385
+ aria-colcount={cards.length}
386
+ className={clsx("relative w-full min-w-0", className)}
387
+ style={style}
388
+ >
389
+ <div
390
+ role="row"
391
+ className={clsx(
392
+ "flex items-end gap-3 overflow-x-auto px-4",
393
+ "snap-x scroll-px-4",
394
+ )}
395
+ style={{
396
+ touchAction: pointer.scrollLocked ? "none" : "pan-x",
397
+ paddingTop: 12,
398
+ paddingBottom: "max(1rem, env(safe-area-inset-bottom, 0px))",
399
+ scrollBehavior: reducedMotion ? "auto" : "smooth",
400
+ }}
401
+ >
402
+ {items.map((item) => (
403
+ <div
404
+ key={item.card.id}
405
+ className="snap-start shrink-0"
406
+ role="presentation"
407
+ >
408
+ {renderItem(item)}
409
+ </div>
410
+ ))}
411
+ </div>
412
+ </div>
413
+ );
414
+ }
415
+
416
+ // strip / stack — flow-based layout that keeps every card individually
417
+ // touch-usable. Stack layout draws cards with a small horizontal cascade.
418
+ return (
419
+ <div
420
+ ref={presentation.containerRef}
421
+ data-dreamboard-hand-view=""
422
+ data-layout={policy.reported}
423
+ data-mode={presentation.mode}
424
+ data-mobile-interaction={mobileInteraction}
425
+ data-reduced-motion={reducedMotion ? "true" : undefined}
426
+ role="grid"
427
+ aria-label={ariaCount}
428
+ aria-rowcount={1}
429
+ aria-colcount={cards.length}
430
+ className={clsx(
431
+ "relative w-full min-w-0 flex items-end justify-center py-4 sm:py-6",
432
+ className,
433
+ )}
434
+ style={{ touchAction: "pan-x", ...style }}
435
+ >
436
+ <div role="row" className="flex flex-wrap justify-center gap-1">
437
+ {items.map((item) => renderItem(item))}
438
+ </div>
439
+ </div>
440
+ );
441
+ }