@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,468 @@
1
+ /**
2
+ * Player hand with adaptive overlap, automatic drawer fallback, and keyboard navigation.
3
+ * For full control, use the `useHandLayout` hook directly.
4
+ */
5
+
6
+ import { clsx } from "clsx";
7
+ import { useCallback, useState } from "react";
8
+ import {
9
+ useHandLayout,
10
+ type CardSize,
11
+ type HandLayout,
12
+ } from "../hooks/useHandLayout.js";
13
+ import { CardFace, type ViewCard } from "./Card.js";
14
+ import { HandDock } from "./HandDock.js";
15
+ import type { ReactNode } from "react";
16
+ import { useTheme } from "../theme/ThemeProvider.js";
17
+
18
+ export interface HandCardRenderProps<CardData extends ViewCard = ViewCard> {
19
+ card: CardData;
20
+ index: number;
21
+ isHovered: boolean;
22
+ isSelected: boolean;
23
+ x: number;
24
+ y: number;
25
+ zIndex: number;
26
+ cardDimensions: { width: number; height: number };
27
+ }
28
+
29
+ export interface HandDrawerRenderProps<CardData extends ViewCard = ViewCard> {
30
+ cards: readonly CardData[];
31
+ selectedIds: readonly string[];
32
+ cardCount: number;
33
+ selectedCount: number;
34
+ disabled: boolean;
35
+ cardDimensions: { width: number; height: number };
36
+ }
37
+
38
+ export interface HandEmptyRenderProps {
39
+ layout: HandLayout;
40
+ }
41
+
42
+ export interface HandContainerRenderProps {
43
+ totalWidth: number;
44
+ totalHeight: number;
45
+ cardDimensions: { width: number; height: number };
46
+ children: ReactNode;
47
+ onMouseMove: (e: React.MouseEvent<HTMLDivElement>) => void;
48
+ onMouseLeave: () => void;
49
+ }
50
+
51
+ export interface HandProps<CardData extends ViewCard = ViewCard> {
52
+ cards: readonly CardData[];
53
+ selectedIds?: readonly string[];
54
+ disabled?: boolean;
55
+ cardSize?: CardSize;
56
+ layout?: HandLayout;
57
+ "aria-label"?: string;
58
+ /**
59
+ * Called when a card is clicked. Wired into the default `renderCard`;
60
+ * ignored when the caller supplies a custom `renderCard`.
61
+ */
62
+ onCardClick?: (cardId: string) => void;
63
+ /** Per-card content renderer forwarded to the default `<CardFace>`. */
64
+ renderCardContent?: (card: CardData) => ReactNode;
65
+ /** Defaults to a positioned `<CardFace>` honouring click/selection state. */
66
+ renderCard?: (props: HandCardRenderProps<CardData>) => ReactNode;
67
+ /**
68
+ * Custom drawer fallback for large hands (when `useHandLayout` switches
69
+ * to drawer mode). When omitted, `Hand` renders a `HandDock` bottom drawer
70
+ * automatically.
71
+ */
72
+ renderDrawer?: (props: HandDrawerRenderProps<CardData>) => ReactNode;
73
+ /** Defaults to a subtle "No cards in hand" placeholder. */
74
+ renderEmpty?: (props: HandEmptyRenderProps) => ReactNode;
75
+ renderContainer?: (props: HandContainerRenderProps) => ReactNode;
76
+ className?: string;
77
+ }
78
+
79
+ const EMPTY_SELECTED_IDS: readonly string[] = [];
80
+
81
+ // DefaultDrawer is intentionally omitted — when no `renderDrawer` prop is
82
+ // provided, `Hand` automatically renders a `HandDock` bottom drawer instead
83
+ // (see the drawer-mode branch in the `Hand` component below).
84
+ // This gives large hands a proper mobile-friendly expandable UI for free.
85
+
86
+ function DefaultEmpty(): ReactNode {
87
+ const theme = useTheme();
88
+ return (
89
+ <span
90
+ style={{
91
+ fontFamily: theme.typography.fontFamily.body,
92
+ fontSize: theme.typography.fontSize.xs,
93
+ color: theme.semantic.text.disabled,
94
+ }}
95
+ >
96
+ No cards in hand
97
+ </span>
98
+ );
99
+ }
100
+
101
+ interface ClickableCardFaceProps<CardData extends ViewCard = ViewCard> {
102
+ card: CardData;
103
+ selected?: boolean;
104
+ disabled?: boolean;
105
+ size?: CardSize;
106
+ onCardClick?: (cardId: CardData["id"]) => void;
107
+ renderContent?: (card: CardData) => ReactNode;
108
+ }
109
+
110
+ function ClickableCardFace<CardData extends ViewCard = ViewCard>({
111
+ card,
112
+ selected,
113
+ disabled,
114
+ size,
115
+ onCardClick,
116
+ renderContent,
117
+ }: ClickableCardFaceProps<CardData>) {
118
+ const interactive = !disabled && Boolean(onCardClick);
119
+ const handleActivate = () => {
120
+ if (!interactive) return;
121
+ onCardClick?.(card.id);
122
+ };
123
+
124
+ return (
125
+ <button
126
+ type="button"
127
+ disabled={disabled}
128
+ aria-pressed={selected ? true : undefined}
129
+ aria-disabled={disabled || undefined}
130
+ tabIndex={disabled ? -1 : 0}
131
+ onClick={handleActivate}
132
+ onKeyDown={(event) => {
133
+ if (event.key === "Enter" || event.key === " ") {
134
+ event.preventDefault();
135
+ handleActivate();
136
+ }
137
+ }}
138
+ className={clsx(
139
+ "border-0 bg-transparent p-0",
140
+ "focus-visible:outline-none",
141
+ interactive ? "cursor-pointer" : "cursor-default",
142
+ )}
143
+ style={{ all: "unset" }}
144
+ >
145
+ <CardFace
146
+ card={card}
147
+ selected={selected}
148
+ disabled={disabled}
149
+ size={size}
150
+ renderContent={renderContent}
151
+ />
152
+ </button>
153
+ );
154
+ }
155
+
156
+ /**
157
+ * Hand component with customizable rendering via render props.
158
+ *
159
+ * `renderCard`, `renderDrawer`, and `renderEmpty` are all optional — the
160
+ * defaults render the built-in `<CardFace>` with the supplied `onCardClick` /
161
+ * `renderCardContent`, a compact drawer fallback on small screens, and a
162
+ * muted empty placeholder. Override any of them when you need a different
163
+ * visual treatment.
164
+ *
165
+ * For complete control over layout and interactions, use the
166
+ * `useHandLayout` hook directly.
167
+ *
168
+ * @example Zero-config
169
+ * ```tsx
170
+ * <Hand cards={cards} onCardClick={(id) => play(id)} />
171
+ * ```
172
+ *
173
+ * @example Custom card content
174
+ * ```tsx
175
+ * <Hand
176
+ * cards={cards}
177
+ * onCardClick={(id) => play(id)}
178
+ * renderCardContent={(card) => <DevCardFace card={card} />}
179
+ * />
180
+ * ```
181
+ */
182
+ export function Hand<CardData extends ViewCard = ViewCard>({
183
+ cards,
184
+ selectedIds = EMPTY_SELECTED_IDS,
185
+ disabled = false,
186
+ cardSize = "md",
187
+ layout = "overlap",
188
+ "aria-label": ariaLabel = "Your hand",
189
+ onCardClick,
190
+ renderCardContent,
191
+ renderCard,
192
+ renderDrawer,
193
+ renderEmpty,
194
+ renderContainer,
195
+ className,
196
+ }: HandProps<CardData>) {
197
+ const defaultRenderCard = useCallback(
198
+ (props: HandCardRenderProps<CardData>): ReactNode => (
199
+ <div
200
+ key={props.card.id}
201
+ className="absolute bottom-0 transition-all duration-150 ease-out"
202
+ style={{
203
+ left: props.x,
204
+ zIndex: props.zIndex,
205
+ transform: `translateY(${props.y}px)`,
206
+ }}
207
+ >
208
+ <ClickableCardFace
209
+ card={props.card}
210
+ selected={props.isSelected}
211
+ disabled={disabled}
212
+ size={cardSize}
213
+ onCardClick={onCardClick}
214
+ renderContent={renderCardContent}
215
+ />
216
+ </div>
217
+ ),
218
+ [cardSize, disabled, onCardClick, renderCardContent],
219
+ );
220
+ const effectiveRenderCard = renderCard ?? defaultRenderCard;
221
+ const effectiveRenderEmpty = renderEmpty ?? DefaultEmpty;
222
+ const cardCount = cards.length;
223
+ const [focusedIndex, setFocusedIndex] = useState(-1);
224
+
225
+ const {
226
+ containerRef,
227
+ cardsContainerRef,
228
+ totalWidth,
229
+ useDrawerMode,
230
+ cardDimensions,
231
+ constants,
232
+ hoveredIndex,
233
+ handleMouseMove,
234
+ handleMouseLeave,
235
+ handleTouchMove,
236
+ handleTouchEnd,
237
+ getCardPosition,
238
+ } = useHandLayout({
239
+ cardCount,
240
+ cardSize,
241
+ layout,
242
+ });
243
+
244
+ const selectedCount = cards.filter((c) => selectedIds.includes(c.id)).length;
245
+
246
+ const handleKeyDown = useCallback(
247
+ (e: React.KeyboardEvent) => {
248
+ if (cardCount === 0) return;
249
+ switch (e.key) {
250
+ case "ArrowLeft":
251
+ e.preventDefault();
252
+ setFocusedIndex((prev) => (prev <= 0 ? cardCount - 1 : prev - 1));
253
+ break;
254
+ case "ArrowRight":
255
+ e.preventDefault();
256
+ setFocusedIndex((prev) => (prev >= cardCount - 1 ? 0 : prev + 1));
257
+ break;
258
+ case "Home":
259
+ e.preventDefault();
260
+ setFocusedIndex(0);
261
+ break;
262
+ case "End":
263
+ e.preventDefault();
264
+ setFocusedIndex(cardCount - 1);
265
+ break;
266
+ }
267
+ },
268
+ [cardCount],
269
+ );
270
+
271
+ // Drawer mode — too many cards to show inline.
272
+ // If the caller provides a custom `renderDrawer`, delegate entirely to it.
273
+ // Otherwise render a `HandDock` bottom drawer so the hand is always
274
+ // reachable on mobile without any extra author wiring.
275
+ if (useDrawerMode && layout === "overlap" && cardCount > 0) {
276
+ if (renderDrawer) {
277
+ return (
278
+ <div
279
+ ref={containerRef}
280
+ className={clsx(
281
+ "relative w-full flex items-center justify-center py-4",
282
+ className,
283
+ )}
284
+ role="group"
285
+ aria-label={`${ariaLabel} - ${cardCount} card${cardCount !== 1 ? "s" : ""}`}
286
+ >
287
+ {renderDrawer({
288
+ cards,
289
+ selectedIds,
290
+ cardCount,
291
+ selectedCount,
292
+ disabled,
293
+ cardDimensions,
294
+ })}
295
+ </div>
296
+ );
297
+ }
298
+
299
+ // Default: HandDock bottom drawer with the cards in a wrap grid.
300
+ // The outer div is kept for the containerRef measurement; HandDock itself
301
+ // is fixed-position and renders outside normal flow.
302
+ return (
303
+ <div
304
+ ref={containerRef}
305
+ className={clsx(
306
+ "relative w-full flex items-center justify-center",
307
+ className,
308
+ )}
309
+ role="group"
310
+ aria-label={`${ariaLabel} - ${cardCount} card${cardCount !== 1 ? "s" : ""}`}
311
+ >
312
+ <HandDock
313
+ label={ariaLabel}
314
+ count={cardCount}
315
+ presentation={{ mode: "drawer", placement: "bottom-center" }}
316
+ >
317
+ <div
318
+ style={{
319
+ display: "flex",
320
+ flexWrap: "wrap",
321
+ gap: 8,
322
+ padding: "4px 0 8px",
323
+ justifyContent: "center",
324
+ }}
325
+ >
326
+ {cards.map((card) => (
327
+ <ClickableCardFace
328
+ key={card.id}
329
+ card={card}
330
+ selected={selectedIds.includes(card.id)}
331
+ disabled={disabled}
332
+ size={cardSize}
333
+ onCardClick={onCardClick}
334
+ renderContent={renderCardContent}
335
+ />
336
+ ))}
337
+ </div>
338
+ </HandDock>
339
+ </div>
340
+ );
341
+ }
342
+
343
+ // Empty hand
344
+ if (cardCount === 0) {
345
+ return (
346
+ <div
347
+ ref={containerRef}
348
+ className={clsx(
349
+ "relative w-full flex items-end justify-center py-4 sm:py-6",
350
+ className,
351
+ )}
352
+ role="group"
353
+ aria-label={`${ariaLabel} - empty`}
354
+ >
355
+ {effectiveRenderEmpty({ layout })}
356
+ </div>
357
+ );
358
+ }
359
+
360
+ const renderedCards = cards.map((card, index) => {
361
+ const isSelected = selectedIds.includes(card.id);
362
+ const isFocused = focusedIndex === index;
363
+ const isHovered = hoveredIndex === index || isFocused;
364
+ const position = getCardPosition(index, isHovered, isSelected);
365
+
366
+ return effectiveRenderCard({
367
+ card,
368
+ index,
369
+ isHovered,
370
+ isSelected,
371
+ x: position.x,
372
+ y: position.y,
373
+ zIndex: position.zIndex,
374
+ cardDimensions,
375
+ });
376
+ });
377
+
378
+ if (layout === "spread") {
379
+ return (
380
+ <div
381
+ ref={containerRef}
382
+ className={clsx(
383
+ "relative w-full flex items-end justify-center py-4 sm:py-6",
384
+ className,
385
+ )}
386
+ role="group"
387
+ aria-label={`${ariaLabel} - ${cardCount} card${cardCount !== 1 ? "s" : ""}`}
388
+ tabIndex={0}
389
+ onKeyDown={handleKeyDown}
390
+ onBlur={() => setFocusedIndex(-1)}
391
+ >
392
+ <div className="flex gap-1 justify-center flex-wrap">
393
+ {renderedCards}
394
+ </div>
395
+ </div>
396
+ );
397
+ }
398
+
399
+ // Overlap/Stack layout
400
+ const totalHeight = cardDimensions.height + constants.hoverLift + 8;
401
+
402
+ const containerProps: HandContainerRenderProps = {
403
+ totalWidth,
404
+ totalHeight,
405
+ cardDimensions,
406
+ children: renderedCards,
407
+ onMouseMove: handleMouseMove,
408
+ onMouseLeave: handleMouseLeave,
409
+ };
410
+
411
+ // Allow custom container rendering
412
+ if (renderContainer) {
413
+ return (
414
+ <div
415
+ ref={containerRef}
416
+ className={clsx(
417
+ "relative w-full flex items-end justify-center py-4 sm:py-6 overflow-visible",
418
+ className,
419
+ )}
420
+ role="group"
421
+ aria-label={`${ariaLabel} - ${cardCount} card${cardCount !== 1 ? "s" : ""}`}
422
+ >
423
+ {renderContainer(containerProps)}
424
+ </div>
425
+ );
426
+ }
427
+
428
+ return (
429
+ <div
430
+ ref={containerRef}
431
+ className={clsx(
432
+ "relative w-full flex items-end justify-center py-4 sm:py-6 overflow-visible",
433
+ className,
434
+ )}
435
+ role="group"
436
+ aria-label={`${ariaLabel} - ${cardCount} card${cardCount !== 1 ? "s" : ""}`}
437
+ tabIndex={0}
438
+ onKeyDown={handleKeyDown}
439
+ onBlur={() => setFocusedIndex(-1)}
440
+ >
441
+ <div
442
+ ref={cardsContainerRef}
443
+ className="relative"
444
+ style={{
445
+ width: layout === "overlap" ? totalWidth : undefined,
446
+ height: totalHeight,
447
+ }}
448
+ onMouseMove={handleMouseMove}
449
+ onMouseLeave={handleMouseLeave}
450
+ onTouchMove={handleTouchMove}
451
+ onTouchEnd={handleTouchEnd}
452
+ >
453
+ {renderedCards}
454
+ </div>
455
+ </div>
456
+ );
457
+ }
458
+
459
+ // Re-export types and hook for users who want full control
460
+ export { useHandLayout } from "../hooks/useHandLayout.js";
461
+ export type {
462
+ UseHandLayoutOptions,
463
+ UseHandLayoutReturn,
464
+ CardPositionProps,
465
+ CardSize,
466
+ HandLayout,
467
+ } from "../hooks/useHandLayout.js";
468
+ export type { ViewCard } from "./Card.js";