@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,79 @@
1
+ import { createContext, useContext, type ReactNode } from "react";
2
+
3
+ export type DiceValue = number | null | undefined;
4
+
5
+ export interface DiceState {
6
+ values: ReadonlyArray<number | undefined> | undefined;
7
+ /** Undefined if any die has not been rolled yet. */
8
+ sum: number | undefined;
9
+ diceCount: number;
10
+ allRolled: boolean;
11
+ }
12
+
13
+ export interface DiceRootProps {
14
+ values?: readonly DiceValue[] | null;
15
+ /** Used when values are not provided. */
16
+ count?: number;
17
+ children: ReactNode;
18
+ }
19
+
20
+ export interface DiceValuesProps {
21
+ children: (state: DiceState) => ReactNode;
22
+ }
23
+
24
+ export interface DiceComponents {
25
+ Root(props: DiceRootProps): ReactNode;
26
+ Values(props: DiceValuesProps): ReactNode;
27
+ }
28
+
29
+ const DiceContext = createContext<DiceState | null>(null);
30
+
31
+ export function normalizeDiceState({
32
+ values,
33
+ count = 2,
34
+ }: {
35
+ values?: readonly DiceValue[] | null;
36
+ count?: number;
37
+ }): DiceState {
38
+ const normalizedValues = values?.map((value) => value ?? undefined);
39
+ const allRolled =
40
+ normalizedValues?.every((value) => value !== undefined) ?? false;
41
+ const sum = allRolled
42
+ ? normalizedValues?.reduce<number>(
43
+ (total, value) => total + (value ?? 0),
44
+ 0,
45
+ )
46
+ : undefined;
47
+
48
+ return {
49
+ values: normalizedValues,
50
+ sum,
51
+ diceCount: normalizedValues?.length ?? count,
52
+ allRolled,
53
+ };
54
+ }
55
+
56
+ export function DiceRoot({ values, count, children }: DiceRootProps) {
57
+ return (
58
+ <DiceContext.Provider value={normalizeDiceState({ values, count })}>
59
+ {children}
60
+ </DiceContext.Provider>
61
+ );
62
+ }
63
+
64
+ export function useDicePrimitiveContext(): DiceState {
65
+ const value = useContext(DiceContext);
66
+ if (!value) {
67
+ throw new Error("Dice primitives must be rendered inside <Dice.Root>.");
68
+ }
69
+ return value;
70
+ }
71
+
72
+ export function DiceValues({ children }: DiceValuesProps) {
73
+ return children(useDicePrimitiveContext());
74
+ }
75
+
76
+ export const Dice: DiceComponents = {
77
+ Root: DiceRoot,
78
+ Values: DiceValues,
79
+ };
@@ -0,0 +1,101 @@
1
+ import {
2
+ cloneElement,
3
+ isValidElement,
4
+ type AriaAttributes,
5
+ type CSSProperties,
6
+ type HTMLAttributes,
7
+ type JSX,
8
+ type ChangeEvent,
9
+ type MouseEvent,
10
+ type PointerEvent,
11
+ type ReactElement,
12
+ type ReactNode,
13
+ } from "react";
14
+
15
+ export type PrimitiveDataAttributes = Record<
16
+ `data-${string}`,
17
+ string | boolean | number | undefined
18
+ >;
19
+
20
+ export type PrimitiveCommonProps = {
21
+ asChild?: boolean;
22
+ children?: ReactNode;
23
+ className?: string;
24
+ style?: CSSProperties;
25
+ } & AriaAttributes &
26
+ PrimitiveDataAttributes;
27
+
28
+ type EventHandler<Event> = (event: Event) => void;
29
+
30
+ export function composeEventHandlers<
31
+ Event extends { defaultPrevented: boolean },
32
+ >(
33
+ authorHandler: EventHandler<Event> | undefined,
34
+ primitiveHandler: EventHandler<Event> | undefined,
35
+ ): EventHandler<Event> | undefined {
36
+ if (!authorHandler) return primitiveHandler;
37
+ if (!primitiveHandler) return authorHandler;
38
+ return (event) => {
39
+ authorHandler(event);
40
+ if (!event.defaultPrevented) {
41
+ primitiveHandler(event);
42
+ }
43
+ };
44
+ }
45
+
46
+ export function renderPrimitive<
47
+ ElementProps extends HTMLAttributes<HTMLElement>,
48
+ >(
49
+ tagName: keyof JSX.IntrinsicElements,
50
+ props: ElementProps & PrimitiveCommonProps,
51
+ ): ReactElement {
52
+ const { asChild, children, ...primitiveProps } = props;
53
+ if (asChild) {
54
+ if (!isValidElement(children)) {
55
+ throw new Error(
56
+ "asChild requires exactly one valid React element child.",
57
+ );
58
+ }
59
+ const element = children as ReactElement<HTMLAttributes<HTMLElement>>;
60
+ const childProps = element.props;
61
+ return cloneElement(element, {
62
+ ...primitiveProps,
63
+ ...childProps,
64
+ className: [primitiveProps.className, childProps.className]
65
+ .filter(Boolean)
66
+ .join(" "),
67
+ style: {
68
+ ...(primitiveProps.style ?? {}),
69
+ ...(childProps.style ?? {}),
70
+ },
71
+ onClick: composeEventHandlers(
72
+ childProps.onClick,
73
+ primitiveProps.onClick as EventHandler<MouseEvent<HTMLElement>>,
74
+ ),
75
+ onPointerDown: composeEventHandlers(
76
+ childProps.onPointerDown,
77
+ primitiveProps.onPointerDown as EventHandler<PointerEvent<HTMLElement>>,
78
+ ),
79
+ onPointerMove: composeEventHandlers(
80
+ childProps.onPointerMove,
81
+ primitiveProps.onPointerMove as EventHandler<PointerEvent<HTMLElement>>,
82
+ ),
83
+ onPointerUp: composeEventHandlers(
84
+ childProps.onPointerUp,
85
+ primitiveProps.onPointerUp as EventHandler<PointerEvent<HTMLElement>>,
86
+ ),
87
+ onPointerCancel: composeEventHandlers(
88
+ childProps.onPointerCancel,
89
+ primitiveProps.onPointerCancel as EventHandler<
90
+ PointerEvent<HTMLElement>
91
+ >,
92
+ ),
93
+ onChange: composeEventHandlers(
94
+ childProps.onChange,
95
+ primitiveProps.onChange as EventHandler<ChangeEvent<HTMLElement>>,
96
+ ),
97
+ });
98
+ }
99
+ const Tag = tagName as keyof JSX.IntrinsicElements;
100
+ return <Tag {...(primitiveProps as object)}>{children}</Tag>;
101
+ }
@@ -0,0 +1,252 @@
1
+ import {
2
+ createContext,
3
+ useContext,
4
+ useEffect,
5
+ useMemo,
6
+ useState,
7
+ type CSSProperties,
8
+ type ReactNode,
9
+ } from "react";
10
+ import { themeToCssVars } from "./css-vars.js";
11
+ import { arcadeTheme } from "./presets/arcade.js";
12
+ import { studioTheme } from "./presets/studio.js";
13
+ import { tabletopTheme } from "./presets/tabletop.js";
14
+ import { mergeTheme, type Theme, type ThemeOverride } from "./tokens.js";
15
+
16
+ /**
17
+ * The set of preset theme ids ships in `@dreamboard-games/sdk/ui`. Authors
18
+ * may register additional ones by passing a fully-resolved {@link Theme}
19
+ * instead of an id.
20
+ */
21
+ export type ThemePresetId = "tabletop" | "arcade" | "studio";
22
+
23
+ const PRESETS: Readonly<Record<ThemePresetId, Theme>> = {
24
+ tabletop: tabletopTheme,
25
+ arcade: arcadeTheme,
26
+ studio: studioTheme,
27
+ };
28
+
29
+ /**
30
+ * Resolve a preset id (or a full theme) into a {@link Theme}.
31
+ *
32
+ * Used internally by {@link ThemeProvider}; exported so authors who
33
+ * compose at the call site (e.g. for a side-by-side preview) can do
34
+ * the same resolution without mounting a provider.
35
+ */
36
+ export function resolveTheme(input: ThemePresetId | Theme | undefined): Theme {
37
+ if (!input) return tabletopTheme;
38
+ if (typeof input === "string") {
39
+ const preset = PRESETS[input];
40
+ if (!preset) {
41
+ throw new Error(
42
+ `[ui-sdk] Unknown theme preset "${input}". Pass a full Theme object or one of: ${Object.keys(
43
+ PRESETS,
44
+ ).join(", ")}.`,
45
+ );
46
+ }
47
+ return preset;
48
+ }
49
+ return input;
50
+ }
51
+
52
+ /** Look up a registered preset by id. Returns `undefined` if missing. */
53
+ export function getThemePreset(id: ThemePresetId): Theme | undefined {
54
+ return PRESETS[id];
55
+ }
56
+
57
+ /**
58
+ * Subset returned by {@link useTheme} — the resolved theme plus a
59
+ * stable copy of its CSS-variable map so consumers can reuse it (for
60
+ * example, to apply tokens to a portal that escapes the provider).
61
+ */
62
+ export interface ThemeContextValue {
63
+ /** Fully resolved theme. */
64
+ readonly theme: Theme;
65
+ /** CSS-var map consumable as `style` on a wrapper. */
66
+ readonly cssVars: CSSProperties;
67
+ }
68
+
69
+ const ThemeCtx = createContext<ThemeContextValue | null>(null);
70
+
71
+ export interface ThemeProviderProps {
72
+ /**
73
+ * Preset id, a full {@link Theme}, or omitted to use the `tabletop`
74
+ * default.
75
+ */
76
+ theme?: ThemePresetId | Theme;
77
+ /**
78
+ * Deep-partial overrides merged onto the resolved base theme. Use
79
+ * for one-off tweaks (e.g. swapping the player palette per game)
80
+ * without writing a full theme.
81
+ */
82
+ override?: ThemeOverride;
83
+ /**
84
+ * Render mode for reduced-motion enforcement.
85
+ *
86
+ * - `auto` (default): respect the user's OS-level preference via
87
+ * `prefers-reduced-motion: reduce`.
88
+ * - `force`: force `motion.reducedMotion = "true"` regardless of the
89
+ * OS preference (useful for screenshot CI).
90
+ * - `ignore`: never override motion (use only when the embedding
91
+ * shell already decides motion behaviour).
92
+ */
93
+ reducedMotion?: "auto" | "force" | "ignore";
94
+ /**
95
+ * Where to mount the wrapper. `block` (default) renders a `div` and
96
+ * applies the CSS variables to it. `none` skips the wrapper entirely
97
+ * — only useful when the parent is already a Dreamboard provider and
98
+ * just needs to publish a different sub-tree (rare).
99
+ */
100
+ as?: "block" | "none";
101
+ /** Extra style merged after the CSS-var map. */
102
+ style?: CSSProperties;
103
+ /** Extra className for the wrapper. */
104
+ className?: string;
105
+ children: ReactNode;
106
+ }
107
+
108
+ function usePrefersReducedMotion(): boolean {
109
+ const [prefers, setPrefers] = useState(false);
110
+ useEffect(() => {
111
+ if (typeof window === "undefined" || !window.matchMedia) return;
112
+ const media = window.matchMedia("(prefers-reduced-motion: reduce)");
113
+ setPrefers(media.matches);
114
+ const handle = (event: MediaQueryListEvent) => setPrefers(event.matches);
115
+ media.addEventListener("change", handle);
116
+ return () => media.removeEventListener("change", handle);
117
+ }, []);
118
+ return prefers;
119
+ }
120
+
121
+ /**
122
+ * Mounts a Dreamboard theme. The provider:
123
+ *
124
+ * 1. Resolves the preset (or accepts a full theme).
125
+ * 2. Applies any deep-partial `override`.
126
+ * 3. Optionally clamps `motion.reducedMotion` to `"true"` based on the
127
+ * OS-level preference (or a forced override).
128
+ * 4. Serialises the resolved theme into CSS variables on a wrapper
129
+ * element so descendants can read tokens via `useTheme()` *or* via
130
+ * `var(--db-...)` references in their own CSS / Tailwind.
131
+ *
132
+ * Defaults such as `<GameLayout>` mount this for you with the configured
133
+ * theme; mount it manually only when you need a sub-tree to use a different
134
+ * theme (e.g. a dark sheet over a light layout).
135
+ */
136
+ export function ThemeProvider({
137
+ theme: themeInput,
138
+ override,
139
+ reducedMotion = "auto",
140
+ as = "block",
141
+ style,
142
+ className,
143
+ children,
144
+ }: ThemeProviderProps) {
145
+ const prefersReduced = usePrefersReducedMotion();
146
+
147
+ const value = useMemo<ThemeContextValue>(() => {
148
+ const base = resolveTheme(themeInput);
149
+ const merged = mergeTheme(base, override);
150
+ const motionResolved =
151
+ reducedMotion === "force"
152
+ ? "true"
153
+ : reducedMotion === "ignore"
154
+ ? merged.motion.reducedMotion
155
+ : prefersReduced
156
+ ? "true"
157
+ : merged.motion.reducedMotion;
158
+ const final: Theme =
159
+ motionResolved === merged.motion.reducedMotion
160
+ ? merged
161
+ : {
162
+ ...merged,
163
+ motion: { ...merged.motion, reducedMotion: motionResolved },
164
+ };
165
+ return { theme: final, cssVars: themeToCssVars(final) };
166
+ }, [themeInput, override, reducedMotion, prefersReduced]);
167
+
168
+ if (as === "none") {
169
+ return <ThemeCtx.Provider value={value}>{children}</ThemeCtx.Provider>;
170
+ }
171
+
172
+ const wrapperStyle: CSSProperties = {
173
+ ...themeToShadcnVars(value.theme),
174
+ ...value.cssVars,
175
+ fontFamily: value.theme.typography.fontFamily.body,
176
+ ...style,
177
+ };
178
+
179
+ return (
180
+ <ThemeCtx.Provider value={value}>
181
+ <div
182
+ data-dreamboard-theme={value.theme.meta.id}
183
+ data-dreamboard-mode={value.theme.meta.mode}
184
+ data-dreamboard-reduced-motion={value.theme.motion.reducedMotion}
185
+ className={className}
186
+ style={wrapperStyle}
187
+ >
188
+ {children}
189
+ </div>
190
+ </ThemeCtx.Provider>
191
+ );
192
+ }
193
+
194
+ /**
195
+ * Read the active {@link Theme} from a {@link ThemeProvider} ancestor.
196
+ *
197
+ * Falls back to the `tabletop` preset (with no overrides) when called
198
+ * outside a provider — components remain renderable in isolated tests
199
+ * and Storybook/Cosmos fixtures without a wrapping shell.
200
+ */
201
+ export function useTheme(): Theme {
202
+ const ctx = useContext(ThemeCtx);
203
+ return ctx?.theme ?? tabletopTheme;
204
+ }
205
+
206
+ /**
207
+ * Read the {@link CSSProperties} that publish the active theme as CSS
208
+ * variables. Useful when porting a theme into a portal or drawer that
209
+ * is rendered outside the provider's DOM subtree.
210
+ */
211
+ export function useThemeCssVars(): CSSProperties {
212
+ const ctx = useContext(ThemeCtx);
213
+ if (ctx) return { ...themeToShadcnVars(ctx.theme), ...ctx.cssVars };
214
+ // Match the fallback theme exposed by `useTheme()`.
215
+ return {
216
+ ...themeToShadcnVars(tabletopTheme),
217
+ ...themeToCssVars(tabletopTheme),
218
+ };
219
+ }
220
+
221
+ export { ThemeCtx };
222
+
223
+ type CssVariableStyle = CSSProperties & {
224
+ [K in `--${string}`]?: string | number;
225
+ };
226
+
227
+ function themeToShadcnVars(theme: Theme): CSSProperties {
228
+ const vars: CssVariableStyle = {
229
+ "--background": theme.semantic.surface.app,
230
+ "--foreground": theme.semantic.text.primary,
231
+ "--card": theme.semantic.surface.card,
232
+ "--card-foreground": theme.semantic.text.primary,
233
+ "--popover": theme.semantic.surface.sheet,
234
+ "--popover-foreground": theme.semantic.text.primary,
235
+ "--primary": theme.semantic.intent.primary.solid,
236
+ "--primary-foreground": theme.semantic.intent.primary.on,
237
+ "--secondary": theme.semantic.intent.secondary.solid,
238
+ "--secondary-foreground": theme.semantic.intent.secondary.on,
239
+ "--muted": theme.semantic.surface.inset,
240
+ "--muted-foreground": theme.semantic.text.muted,
241
+ "--accent": theme.semantic.intent.info.soft,
242
+ "--accent-foreground": theme.semantic.intent.info.onSoft,
243
+ "--destructive": theme.semantic.intent.danger.solid,
244
+ "--destructive-foreground": theme.semantic.intent.danger.on,
245
+ "--border": theme.semantic.border.default,
246
+ "--input": theme.semantic.border.default,
247
+ "--ring": theme.semantic.border.focus,
248
+ "--font-sans": theme.typography.fontFamily.body,
249
+ "--font-display": theme.typography.fontFamily.display,
250
+ };
251
+ return vars;
252
+ }
@@ -0,0 +1,61 @@
1
+ /**
2
+ * Board theming helpers.
3
+ *
4
+ * The SDK's board renderers (`<HexGrid>`, `<SquareGrid>`,
5
+ * `<NetworkGraph>`, `<ZoneMap>`, `<TrackBoard>`, `<SlotSystem>`) need a
6
+ * tiny but consistent set of visual tokens — frame border, frame fill,
7
+ * eligible-target hint, and active hover ring — so authors don't have
8
+ * to re-pick colours per board. {@link useBoardTheme} exposes those
9
+ * tokens, derived from the active {@link Theme}.
10
+ */
11
+
12
+ import { useMemo } from "react";
13
+ import { useTheme } from "./ThemeProvider.js";
14
+ import type { Theme } from "./tokens.js";
15
+
16
+ /**
17
+ * Resolved board tokens consumed by board renderers and per-tile
18
+ * default renderers. All values are CSS-ready strings.
19
+ */
20
+ export interface BoardTheme {
21
+ /** Border drawn around the board's outer frame. */
22
+ readonly frameBorder: string;
23
+ /** Background painted under the tile renderer (visible at tile gaps). */
24
+ readonly frameBackground: string;
25
+ /** Hover-state ring drawn over an interactive target. */
26
+ readonly hoverRing: string;
27
+ /** Idle hint colour for an eligible-but-unhovered target. */
28
+ readonly eligibleHint: string;
29
+ /** Border drawn on inactive tiles (matches semantic.border.default). */
30
+ readonly tileBorder: string;
31
+ /** Subtle dotted border for non-interactive scaffolding (e.g. ports). */
32
+ readonly tileBorderSubtle: string;
33
+ /** Foreground used for board labels (numbers, port ratios). */
34
+ readonly tileText: string;
35
+ /** Recommended ring stroke width in px for hover/eligible markers. */
36
+ readonly ringWidth: number;
37
+ }
38
+
39
+ /**
40
+ * Derive a {@link BoardTheme} from the active theme. Memoised on the
41
+ * theme reference so consumers can pass it straight to a board's
42
+ * `defaultTileProps` / `defaultEdgeProps` slots.
43
+ */
44
+ export function useBoardTheme(): BoardTheme {
45
+ const theme = useTheme();
46
+ return useMemo(() => deriveBoardTheme(theme), [theme]);
47
+ }
48
+
49
+ /** Standalone derivation for renderers that already hold a {@link Theme}. */
50
+ export function deriveBoardTheme(theme: Theme): BoardTheme {
51
+ return {
52
+ frameBorder: theme.component.board.frameBorder,
53
+ frameBackground: theme.component.board.frameBackground,
54
+ hoverRing: theme.component.board.hoverRing,
55
+ eligibleHint: theme.component.board.eligibleHint,
56
+ tileBorder: theme.semantic.border.default,
57
+ tileBorderSubtle: theme.semantic.border.subtle,
58
+ tileText: theme.semantic.text.primary,
59
+ ringWidth: 3,
60
+ };
61
+ }
@@ -0,0 +1,105 @@
1
+ import type { CSSProperties } from "react";
2
+ import type { Theme } from "./tokens.js";
3
+
4
+ /**
5
+ * CSS variable bridge for {@link Theme}.
6
+ *
7
+ * The {@link ThemeProvider} mounts a wrapper element and applies the
8
+ * map returned by {@link themeToCssVars} as inline `style`, which
9
+ * declares one CSS custom property per leaf token. Components inside
10
+ * the provider can:
11
+ *
12
+ * 1. Read tokens via `useTheme()` (typed) and use them inline, or
13
+ * 2. Read the CSS variable directly in Tailwind / className styling
14
+ * via the helpers in {@link var()} below.
15
+ *
16
+ * The variable naming scheme is `--db-<dot.path>` with `.` and any
17
+ * non-identifier characters replaced by `-`. Example:
18
+ *
19
+ * - `theme.semantic.intent.primary.solid` → `--db-semantic-intent-primary-solid`
20
+ * - `theme.player[0].solid` → `--db-player-1-solid`
21
+ * - `theme.color.brand.600` → `--db-color-brand-600`
22
+ *
23
+ * The 1-based player suffix (rather than 0-based) keeps the variable
24
+ * name aligned with how seats are labelled in product UI ("Player 1").
25
+ */
26
+
27
+ const PREFIX = "--db";
28
+
29
+ /** Convert a token-tree path to a kebab-case CSS variable name. */
30
+ function toVarName(parts: ReadonlyArray<string | number>): string {
31
+ const tail = parts
32
+ .map((part) => String(part))
33
+ .join("-")
34
+ .replace(/[^a-zA-Z0-9-]/g, "-")
35
+ .replace(/-+/g, "-");
36
+ return `${PREFIX}-${tail}`;
37
+ }
38
+
39
+ /**
40
+ * Walk the theme tree and produce a flat `{ "--db-...": value }` map.
41
+ * Arrays are walked positionally with a 1-based index (so
42
+ * `theme.player[0]` → `--db-player-1-...`).
43
+ */
44
+ export function themeToCssVars(theme: Theme): CSSProperties {
45
+ const out: Record<string, string> = {};
46
+ walk(theme as unknown as Record<string, unknown>, [], out);
47
+ return out as CSSProperties;
48
+ }
49
+
50
+ function walk(
51
+ node: unknown,
52
+ path: ReadonlyArray<string | number>,
53
+ out: Record<string, string>,
54
+ ): void {
55
+ if (node === null || node === undefined) return;
56
+ if (typeof node === "string" || typeof node === "number") {
57
+ out[toVarName(path)] = String(node);
58
+ return;
59
+ }
60
+ if (Array.isArray(node)) {
61
+ node.forEach((child, idx) => {
62
+ // 1-based index for player slots, etc.
63
+ walk(child, [...path, idx + 1], out);
64
+ });
65
+ return;
66
+ }
67
+ if (typeof node === "object") {
68
+ for (const [key, value] of Object.entries(
69
+ node as Record<string, unknown>,
70
+ )) {
71
+ walk(value, [...path, key], out);
72
+ }
73
+ }
74
+ }
75
+
76
+ /**
77
+ * Build a `var(--db-...)` reference for a token path.
78
+ *
79
+ * @example
80
+ * ```ts
81
+ * cssVar("semantic", "intent", "primary", "solid")
82
+ * // => "var(--db-semantic-intent-primary-solid)"
83
+ * cssVar("player", 1, "solid")
84
+ * // => "var(--db-player-1-solid)"
85
+ * ```
86
+ */
87
+ export function cssVar(...path: Array<string | number>): string {
88
+ return `var(${toVarName(path)})`;
89
+ }
90
+
91
+ /**
92
+ * Convenience for declaring a `var(--db-...)` with an inline fallback.
93
+ *
94
+ * @example
95
+ * ```ts
96
+ * cssVarOr("transparent", "semantic", "border", "default")
97
+ * // => "var(--db-semantic-border-default, transparent)"
98
+ * ```
99
+ */
100
+ export function cssVarOr(
101
+ fallback: string,
102
+ ...path: Array<string | number>
103
+ ): string {
104
+ return `var(${toVarName(path)}, ${fallback})`;
105
+ }