@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,378 @@
1
+ /**
2
+ * Presentational card shell. Feed from reducer views, not runtime state.
3
+ *
4
+ * `CardFace` is the only public reusable card shell in `@dreamboard-games/sdk/ui`.
5
+ * It renders display content with controlled {@link InteractionVisualState}
6
+ * (eligible / selected / disabled / invalid / submitted / previewing) and
7
+ * exposes stable `data-*` attributes. It does not own activation behavior —
8
+ * compose it with a generic `<button>` (or with a runtime adapter) when an
9
+ * authored surface needs to react to taps or keyboard activation.
10
+ */
11
+
12
+ import { clsx } from "clsx";
13
+ import type { ViewCard } from "../../types/index.js";
14
+ import { useTheme } from "../theme/ThemeProvider.js";
15
+ import { motionDuration } from "../theme/derive.js";
16
+ import {
17
+ visualStateDataAttributes,
18
+ type InteractionVisualState,
19
+ } from "../types/visual-state.js";
20
+
21
+ export interface CardFaceProps<CardData extends ViewCard = ViewCard>
22
+ extends
23
+ Omit<React.HTMLAttributes<HTMLDivElement>, "children">,
24
+ InteractionVisualState {
25
+ card: CardData;
26
+ size?: "sm" | "md" | "lg";
27
+ faceDown?: boolean;
28
+ renderContent?: (card: CardData) => React.ReactNode;
29
+ children?: React.ReactNode;
30
+ }
31
+
32
+ /**
33
+ * Reserved keys on `ViewCard.properties` that the default content
34
+ * renderer recognises and styles consistently. Authors who set these
35
+ * keys at view-build time get a properly laid-out card face for free
36
+ * (icon → title → subtitle → effect → cost), instead of the previous
37
+ * "first three properties as label/value" fallback that produced
38
+ * inconsistent card faces across games.
39
+ *
40
+ * Any other property keys are still rendered, just below the
41
+ * conventional cluster, so authors can mix and match without losing
42
+ * data — but the canonical surface should pick from these slots first.
43
+ *
44
+ * Slots:
45
+ *
46
+ * - `icon` — short visual flag (emoji, glyph). Rendered large at the
47
+ * top of the card.
48
+ * - `title` — primary label. Falls back to `card.name`.
49
+ * - `subtitle` — short qualifier ("1 VP", "Action", "Resource").
50
+ * - `effect` — body text describing what the card does.
51
+ * - `cost` — short cost line ("3 brick"); rendered as a chip footer.
52
+ */
53
+ const RESERVED_CONTENT_KEYS = [
54
+ "icon",
55
+ "title",
56
+ "subtitle",
57
+ "effect",
58
+ "cost",
59
+ ] as const;
60
+
61
+ type ReservedKey = (typeof RESERVED_CONTENT_KEYS)[number];
62
+
63
+ const RESERVED_KEY_SET: ReadonlySet<string> = new Set(RESERVED_CONTENT_KEYS);
64
+
65
+ function readReserved(
66
+ props: Record<string, unknown> | undefined,
67
+ key: ReservedKey,
68
+ ): string | undefined {
69
+ if (!props) return undefined;
70
+ const raw = props[key];
71
+ if (raw === undefined || raw === null) return undefined;
72
+ return String(raw);
73
+ }
74
+
75
+ /** Default card content renderer. */
76
+ function DefaultCardContent({
77
+ card,
78
+ size,
79
+ }: {
80
+ card: ViewCard;
81
+ size: NonNullable<CardFaceProps["size"]>;
82
+ }) {
83
+ const theme = useTheme();
84
+ const props = card.properties as Record<string, unknown> | undefined;
85
+ const icon = readReserved(props, "icon");
86
+ const title = readReserved(props, "title") ?? card.name;
87
+ const subtitle = readReserved(props, "subtitle");
88
+ const effect = readReserved(props, "effect");
89
+ const cost = readReserved(props, "cost");
90
+ const extra = props
91
+ ? Object.entries(props).filter(([key]) => !RESERVED_KEY_SET.has(key))
92
+ : [];
93
+
94
+ const hasAnyReserved = icon || title || subtitle || effect || cost;
95
+ const compact = size === "sm";
96
+
97
+ return (
98
+ <div
99
+ style={{
100
+ display: "flex",
101
+ flexDirection: "column",
102
+ alignItems: "center",
103
+ height: "100%",
104
+ minHeight: 0,
105
+ boxSizing: "border-box",
106
+ overflow: "hidden",
107
+ textAlign: "center",
108
+ padding: compact ? theme.space[1] : theme.space[2],
109
+ gap: compact ? theme.space[0.5] : theme.space[1],
110
+ }}
111
+ >
112
+ {icon ? (
113
+ <span
114
+ aria-hidden
115
+ style={{
116
+ flexShrink: 0,
117
+ fontSize: compact
118
+ ? theme.typography.fontSize.lg
119
+ : theme.typography.fontSize["2xl"],
120
+ lineHeight: 1,
121
+ }}
122
+ >
123
+ {icon}
124
+ </span>
125
+ ) : null}
126
+ {title ? (
127
+ <span
128
+ style={{
129
+ flexShrink: 0,
130
+ fontFamily: theme.typography.fontFamily.display,
131
+ fontSize: theme.typography.fontSize.sm,
132
+ fontWeight: theme.typography.fontWeight.bold,
133
+ color: theme.semantic.text.primary,
134
+ lineHeight: theme.typography.lineHeight.tight,
135
+ overflow: "hidden",
136
+ textOverflow: "ellipsis",
137
+ display: "-webkit-box",
138
+ WebkitLineClamp: 2,
139
+ WebkitBoxOrient: "vertical",
140
+ wordBreak: "break-word",
141
+ }}
142
+ >
143
+ {title}
144
+ </span>
145
+ ) : null}
146
+ {subtitle ? (
147
+ <span
148
+ style={{
149
+ flexShrink: 0,
150
+ fontFamily: theme.typography.fontFamily.body,
151
+ fontSize: theme.typography.fontSize.xs,
152
+ fontWeight: theme.typography.fontWeight.medium,
153
+ color: theme.semantic.text.muted,
154
+ letterSpacing: theme.typography.letterSpacing.wide,
155
+ textTransform: "uppercase",
156
+ }}
157
+ >
158
+ {subtitle}
159
+ </span>
160
+ ) : null}
161
+ {effect ? (
162
+ <span
163
+ style={{
164
+ flex: "1 1 auto",
165
+ minHeight: 0,
166
+ fontFamily: theme.typography.fontFamily.body,
167
+ fontSize: theme.typography.fontSize.xs,
168
+ color: theme.semantic.text.muted,
169
+ lineHeight: theme.typography.lineHeight.normal,
170
+ display: "-webkit-box",
171
+ WebkitLineClamp: compact ? 1 : size === "md" ? 2 : 3,
172
+ WebkitBoxOrient: "vertical",
173
+ overflow: "hidden",
174
+ }}
175
+ >
176
+ {effect}
177
+ </span>
178
+ ) : null}
179
+ {cost ? (
180
+ <span
181
+ style={{
182
+ display: "inline-flex",
183
+ flexShrink: 0,
184
+ alignItems: "center",
185
+ gap: theme.space[1],
186
+ paddingBlock: theme.space[0.5],
187
+ paddingInline: theme.space[2],
188
+ borderRadius: theme.radius.pill,
189
+ background: theme.semantic.surface.inset,
190
+ border: `1px solid ${theme.semantic.border.subtle}`,
191
+ fontFamily: theme.typography.fontFamily.tabular,
192
+ fontSize: theme.typography.fontSize.xs,
193
+ fontWeight: theme.typography.fontWeight.bold,
194
+ color: theme.semantic.text.primary,
195
+ marginTop: "auto",
196
+ }}
197
+ >
198
+ {cost}
199
+ </span>
200
+ ) : null}
201
+ {!hasAnyReserved && extra.length > 0
202
+ ? extra.slice(0, 3).map(([key, value]) => (
203
+ <span
204
+ key={key}
205
+ style={{
206
+ flexShrink: 0,
207
+ fontFamily: theme.typography.fontFamily.body,
208
+ fontSize: theme.typography.fontSize.xs,
209
+ color: theme.semantic.text.primary,
210
+ }}
211
+ >
212
+ <span style={{ color: theme.semantic.text.muted }}>{key}:</span>{" "}
213
+ <span style={{ fontWeight: theme.typography.fontWeight.bold }}>
214
+ {String(value)}
215
+ </span>
216
+ </span>
217
+ ))
218
+ : null}
219
+ {!hasAnyReserved && extra.length === 0 ? (
220
+ <span
221
+ style={{
222
+ flexShrink: 0,
223
+ fontFamily: theme.typography.fontFamily.mono,
224
+ fontSize: theme.typography.fontSize.xs,
225
+ color: theme.semantic.text.disabled,
226
+ }}
227
+ >
228
+ {card.id.slice(0, 8)}
229
+ </span>
230
+ ) : null}
231
+ </div>
232
+ );
233
+ }
234
+
235
+ const CARD_SIZE_CLASSES = {
236
+ sm: "w-16 h-24 sm:w-20 sm:h-28",
237
+ md: "w-20 h-32 sm:w-24 sm:h-36",
238
+ lg: "w-24 h-36 sm:w-32 sm:h-48",
239
+ } as const;
240
+
241
+ export function CardFace<CardData extends ViewCard = ViewCard>({
242
+ card,
243
+ eligible,
244
+ selected = false,
245
+ disabled = false,
246
+ invalid,
247
+ submitted,
248
+ previewing,
249
+ intentProgress,
250
+ size = "md",
251
+ faceDown = false,
252
+ renderContent,
253
+ children,
254
+ className,
255
+ style,
256
+ ...props
257
+ }: CardFaceProps<CardData>) {
258
+ const theme = useTheme();
259
+ const calmCornerRadius = theme.radius.md;
260
+ const accentCornerRadius = `${theme.radius.lg} ${theme.radius.sm} ${theme.radius.lg} ${theme.radius.sm} / ${theme.radius.sm} ${theme.radius.lg} ${theme.radius.sm} ${theme.radius.lg}`;
261
+ const selectionRingColor = theme.component.card.selectedRing;
262
+ const eligibleRingColor = theme.semantic.intent.warning.border;
263
+ const invalidRingColor = theme.semantic.intent.danger.border;
264
+ const submittedRingColor = theme.semantic.intent.success.border;
265
+ const restBorderColor = theme.component.card.border;
266
+ const dataAttributes = visualStateDataAttributes({
267
+ eligible,
268
+ selected,
269
+ disabled,
270
+ invalid,
271
+ submitted,
272
+ previewing,
273
+ intentProgress,
274
+ });
275
+ // Routine resting cards use a calm modern surface. Hard-shadow / irregular
276
+ // corner accents are reserved for explicit emphasis states so a fanned
277
+ // hand reads as ordered play instead of repeated decoration.
278
+ const accentColor = invalid
279
+ ? invalidRingColor
280
+ : submitted
281
+ ? submittedRingColor
282
+ : selected
283
+ ? selectionRingColor
284
+ : previewing
285
+ ? selectionRingColor
286
+ : eligible
287
+ ? eligibleRingColor
288
+ : undefined;
289
+ const isAccented = Boolean(accentColor);
290
+ const accentShadow = selected
291
+ ? `6px 6px 0px 0px ${selectionRingColor}`
292
+ : previewing
293
+ ? `4px 4px 0px 0px ${selectionRingColor}`
294
+ : invalid
295
+ ? `4px 4px 0px 0px ${invalidRingColor}`
296
+ : submitted
297
+ ? `4px 4px 0px 0px ${submittedRingColor}`
298
+ : eligible
299
+ ? `4px 4px 0px 0px ${eligibleRingColor}`
300
+ : undefined;
301
+ const restShadow = theme.elevation.rest;
302
+
303
+ return (
304
+ <div
305
+ {...props}
306
+ {...dataAttributes}
307
+ data-dreamboard-card-face=""
308
+ data-accented={isAccented ? "true" : undefined}
309
+ data-face-down={faceDown ? "true" : undefined}
310
+ className={clsx("relative", CARD_SIZE_CLASSES[size], className)}
311
+ style={{
312
+ opacity: disabled ? 0.5 : submitted ? 0.85 : 1,
313
+ transitionProperty: "transform, box-shadow, border-color",
314
+ transitionDuration: motionDuration(theme, "normal"),
315
+ transitionTimingFunction: theme.motion.easing.out,
316
+ transform: previewing ? "translateY(-2px) scale(1.02)" : undefined,
317
+ ...style,
318
+ }}
319
+ >
320
+ {faceDown ? (
321
+ // Card back: keep the playful diagonal pattern as artwork, but
322
+ // anchor the shell on the same calm baseline as a face-up card so
323
+ // a row of opponent hands or a draw pile reads as ordered chrome
324
+ // rather than repeated decoration. Emphasis states (selected,
325
+ // eligible, invalid, submitted, previewing) still upgrade to the
326
+ // accent radius + hard-offset shadow.
327
+ <div
328
+ className="absolute inset-0 overflow-hidden pointer-events-none"
329
+ style={{
330
+ background: theme.component.card.backBackground,
331
+ borderStyle: "solid",
332
+ borderWidth: isAccented ? 3 : 1,
333
+ borderColor: accentColor ?? restBorderColor,
334
+ borderRadius: isAccented ? accentCornerRadius : calmCornerRadius,
335
+ boxShadow: accentShadow ?? restShadow,
336
+ }}
337
+ >
338
+ <div
339
+ className="absolute inset-2"
340
+ style={{
341
+ borderRadius: calmCornerRadius,
342
+ background:
343
+ "repeating-linear-gradient(45deg, transparent, transparent 10px, rgba(255,255,255,0.2) 10px, rgba(255,255,255,0.2) 20px)",
344
+ }}
345
+ />
346
+ </div>
347
+ ) : (
348
+ <div
349
+ className="absolute inset-0 overflow-hidden pointer-events-none"
350
+ style={{
351
+ background: submitted
352
+ ? theme.semantic.intent.success.soft
353
+ : theme.semantic.surface.card,
354
+ borderStyle: "solid",
355
+ borderWidth: isAccented ? 3 : 1,
356
+ borderColor: accentColor ?? restBorderColor,
357
+ borderRadius: isAccented ? accentCornerRadius : calmCornerRadius,
358
+ boxShadow: accentShadow ?? restShadow,
359
+ outline:
360
+ selected || invalid
361
+ ? `2px solid ${accentColor ?? selectionRingColor}`
362
+ : undefined,
363
+ outlineOffset: selected || invalid ? "1px" : undefined,
364
+ }}
365
+ >
366
+ {children ??
367
+ (renderContent ? (
368
+ renderContent(card)
369
+ ) : (
370
+ <DefaultCardContent card={card} size={size} />
371
+ ))}
372
+ </div>
373
+ )}
374
+ </div>
375
+ );
376
+ }
377
+
378
+ export type { ViewCard };