@dreamboard-games/ui-sdk 0.0.41

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 (533) hide show
  1. package/LICENSE +89 -0
  2. package/NOTICE +1 -0
  3. package/README.md +154 -0
  4. package/dist/components/ActionButton.d.ts +13 -0
  5. package/dist/components/ActionButton.d.ts.map +1 -0
  6. package/dist/components/ActionButton.js +14 -0
  7. package/dist/components/ActionPanel.d.ts +33 -0
  8. package/dist/components/ActionPanel.d.ts.map +1 -0
  9. package/dist/components/ActionPanel.js +148 -0
  10. package/dist/components/Card.d.ts +29 -0
  11. package/dist/components/Card.d.ts.map +1 -0
  12. package/dist/components/Card.js +220 -0
  13. package/dist/components/ChromeSuppressionContext.d.ts +7 -0
  14. package/dist/components/ChromeSuppressionContext.d.ts.map +1 -0
  15. package/dist/components/ChromeSuppressionContext.js +34 -0
  16. package/dist/components/CostDisplay.d.ts +22 -0
  17. package/dist/components/CostDisplay.d.ts.map +1 -0
  18. package/dist/components/CostDisplay.js +41 -0
  19. package/dist/components/DiceRoller.d.ts +30 -0
  20. package/dist/components/DiceRoller.d.ts.map +1 -0
  21. package/dist/components/DiceRoller.js +319 -0
  22. package/dist/components/Drawer.d.ts +19 -0
  23. package/dist/components/Drawer.d.ts.map +1 -0
  24. package/dist/components/Drawer.js +55 -0
  25. package/dist/components/ErrorBoundary.d.ts +24 -0
  26. package/dist/components/ErrorBoundary.d.ts.map +1 -0
  27. package/dist/components/ErrorBoundary.js +37 -0
  28. package/dist/components/GameEndDisplay.d.ts +27 -0
  29. package/dist/components/GameEndDisplay.d.ts.map +1 -0
  30. package/dist/components/GameEndDisplay.js +185 -0
  31. package/dist/components/GameSkeleton.d.ts +12 -0
  32. package/dist/components/GameSkeleton.d.ts.map +1 -0
  33. package/dist/components/GameSkeleton.js +54 -0
  34. package/dist/components/Hand.d.ts +99 -0
  35. package/dist/components/Hand.d.ts.map +1 -0
  36. package/dist/components/Hand.js +162 -0
  37. package/dist/components/HandDock.d.ts +35 -0
  38. package/dist/components/HandDock.d.ts.map +1 -0
  39. package/dist/components/HandDock.js +124 -0
  40. package/dist/components/InteractionForm.d.ts +50 -0
  41. package/dist/components/InteractionForm.d.ts.map +1 -0
  42. package/dist/components/InteractionForm.js +402 -0
  43. package/dist/components/MoreActions.d.ts +49 -0
  44. package/dist/components/MoreActions.d.ts.map +1 -0
  45. package/dist/components/MoreActions.js +64 -0
  46. package/dist/components/PhaseIndicator.d.ts +35 -0
  47. package/dist/components/PhaseIndicator.d.ts.map +1 -0
  48. package/dist/components/PhaseIndicator.js +212 -0
  49. package/dist/components/PlayArea.d.ts +28 -0
  50. package/dist/components/PlayArea.d.ts.map +1 -0
  51. package/dist/components/PlayArea.js +48 -0
  52. package/dist/components/PluginRuntime.d.ts +37 -0
  53. package/dist/components/PluginRuntime.d.ts.map +1 -0
  54. package/dist/components/PluginRuntime.js +47 -0
  55. package/dist/components/PrimaryActionButton.d.ts +98 -0
  56. package/dist/components/PrimaryActionButton.d.ts.map +1 -0
  57. package/dist/components/PrimaryActionButton.js +183 -0
  58. package/dist/components/PrimaryButton.d.ts +20 -0
  59. package/dist/components/PrimaryButton.d.ts.map +1 -0
  60. package/dist/components/PrimaryButton.js +5 -0
  61. package/dist/components/PromptDialogHost.d.ts +15 -0
  62. package/dist/components/PromptDialogHost.d.ts.map +1 -0
  63. package/dist/components/PromptDialogHost.js +22 -0
  64. package/dist/components/ResourceCounter.d.ts +38 -0
  65. package/dist/components/ResourceCounter.d.ts.map +1 -0
  66. package/dist/components/ResourceCounter.js +118 -0
  67. package/dist/components/ThemedButton.d.ts +12 -0
  68. package/dist/components/ThemedButton.d.ts.map +1 -0
  69. package/dist/components/ThemedButton.js +38 -0
  70. package/dist/components/Toast.d.ts +35 -0
  71. package/dist/components/Toast.d.ts.map +1 -0
  72. package/dist/components/Toast.js +116 -0
  73. package/dist/components/board/HexGrid.d.ts +344 -0
  74. package/dist/components/board/HexGrid.d.ts.map +1 -0
  75. package/dist/components/board/HexGrid.js +340 -0
  76. package/dist/components/board/NetworkGraph.d.ts +100 -0
  77. package/dist/components/board/NetworkGraph.d.ts.map +1 -0
  78. package/dist/components/board/NetworkGraph.js +123 -0
  79. package/dist/components/board/SlotSystem.d.ts +71 -0
  80. package/dist/components/board/SlotSystem.d.ts.map +1 -0
  81. package/dist/components/board/SlotSystem.js +87 -0
  82. package/dist/components/board/SquareGrid.d.ts +188 -0
  83. package/dist/components/board/SquareGrid.d.ts.map +1 -0
  84. package/dist/components/board/SquareGrid.js +328 -0
  85. package/dist/components/board/TrackBoard.d.ts +113 -0
  86. package/dist/components/board/TrackBoard.d.ts.map +1 -0
  87. package/dist/components/board/TrackBoard.js +135 -0
  88. package/dist/components/board/ZoneMap.d.ts +88 -0
  89. package/dist/components/board/ZoneMap.d.ts.map +1 -0
  90. package/dist/components/board/ZoneMap.js +133 -0
  91. package/dist/components/board/hex-board-view.d.ts +69 -0
  92. package/dist/components/board/hex-board-view.d.ts.map +1 -0
  93. package/dist/components/board/hex-board-view.js +60 -0
  94. package/dist/components/board/index.d.ts +23 -0
  95. package/dist/components/board/index.d.ts.map +1 -0
  96. package/dist/components/board/index.js +40 -0
  97. package/dist/components/board/interaction-accessibility.d.ts +5 -0
  98. package/dist/components/board/interaction-accessibility.d.ts.map +1 -0
  99. package/dist/components/board/interaction-accessibility.js +13 -0
  100. package/dist/components/board/target-layer.d.ts +13 -0
  101. package/dist/components/board/target-layer.d.ts.map +1 -0
  102. package/dist/components/board/target-layer.js +10 -0
  103. package/dist/components/card-render-content.type-test.d.ts +2 -0
  104. package/dist/components/card-render-content.type-test.d.ts.map +1 -0
  105. package/dist/components/card-render-content.type-test.js +1 -0
  106. package/dist/components/index.d.ts +34 -0
  107. package/dist/components/index.d.ts.map +1 -0
  108. package/dist/components/index.js +35 -0
  109. package/dist/components/interaction-dialog-behavior.d.ts +15 -0
  110. package/dist/components/interaction-dialog-behavior.d.ts.map +1 -0
  111. package/dist/components/interaction-dialog-behavior.js +9 -0
  112. package/dist/components/surfaces/BlockerSurface.d.ts +27 -0
  113. package/dist/components/surfaces/BlockerSurface.d.ts.map +1 -0
  114. package/dist/components/surfaces/BlockerSurface.js +38 -0
  115. package/dist/components/surfaces/BoardSurface.d.ts +77 -0
  116. package/dist/components/surfaces/BoardSurface.d.ts.map +1 -0
  117. package/dist/components/surfaces/BoardSurface.js +180 -0
  118. package/dist/components/surfaces/ChromeSurface.d.ts +29 -0
  119. package/dist/components/surfaces/ChromeSurface.d.ts.map +1 -0
  120. package/dist/components/surfaces/ChromeSurface.js +34 -0
  121. package/dist/components/surfaces/ExhaustivenessAudit.d.ts +32 -0
  122. package/dist/components/surfaces/ExhaustivenessAudit.d.ts.map +1 -0
  123. package/dist/components/surfaces/ExhaustivenessAudit.js +65 -0
  124. package/dist/components/surfaces/InboxSurface.d.ts +40 -0
  125. package/dist/components/surfaces/InboxSurface.d.ts.map +1 -0
  126. package/dist/components/surfaces/InboxSurface.js +99 -0
  127. package/dist/components/surfaces/MarketSurface.d.ts +62 -0
  128. package/dist/components/surfaces/MarketSurface.d.ts.map +1 -0
  129. package/dist/components/surfaces/MarketSurface.js +242 -0
  130. package/dist/components/surfaces/PanelSurface.d.ts +111 -0
  131. package/dist/components/surfaces/PanelSurface.d.ts.map +1 -0
  132. package/dist/components/surfaces/PanelSurface.js +180 -0
  133. package/dist/components/surfaces/PlayerCardsSurface.d.ts +104 -0
  134. package/dist/components/surfaces/PlayerCardsSurface.d.ts.map +1 -0
  135. package/dist/components/surfaces/PlayerCardsSurface.js +178 -0
  136. package/dist/components/surfaces/internal/CardZoneFollowUpForm.d.ts +7 -0
  137. package/dist/components/surfaces/internal/CardZoneFollowUpForm.d.ts.map +1 -0
  138. package/dist/components/surfaces/internal/CardZoneFollowUpForm.js +9 -0
  139. package/dist/components/surfaces/internal/DefaultInteractionButton.d.ts +71 -0
  140. package/dist/components/surfaces/internal/DefaultInteractionButton.d.ts.map +1 -0
  141. package/dist/components/surfaces/internal/DefaultInteractionButton.js +82 -0
  142. package/dist/components/surfaces/internal/useCardZoneInteractions.d.ts +21 -0
  143. package/dist/components/surfaces/internal/useCardZoneInteractions.d.ts.map +1 -0
  144. package/dist/components/surfaces/internal/useCardZoneInteractions.js +202 -0
  145. package/dist/components/surfaces/types.d.ts +59 -0
  146. package/dist/components/surfaces/types.d.ts.map +1 -0
  147. package/dist/components/surfaces/types.js +1 -0
  148. package/dist/context/ClientParamSchemaContext.d.ts +21 -0
  149. package/dist/context/ClientParamSchemaContext.d.ts.map +1 -0
  150. package/dist/context/ClientParamSchemaContext.js +12 -0
  151. package/dist/context/InteractionDraftContext.d.ts +69 -0
  152. package/dist/context/InteractionDraftContext.d.ts.map +1 -0
  153. package/dist/context/InteractionDraftContext.js +145 -0
  154. package/dist/context/PluginSessionContext.d.ts +33 -0
  155. package/dist/context/PluginSessionContext.d.ts.map +1 -0
  156. package/dist/context/PluginSessionContext.js +38 -0
  157. package/dist/context/PluginStateContext.d.ts +116 -0
  158. package/dist/context/PluginStateContext.d.ts.map +1 -0
  159. package/dist/context/PluginStateContext.js +186 -0
  160. package/dist/context/RuntimeContext.d.ts +49 -0
  161. package/dist/context/RuntimeContext.d.ts.map +1 -0
  162. package/dist/context/RuntimeContext.js +67 -0
  163. package/dist/defaults/components.d.ts +52 -0
  164. package/dist/defaults/components.d.ts.map +1 -0
  165. package/dist/defaults/components.js +159 -0
  166. package/dist/defaults/index.d.ts +2 -0
  167. package/dist/defaults/index.d.ts.map +1 -0
  168. package/dist/defaults/index.js +1 -0
  169. package/dist/errors/ValidationError.d.ts +10 -0
  170. package/dist/errors/ValidationError.d.ts.map +1 -0
  171. package/dist/errors/ValidationError.js +23 -0
  172. package/dist/helpers/cards.d.ts +3 -0
  173. package/dist/helpers/cards.d.ts.map +1 -0
  174. package/dist/helpers/cards.js +11 -0
  175. package/dist/helpers/track-board.d.ts +79 -0
  176. package/dist/helpers/track-board.d.ts.map +1 -0
  177. package/dist/helpers/track-board.js +56 -0
  178. package/dist/hooks/useActivePlayers.d.ts +16 -0
  179. package/dist/hooks/useActivePlayers.d.ts.map +1 -0
  180. package/dist/hooks/useActivePlayers.js +17 -0
  181. package/dist/hooks/useBoardInteractions.d.ts +110 -0
  182. package/dist/hooks/useBoardInteractions.d.ts.map +1 -0
  183. package/dist/hooks/useBoardInteractions.js +248 -0
  184. package/dist/hooks/useBoardTopology.d.ts +23 -0
  185. package/dist/hooks/useBoardTopology.d.ts.map +1 -0
  186. package/dist/hooks/useBoardTopology.js +128 -0
  187. package/dist/hooks/useCards.d.ts +3 -0
  188. package/dist/hooks/useCards.d.ts.map +1 -0
  189. package/dist/hooks/useCards.js +5 -0
  190. package/dist/hooks/useGameSelector.d.ts +13 -0
  191. package/dist/hooks/useGameSelector.d.ts.map +1 -0
  192. package/dist/hooks/useGameSelector.js +67 -0
  193. package/dist/hooks/useGameView.d.ts +6 -0
  194. package/dist/hooks/useGameView.d.ts.map +1 -0
  195. package/dist/hooks/useGameView.js +7 -0
  196. package/dist/hooks/useHandLayout.d.ts +120 -0
  197. package/dist/hooks/useHandLayout.d.ts.map +1 -0
  198. package/dist/hooks/useHandLayout.js +235 -0
  199. package/dist/hooks/useHexBoard.d.ts +19 -0
  200. package/dist/hooks/useHexBoard.d.ts.map +1 -0
  201. package/dist/hooks/useHexBoard.js +28 -0
  202. package/dist/hooks/useHexGrid.d.ts +56 -0
  203. package/dist/hooks/useHexGrid.d.ts.map +1 -0
  204. package/dist/hooks/useHexGrid.js +112 -0
  205. package/dist/hooks/useInteractionByKey.d.ts +29 -0
  206. package/dist/hooks/useInteractionByKey.d.ts.map +1 -0
  207. package/dist/hooks/useInteractionByKey.js +263 -0
  208. package/dist/hooks/useInteractionHandle.d.ts +103 -0
  209. package/dist/hooks/useInteractionHandle.d.ts.map +1 -0
  210. package/dist/hooks/useInteractionHandle.js +254 -0
  211. package/dist/hooks/useIsMobile.d.ts +7 -0
  212. package/dist/hooks/useIsMobile.d.ts.map +1 -0
  213. package/dist/hooks/useIsMobile.js +29 -0
  214. package/dist/hooks/useIsMyTurn.d.ts +6 -0
  215. package/dist/hooks/useIsMyTurn.d.ts.map +1 -0
  216. package/dist/hooks/useIsMyTurn.js +11 -0
  217. package/dist/hooks/useLobby.d.ts +28 -0
  218. package/dist/hooks/useLobby.d.ts.map +1 -0
  219. package/dist/hooks/useLobby.js +60 -0
  220. package/dist/hooks/useMe.d.ts +11 -0
  221. package/dist/hooks/useMe.d.ts.map +1 -0
  222. package/dist/hooks/useMe.js +32 -0
  223. package/dist/hooks/usePanZoom.d.ts +113 -0
  224. package/dist/hooks/usePanZoom.d.ts.map +1 -0
  225. package/dist/hooks/usePanZoom.js +165 -0
  226. package/dist/hooks/usePlayerInfo.d.ts +4 -0
  227. package/dist/hooks/usePlayerInfo.d.ts.map +1 -0
  228. package/dist/hooks/usePlayerInfo.js +21 -0
  229. package/dist/hooks/usePlayerTurnOrder.d.ts +15 -0
  230. package/dist/hooks/usePlayerTurnOrder.d.ts.map +1 -0
  231. package/dist/hooks/usePlayerTurnOrder.js +22 -0
  232. package/dist/hooks/usePluginRuntime.d.ts +45 -0
  233. package/dist/hooks/usePluginRuntime.d.ts.map +1 -0
  234. package/dist/hooks/usePluginRuntime.js +92 -0
  235. package/dist/hooks/useSeatInbox.d.ts +22 -0
  236. package/dist/hooks/useSeatInbox.d.ts.map +1 -0
  237. package/dist/hooks/useSeatInbox.js +43 -0
  238. package/dist/hooks/useSimultaneousPhase.d.ts +7 -0
  239. package/dist/hooks/useSimultaneousPhase.d.ts.map +1 -0
  240. package/dist/hooks/useSimultaneousPhase.js +8 -0
  241. package/dist/hooks/useSquareBoard.d.ts +21 -0
  242. package/dist/hooks/useSquareBoard.d.ts.map +1 -0
  243. package/dist/hooks/useSquareBoard.js +67 -0
  244. package/dist/hooks/useSquareGrid.d.ts +96 -0
  245. package/dist/hooks/useSquareGrid.d.ts.map +1 -0
  246. package/dist/hooks/useSquareGrid.js +152 -0
  247. package/dist/index.d.ts +30 -0
  248. package/dist/index.d.ts.map +1 -0
  249. package/dist/index.js +20 -0
  250. package/dist/internal/ui/alert.d.ts +8 -0
  251. package/dist/internal/ui/alert.d.ts.map +1 -0
  252. package/dist/internal/ui/alert.js +11 -0
  253. package/dist/internal/ui/button.d.ts +10 -0
  254. package/dist/internal/ui/button.d.ts.map +1 -0
  255. package/dist/internal/ui/button.js +21 -0
  256. package/dist/internal/ui/dialog.d.ts +16 -0
  257. package/dist/internal/ui/dialog.d.ts.map +1 -0
  258. package/dist/internal/ui/dialog.js +35 -0
  259. package/dist/internal/ui/input.d.ts +3 -0
  260. package/dist/internal/ui/input.d.ts.map +1 -0
  261. package/dist/internal/ui/input.js +5 -0
  262. package/dist/internal/ui/label.d.ts +4 -0
  263. package/dist/internal/ui/label.d.ts.map +1 -0
  264. package/dist/internal/ui/label.js +7 -0
  265. package/dist/internal/ui/select.d.ts +9 -0
  266. package/dist/internal/ui/select.d.ts.map +1 -0
  267. package/dist/internal/ui/select.js +23 -0
  268. package/dist/internal/ui/tooltip.d.ts +7 -0
  269. package/dist/internal/ui/tooltip.d.ts.map +1 -0
  270. package/dist/internal/ui/tooltip.js +16 -0
  271. package/dist/internal/ui/utils.d.ts +3 -0
  272. package/dist/internal/ui/utils.d.ts.map +1 -0
  273. package/dist/internal/ui/utils.js +4 -0
  274. package/dist/internal.d.ts +7 -0
  275. package/dist/internal.d.ts.map +1 -0
  276. package/dist/internal.js +4 -0
  277. package/dist/plugin-styles.css +246 -0
  278. package/dist/primitives/board.d.ts +29 -0
  279. package/dist/primitives/board.d.ts.map +1 -0
  280. package/dist/primitives/board.js +163 -0
  281. package/dist/primitives/game-ui-provider.d.ts +12 -0
  282. package/dist/primitives/game-ui-provider.d.ts.map +1 -0
  283. package/dist/primitives/game-ui-provider.js +7 -0
  284. package/dist/primitives/index.d.ts +8 -0
  285. package/dist/primitives/index.d.ts.map +1 -0
  286. package/dist/primitives/index.js +7 -0
  287. package/dist/primitives/interaction.d.ts +52 -0
  288. package/dist/primitives/interaction.d.ts.map +1 -0
  289. package/dist/primitives/interaction.js +250 -0
  290. package/dist/primitives/phase.d.ts +15 -0
  291. package/dist/primitives/phase.d.ts.map +1 -0
  292. package/dist/primitives/phase.js +18 -0
  293. package/dist/primitives/player-roster.d.ts +64 -0
  294. package/dist/primitives/player-roster.d.ts.map +1 -0
  295. package/dist/primitives/player-roster.js +149 -0
  296. package/dist/primitives/primitive-props.d.ts +15 -0
  297. package/dist/primitives/primitive-props.d.ts.map +1 -0
  298. package/dist/primitives/primitive-props.js +39 -0
  299. package/dist/primitives/prompt.d.ts +44 -0
  300. package/dist/primitives/prompt.d.ts.map +1 -0
  301. package/dist/primitives/prompt.js +101 -0
  302. package/dist/primitives/zone.d.ts +31 -0
  303. package/dist/primitives/zone.d.ts.map +1 -0
  304. package/dist/primitives/zone.js +58 -0
  305. package/dist/reducer.d.ts +21 -0
  306. package/dist/reducer.d.ts.map +1 -0
  307. package/dist/reducer.js +14 -0
  308. package/dist/runtime/createPluginRuntimeAPI.d.ts +67 -0
  309. package/dist/runtime/createPluginRuntimeAPI.d.ts.map +1 -0
  310. package/dist/runtime/createPluginRuntimeAPI.js +419 -0
  311. package/dist/theme/ThemeProvider.d.ts +98 -0
  312. package/dist/theme/ThemeProvider.d.ts.map +1 -0
  313. package/dist/theme/ThemeProvider.js +148 -0
  314. package/dist/theme/board.d.ts +42 -0
  315. package/dist/theme/board.d.ts.map +1 -0
  316. package/dist/theme/board.js +34 -0
  317. package/dist/theme/css-vars.d.ts +31 -0
  318. package/dist/theme/css-vars.d.ts.map +1 -0
  319. package/dist/theme/css-vars.js +88 -0
  320. package/dist/theme/derive.d.ts +66 -0
  321. package/dist/theme/derive.d.ts.map +1 -0
  322. package/dist/theme/derive.js +161 -0
  323. package/dist/theme/index.d.ts +22 -0
  324. package/dist/theme/index.d.ts.map +1 -0
  325. package/dist/theme/index.js +20 -0
  326. package/dist/theme/presets/arcade.d.ts +10 -0
  327. package/dist/theme/presets/arcade.d.ts.map +1 -0
  328. package/dist/theme/presets/arcade.js +257 -0
  329. package/dist/theme/presets/studio.d.ts +10 -0
  330. package/dist/theme/presets/studio.d.ts.map +1 -0
  331. package/dist/theme/presets/studio.js +257 -0
  332. package/dist/theme/presets/tabletop.d.ts +15 -0
  333. package/dist/theme/presets/tabletop.d.ts.map +1 -0
  334. package/dist/theme/presets/tabletop.js +262 -0
  335. package/dist/theme/tokens.d.ts +345 -0
  336. package/dist/theme/tokens.d.ts.map +1 -0
  337. package/dist/theme/tokens.js +57 -0
  338. package/dist/types/player-state.d.ts +337 -0
  339. package/dist/types/player-state.d.ts.map +1 -0
  340. package/dist/types/player-state.js +1 -0
  341. package/dist/types/plugin-state.d.ts +324 -0
  342. package/dist/types/plugin-state.d.ts.map +1 -0
  343. package/dist/types/plugin-state.js +1 -0
  344. package/dist/types/reducer-state.d.ts +10 -0
  345. package/dist/types/reducer-state.d.ts.map +1 -0
  346. package/dist/types/reducer-state.js +1 -0
  347. package/dist/types/runtime-api.d.ts +99 -0
  348. package/dist/types/runtime-api.d.ts.map +1 -0
  349. package/dist/types/runtime-api.js +1 -0
  350. package/dist/types/tiled-board.d.ts +187 -0
  351. package/dist/types/tiled-board.d.ts.map +1 -0
  352. package/dist/types/tiled-board.js +226 -0
  353. package/dist/ui-contract.d.ts +78 -0
  354. package/dist/ui-contract.d.ts.map +1 -0
  355. package/dist/ui-contract.js +15 -0
  356. package/dist/ui-sdk.d.ts +3409 -0
  357. package/dist/utils/interaction-inputs.d.ts +22 -0
  358. package/dist/utils/interaction-inputs.d.ts.map +1 -0
  359. package/dist/utils/interaction-inputs.js +219 -0
  360. package/dist/utils/interaction-labels.d.ts +4 -0
  361. package/dist/utils/interaction-labels.d.ts.map +1 -0
  362. package/dist/utils/interaction-labels.js +18 -0
  363. package/dist/utils/interaction-status.d.ts +15 -0
  364. package/dist/utils/interaction-status.d.ts.map +1 -0
  365. package/dist/utils/interaction-status.js +31 -0
  366. package/package.json +101 -0
  367. package/src/components/ActionButton.tsx +48 -0
  368. package/src/components/ActionPanel.tsx +310 -0
  369. package/src/components/Card.tsx +385 -0
  370. package/src/components/ChromeSuppressionContext.tsx +70 -0
  371. package/src/components/CostDisplay.test.tsx +23 -0
  372. package/src/components/CostDisplay.tsx +145 -0
  373. package/src/components/DiceRoller.tsx +601 -0
  374. package/src/components/Drawer.tsx +179 -0
  375. package/src/components/ErrorBoundary.tsx +119 -0
  376. package/src/components/GameEndDisplay.test.tsx +19 -0
  377. package/src/components/GameEndDisplay.tsx +398 -0
  378. package/src/components/GameSkeleton.tsx +260 -0
  379. package/src/components/Hand.tsx +387 -0
  380. package/src/components/HandDock.tsx +257 -0
  381. package/src/components/InteractionForm.test.tsx +303 -0
  382. package/src/components/InteractionForm.tsx +1029 -0
  383. package/src/components/MoreActions.test.tsx +93 -0
  384. package/src/components/MoreActions.tsx +143 -0
  385. package/src/components/PhaseIndicator.tsx +341 -0
  386. package/src/components/PlayArea.tsx +125 -0
  387. package/src/components/PluginRuntime.tsx +92 -0
  388. package/src/components/PrimaryActionButton.test.tsx +138 -0
  389. package/src/components/PrimaryActionButton.tsx +351 -0
  390. package/src/components/PrimaryButton.tsx +44 -0
  391. package/src/components/PromptDialogHost.tsx +92 -0
  392. package/src/components/ResourceCounter.test.tsx +29 -0
  393. package/src/components/ResourceCounter.tsx +275 -0
  394. package/src/components/ThemedButton.tsx +78 -0
  395. package/src/components/Toast.tsx +251 -0
  396. package/src/components/__fixtures__/ActionButton.fixture.tsx +234 -0
  397. package/src/components/__fixtures__/ActionPanel.fixture.tsx +298 -0
  398. package/src/components/__fixtures__/Card.fixture.tsx +185 -0
  399. package/src/components/__fixtures__/CostDisplay.fixture.tsx +156 -0
  400. package/src/components/__fixtures__/DiceRoller.fixture.tsx +435 -0
  401. package/src/components/__fixtures__/Drawer.fixture.tsx +113 -0
  402. package/src/components/__fixtures__/ErrorBoundary.fixture.tsx +82 -0
  403. package/src/components/__fixtures__/GameEndDisplay.fixture.tsx +188 -0
  404. package/src/components/__fixtures__/GameSkeleton.fixture.tsx +46 -0
  405. package/src/components/__fixtures__/Hand.fixture.tsx +522 -0
  406. package/src/components/__fixtures__/HexGrid.fixture.tsx +1181 -0
  407. package/src/components/__fixtures__/NetworkGraph.fixture.tsx +599 -0
  408. package/src/components/__fixtures__/PhaseIndicator.fixture.tsx +181 -0
  409. package/src/components/__fixtures__/PlayArea.fixture.tsx +221 -0
  410. package/src/components/__fixtures__/ResourceCounter.fixture.tsx +227 -0
  411. package/src/components/__fixtures__/SlotSystem.fixture.tsx +824 -0
  412. package/src/components/__fixtures__/SquareGrid.fixture.tsx +764 -0
  413. package/src/components/__fixtures__/Toast.fixture.tsx +97 -0
  414. package/src/components/__fixtures__/TrackBoard.fixture.tsx +685 -0
  415. package/src/components/__fixtures__/ZoneMap.fixture.tsx +754 -0
  416. package/src/components/board/HexGrid.tsx +1294 -0
  417. package/src/components/board/NetworkGraph.tsx +476 -0
  418. package/src/components/board/SlotSystem.tsx +339 -0
  419. package/src/components/board/SquareGrid.tsx +1165 -0
  420. package/src/components/board/TrackBoard.tsx +496 -0
  421. package/src/components/board/ZoneMap.tsx +448 -0
  422. package/src/components/board/hex-board-view.test.tsx +114 -0
  423. package/src/components/board/hex-board-view.ts +123 -0
  424. package/src/components/board/index.ts +142 -0
  425. package/src/components/board/interaction-accessibility.ts +21 -0
  426. package/src/components/board/target-layer-grids.test.tsx +420 -0
  427. package/src/components/board/target-layer.ts +30 -0
  428. package/src/components/card-render-content.type-test.ts +27 -0
  429. package/src/components/index.ts +208 -0
  430. package/src/components/interaction-dialog-behavior.test.ts +23 -0
  431. package/src/components/interaction-dialog-behavior.ts +22 -0
  432. package/src/components/surfaces/BlockerSurface.test.tsx +158 -0
  433. package/src/components/surfaces/BlockerSurface.tsx +127 -0
  434. package/src/components/surfaces/BoardSurface.tsx +340 -0
  435. package/src/components/surfaces/ChromeSurface.tsx +123 -0
  436. package/src/components/surfaces/ExhaustivenessAudit.tsx +91 -0
  437. package/src/components/surfaces/InboxSurface.test.tsx +149 -0
  438. package/src/components/surfaces/InboxSurface.tsx +245 -0
  439. package/src/components/surfaces/MarketSurface.tsx +544 -0
  440. package/src/components/surfaces/PanelSurface.test.tsx +496 -0
  441. package/src/components/surfaces/PanelSurface.tsx +458 -0
  442. package/src/components/surfaces/PlayerCardsSurface.tsx +525 -0
  443. package/src/components/surfaces/internal/CardZoneFollowUpForm.tsx +35 -0
  444. package/src/components/surfaces/internal/DefaultInteractionButton.tsx +219 -0
  445. package/src/components/surfaces/internal/useCardZoneInteractions.ts +311 -0
  446. package/src/components/surfaces/types.ts +100 -0
  447. package/src/context/ClientParamSchemaContext.tsx +44 -0
  448. package/src/context/InteractionDraftContext.tsx +204 -0
  449. package/src/context/PluginSessionContext.tsx +47 -0
  450. package/src/context/PluginStateContext.tsx +254 -0
  451. package/src/context/RuntimeContext.tsx +96 -0
  452. package/src/defaults/components.tsx +442 -0
  453. package/src/defaults/defaults.test.tsx +230 -0
  454. package/src/defaults/index.ts +1 -0
  455. package/src/errors/ValidationError.ts +29 -0
  456. package/src/helpers/cards.ts +19 -0
  457. package/src/helpers/track-board.ts +211 -0
  458. package/src/hooks/useActivePlayers.ts +19 -0
  459. package/src/hooks/useBoardInteractions.test.tsx +622 -0
  460. package/src/hooks/useBoardInteractions.ts +434 -0
  461. package/src/hooks/useBoardTopology.ts +316 -0
  462. package/src/hooks/useCards.test.tsx +129 -0
  463. package/src/hooks/useCards.ts +10 -0
  464. package/src/hooks/useGameSelector.ts +105 -0
  465. package/src/hooks/useGameView.ts +9 -0
  466. package/src/hooks/useHandLayout.ts +349 -0
  467. package/src/hooks/useHexBoard.ts +74 -0
  468. package/src/hooks/useHexGrid.ts +185 -0
  469. package/src/hooks/useInteractionByKey.ts +349 -0
  470. package/src/hooks/useInteractionHandle.ts +437 -0
  471. package/src/hooks/useIsMobile.ts +35 -0
  472. package/src/hooks/useIsMyTurn.test.tsx +99 -0
  473. package/src/hooks/useIsMyTurn.ts +15 -0
  474. package/src/hooks/useLobby.ts +76 -0
  475. package/src/hooks/useMe.ts +48 -0
  476. package/src/hooks/usePanZoom.ts +278 -0
  477. package/src/hooks/usePlayerInfo.ts +28 -0
  478. package/src/hooks/usePlayerTurnOrder.ts +23 -0
  479. package/src/hooks/usePluginRuntime.test.tsx +102 -0
  480. package/src/hooks/usePluginRuntime.ts +130 -0
  481. package/src/hooks/useSeatInbox.ts +61 -0
  482. package/src/hooks/useSimultaneousPhase.ts +10 -0
  483. package/src/hooks/useSquareBoard.ts +124 -0
  484. package/src/hooks/useSquareGrid.ts +328 -0
  485. package/src/index.test.ts +474 -0
  486. package/src/index.ts +148 -0
  487. package/src/internal/ui/alert.tsx +51 -0
  488. package/src/internal/ui/button.tsx +58 -0
  489. package/src/internal/ui/dialog.tsx +134 -0
  490. package/src/internal/ui/input.tsx +21 -0
  491. package/src/internal/ui/label.tsx +21 -0
  492. package/src/internal/ui/select.tsx +129 -0
  493. package/src/internal/ui/tooltip.tsx +54 -0
  494. package/src/internal/ui/utils.ts +5 -0
  495. package/src/internal.ts +18 -0
  496. package/src/plugin-styles.css +246 -0
  497. package/src/primitives/board.test.tsx +139 -0
  498. package/src/primitives/board.tsx +267 -0
  499. package/src/primitives/game-ui-provider.tsx +35 -0
  500. package/src/primitives/index.ts +83 -0
  501. package/src/primitives/interaction.test.tsx +420 -0
  502. package/src/primitives/interaction.tsx +405 -0
  503. package/src/primitives/phase.test.tsx +82 -0
  504. package/src/primitives/phase.tsx +43 -0
  505. package/src/primitives/player-roster.test.tsx +168 -0
  506. package/src/primitives/player-roster.tsx +301 -0
  507. package/src/primitives/primitive-props.tsx +82 -0
  508. package/src/primitives/prompt.test.tsx +159 -0
  509. package/src/primitives/prompt.tsx +203 -0
  510. package/src/primitives/zone.tsx +113 -0
  511. package/src/reducer.ts +42 -0
  512. package/src/runtime/createPluginRuntimeAPI.ts +605 -0
  513. package/src/theme/ThemeProvider.test.tsx +36 -0
  514. package/src/theme/ThemeProvider.tsx +252 -0
  515. package/src/theme/board.ts +61 -0
  516. package/src/theme/css-vars.ts +105 -0
  517. package/src/theme/derive.ts +240 -0
  518. package/src/theme/index.ts +61 -0
  519. package/src/theme/presets/arcade.ts +261 -0
  520. package/src/theme/presets/studio.ts +261 -0
  521. package/src/theme/presets/tabletop.ts +266 -0
  522. package/src/theme/theme.test.ts +258 -0
  523. package/src/theme/tokens.ts +392 -0
  524. package/src/types/player-state.ts +445 -0
  525. package/src/types/plugin-state.ts +407 -0
  526. package/src/types/reducer-state.ts +24 -0
  527. package/src/types/runtime-api.ts +114 -0
  528. package/src/types/tiled-board.ts +785 -0
  529. package/src/ui-contract.ts +168 -0
  530. package/src/utils/interaction-inputs.test.ts +109 -0
  531. package/src/utils/interaction-inputs.ts +331 -0
  532. package/src/utils/interaction-labels.ts +23 -0
  533. package/src/utils/interaction-status.ts +59 -0
@@ -0,0 +1,419 @@
1
+ import { z } from "zod";
2
+ /**
3
+ * Message schemas from main app to plugin
4
+ * We define them here to avoid circular dependencies with apps/web
5
+ */
6
+ // Main → Plugin: Initialize plugin with session info
7
+ const InitMessageSchema = z.object({
8
+ type: z.literal("init"),
9
+ sessionId: z.string(),
10
+ controllablePlayerIds: z.array(z.string()),
11
+ controllingPlayerId: z.string(),
12
+ userId: z.string().nullable(),
13
+ });
14
+ // Main → Plugin: Health check ping
15
+ const PingMessageSchema = z.object({
16
+ type: z.literal("ping"),
17
+ });
18
+ // Main → Plugin: State sync - sends complete state snapshot
19
+ const StateSyncMessageSchema = z.object({
20
+ type: z.literal("state-sync"),
21
+ syncId: z.number(),
22
+ state: z.custom((data) => {
23
+ return (typeof data === "object" &&
24
+ data !== null &&
25
+ "session" in data &&
26
+ "notifications" in data);
27
+ }),
28
+ });
29
+ // Main → Plugin: Validation result response
30
+ const ValidateInteractionResultMessageSchema = z.object({
31
+ type: z.literal("validate-interaction-result"),
32
+ messageId: z.string(),
33
+ result: z.object({
34
+ valid: z.boolean(),
35
+ errorCode: z.string().nullable().optional(),
36
+ message: z.string().nullable().optional(),
37
+ }),
38
+ });
39
+ const SubmitResultMessageSchema = z.object({
40
+ type: z.literal("submit-result"),
41
+ messageId: z.string(),
42
+ accepted: z.boolean(),
43
+ errorCode: z.string().nullable().optional(),
44
+ message: z.string().nullable().optional(),
45
+ });
46
+ // Union of all messages from main → plugin
47
+ const MainToPluginMessageSchema = z.discriminatedUnion("type", [
48
+ InitMessageSchema,
49
+ PingMessageSchema,
50
+ StateSyncMessageSchema,
51
+ ValidateInteractionResultMessageSchema,
52
+ SubmitResultMessageSchema,
53
+ ]);
54
+ /**
55
+ * Mint a client-side correlation id for a single submitted interaction.
56
+ * This id flows plugin -> host gateway -> backend HTTP header
57
+ * (`X-Dreamboard-Client-Action-Id`), and back to the host via the recorded
58
+ * `version -> actionId` map so the full t0..t8 latency trace can be
59
+ * assembled for Tier-0 input-latency observability. Falls back to a
60
+ * timestamp-seeded pseudo-uuid on environments without `crypto.randomUUID`
61
+ * (older sandboxed browsers in tests) so we never crash the plugin.
62
+ */
63
+ const mintClientActionId = () => {
64
+ const cryptoLike = globalThis.crypto;
65
+ if (cryptoLike?.randomUUID) {
66
+ try {
67
+ return cryptoLike.randomUUID();
68
+ }
69
+ catch {
70
+ // fall through to fallback
71
+ }
72
+ }
73
+ const rand = Math.random().toString(16).slice(2);
74
+ return `cid-${Date.now().toString(16)}-${rand}`;
75
+ };
76
+ const PLUGIN_RUNTIME_SINGLETON_KEY = "__dreamboardPluginRuntimeApi";
77
+ /**
78
+ * Creates a RuntimeAPI implementation for plugin iframes.
79
+ *
80
+ * Architecture (state-sync):
81
+ * - Host maintains all state in GameSessionStore
82
+ * - Host sends complete state snapshots via state-sync messages
83
+ * - Plugin stores received state and notifies subscribers
84
+ * - No buffering needed - plugin only renders when state exists
85
+ *
86
+ * Security:
87
+ * - Plugin runs in sandboxed iframe (no network access, no same-origin)
88
+ * - All backend communication goes through main app
89
+ *
90
+ * @returns PluginRuntimeAPI instance
91
+ */
92
+ export function createPluginRuntimeAPI() {
93
+ const existingRuntime = globalThis[PLUGIN_RUNTIME_SINGLETON_KEY];
94
+ if (existingRuntime) {
95
+ return existingRuntime;
96
+ }
97
+ // State-sync state
98
+ let currentStateSnapshot = null;
99
+ const stateListeners = new Set();
100
+ // Session state
101
+ const sessionState = {
102
+ status: "loading",
103
+ sessionId: null,
104
+ controllablePlayerIds: [],
105
+ controllingPlayerId: null,
106
+ userId: null,
107
+ };
108
+ const sessionStateListeners = new Set();
109
+ // Pending validation requests
110
+ const pendingValidations = new Map();
111
+ let validationIdCounter = 0;
112
+ const pendingSubmissions = new Map();
113
+ let submitIdCounter = 0;
114
+ // Helper functions
115
+ const notifySessionStateChange = () => {
116
+ sessionStateListeners.forEach((listener) => {
117
+ try {
118
+ listener({ ...sessionState });
119
+ }
120
+ catch {
121
+ // Silently catch listener errors
122
+ }
123
+ });
124
+ };
125
+ const notifyStateListeners = () => {
126
+ if (!currentStateSnapshot)
127
+ return;
128
+ const snapshot = currentStateSnapshot;
129
+ stateListeners.forEach((listener) => {
130
+ try {
131
+ listener(snapshot);
132
+ }
133
+ catch {
134
+ // Silently catch listener errors
135
+ }
136
+ });
137
+ };
138
+ const createSubmissionError = (errorCode, message) => {
139
+ const error = new Error(message ?? "Submission failed");
140
+ error.name = "SubmissionError";
141
+ error.errorCode = errorCode;
142
+ return error;
143
+ };
144
+ const submitViaParent = (payload) => new Promise((resolve, reject) => {
145
+ const messageId = `submit-${++submitIdCounter}`;
146
+ pendingSubmissions.set(messageId, { resolve, reject });
147
+ // Plugin-iframe `Date.now()` ships alongside the postMessage
148
+ // as the `t0_click` timestamp for Tier-0 input-latency
149
+ // observability. Date.now() (not performance.now()) is
150
+ // intentional: the iframe and the host share a wall-clock
151
+ // base but not a `performance.now()` origin.
152
+ const clientSubmittedAtMs = Date.now();
153
+ if (payload.clientActionId && typeof performance !== "undefined") {
154
+ try {
155
+ performance.mark(`dreamboard.t0_click.${payload.clientActionId}`, {
156
+ detail: { clientActionId: payload.clientActionId },
157
+ });
158
+ }
159
+ catch {
160
+ // performance.mark detail arg not supported in older browsers; ignore
161
+ }
162
+ }
163
+ window.parent.postMessage({ ...payload, messageId, clientSubmittedAtMs }, "*");
164
+ setTimeout(() => {
165
+ const pending = pendingSubmissions.get(messageId);
166
+ if (!pending) {
167
+ return;
168
+ }
169
+ pendingSubmissions.delete(messageId);
170
+ pending.reject(createSubmissionError("submission-timeout", "Submission request timed out"));
171
+ }, 10000);
172
+ });
173
+ // Message handler
174
+ const handleMessage = (event) => {
175
+ const rawMessage = event.data;
176
+ const parseResult = MainToPluginMessageSchema.safeParse(rawMessage);
177
+ if (!parseResult.success) {
178
+ // Only warn for messages that look like they're meant for us
179
+ if (rawMessage?.type && typeof rawMessage.type === "string") {
180
+ // eslint-disable-next-line no-console
181
+ console.warn("[Plugin RuntimeAPI] Invalid message received:", rawMessage.type);
182
+ }
183
+ return;
184
+ }
185
+ const message = parseResult.data;
186
+ switch (message.type) {
187
+ case "init": {
188
+ // eslint-disable-next-line no-console
189
+ console.log("[Plugin RuntimeAPI] Received init message");
190
+ sessionState.status = "ready";
191
+ sessionState.sessionId = message.sessionId;
192
+ sessionState.controllablePlayerIds =
193
+ message.controllablePlayerIds;
194
+ sessionState.controllingPlayerId =
195
+ message.controllingPlayerId;
196
+ sessionState.userId = message.userId;
197
+ notifySessionStateChange();
198
+ window.parent.postMessage({ type: "ready" }, "*");
199
+ break;
200
+ }
201
+ case "ping": {
202
+ window.parent.postMessage({ type: "pong" }, "*");
203
+ break;
204
+ }
205
+ case "state-sync": {
206
+ // Handle state-sync from host
207
+ // eslint-disable-next-line no-console
208
+ console.log("[Plugin RuntimeAPI] Received state-sync, syncId:", message.syncId);
209
+ // Tier-0 perf: capture `t7_state_sync_received` wall-clock
210
+ // timestamp up-front so the host can stitch it onto the
211
+ // perf HUD via the outgoing state-ack message.
212
+ const clientReceivedAtMs = Date.now();
213
+ if (typeof performance !== "undefined") {
214
+ try {
215
+ performance.mark(`dreamboard.t7_state_sync_received.sync-${message.syncId}`, { detail: { syncId: message.syncId } });
216
+ }
217
+ catch {
218
+ // performance.mark detail arg not supported; ignore
219
+ }
220
+ }
221
+ currentStateSnapshot = message.state;
222
+ // Update session state from snapshot
223
+ if (message.state.session) {
224
+ sessionState.sessionId = message.state.session.sessionId;
225
+ sessionState.controllablePlayerIds =
226
+ message.state.session.controllablePlayerIds;
227
+ sessionState.controllingPlayerId =
228
+ message.state.session.controllingPlayerId;
229
+ sessionState.userId = message.state.session.userId;
230
+ sessionState.status = "ready";
231
+ notifySessionStateChange();
232
+ }
233
+ // Notify state listeners
234
+ notifyStateListeners();
235
+ // Send acknowledgment (carrying t7 timestamp for the host
236
+ // perf HUD; host ignores it when perf is disabled).
237
+ window.parent.postMessage({
238
+ type: "state-ack",
239
+ syncId: message.syncId,
240
+ clientReceivedAtMs,
241
+ }, "*");
242
+ // Tier-0 perf: after notifyStateListeners has kicked React's
243
+ // render, schedule a post-commit microtask + rAF chain so the
244
+ // follow-up `state-rendered` message lands close to when the
245
+ // plugin's DOM would have been painted. `queueMicrotask` is
246
+ // used first because most React listeners finish synchronously;
247
+ // `requestAnimationFrame` then bounces to the next paint tick.
248
+ const schedulePostRender = () => {
249
+ const send = () => {
250
+ const clientRenderedAtMs = Date.now();
251
+ if (typeof performance !== "undefined") {
252
+ try {
253
+ performance.mark(`dreamboard.t8_render_commit.sync-${message.syncId}`, { detail: { syncId: message.syncId } });
254
+ }
255
+ catch {
256
+ // ignore
257
+ }
258
+ }
259
+ window.parent.postMessage({
260
+ type: "state-rendered",
261
+ syncId: message.syncId,
262
+ clientReceivedAtMs,
263
+ clientRenderedAtMs,
264
+ }, "*");
265
+ };
266
+ if (typeof requestAnimationFrame === "function") {
267
+ requestAnimationFrame(send);
268
+ }
269
+ else {
270
+ queueMicrotask(send);
271
+ }
272
+ };
273
+ queueMicrotask(schedulePostRender);
274
+ break;
275
+ }
276
+ case "validate-interaction-result": {
277
+ const resolver = pendingValidations.get(message.messageId);
278
+ if (resolver) {
279
+ pendingValidations.delete(message.messageId);
280
+ resolver({
281
+ valid: message.result.valid,
282
+ errorCode: message.result.errorCode ?? undefined,
283
+ message: message.result.message ?? undefined,
284
+ });
285
+ }
286
+ break;
287
+ }
288
+ case "submit-result": {
289
+ const pending = pendingSubmissions.get(message.messageId);
290
+ if (!pending) {
291
+ break;
292
+ }
293
+ if (typeof performance !== "undefined") {
294
+ try {
295
+ performance.mark(`dreamboard.t3b_ack.${message.messageId}`, {
296
+ detail: { messageId: message.messageId },
297
+ });
298
+ }
299
+ catch {
300
+ // ignore
301
+ }
302
+ }
303
+ pendingSubmissions.delete(message.messageId);
304
+ if (message.accepted) {
305
+ pending.resolve();
306
+ }
307
+ else {
308
+ pending.reject(createSubmissionError(message.errorCode ?? undefined, message.message ?? undefined));
309
+ }
310
+ break;
311
+ }
312
+ }
313
+ };
314
+ window.addEventListener("message", handleMessage);
315
+ // Error handlers
316
+ const sendErrorToParent = (message, code, stack) => {
317
+ // eslint-disable-next-line no-console
318
+ console.error(`[Plugin RuntimeAPI] ${code}:`, message, stack || "");
319
+ window.parent.postMessage({
320
+ type: "error",
321
+ message: stack ? `${message}\n${stack}` : message,
322
+ code,
323
+ }, "*");
324
+ };
325
+ window.onerror = (message, source, lineno, colno, error) => {
326
+ const errorMessage = typeof message === "string" ? message : error?.message || "Unknown error";
327
+ const location = source ? ` at ${source}:${lineno}:${colno}` : "";
328
+ sendErrorToParent(errorMessage + location, "UNCAUGHT_ERROR", error?.stack);
329
+ return false;
330
+ };
331
+ window.onunhandledrejection = (event) => {
332
+ const reason = event.reason;
333
+ const message = reason instanceof Error
334
+ ? reason.message
335
+ : typeof reason === "string"
336
+ ? reason
337
+ : JSON.stringify(reason);
338
+ const stack = reason instanceof Error ? reason.stack : undefined;
339
+ sendErrorToParent(message, "UNHANDLED_REJECTION", stack);
340
+ };
341
+ // eslint-disable-next-line no-console
342
+ console.log("[Plugin RuntimeAPI] ✅ Initialized (state-sync architecture)");
343
+ const runtime = {
344
+ // State-sync methods
345
+ getSnapshot: () => currentStateSnapshot,
346
+ subscribeToState: (listener) => {
347
+ stateListeners.add(listener);
348
+ // Immediately notify with current state if available
349
+ if (currentStateSnapshot) {
350
+ try {
351
+ listener(currentStateSnapshot);
352
+ }
353
+ catch {
354
+ // Silently catch listener errors
355
+ }
356
+ }
357
+ return () => {
358
+ stateListeners.delete(listener);
359
+ };
360
+ },
361
+ validateInteraction: async (playerId, interactionId, params) => {
362
+ return new Promise((resolve) => {
363
+ const messageId = `validate-${++validationIdCounter}`;
364
+ pendingValidations.set(messageId, resolve);
365
+ window.parent.postMessage({
366
+ type: "validate-interaction",
367
+ playerId,
368
+ interactionId,
369
+ params,
370
+ messageId,
371
+ }, "*");
372
+ // Timeout after 10 seconds to avoid hanging forever
373
+ setTimeout(() => {
374
+ if (pendingValidations.has(messageId)) {
375
+ pendingValidations.delete(messageId);
376
+ resolve({
377
+ valid: false,
378
+ errorCode: "validation-timeout",
379
+ message: "Validation request timed out",
380
+ });
381
+ }
382
+ }, 10000);
383
+ });
384
+ },
385
+ submitInteraction: async (playerId, interactionId, params) => submitViaParent({
386
+ type: "interaction",
387
+ playerId,
388
+ interactionId,
389
+ params,
390
+ clientActionId: mintClientActionId(),
391
+ }),
392
+ getSessionState: () => ({ ...sessionState }),
393
+ disconnect: () => {
394
+ window.removeEventListener("message", handleMessage);
395
+ window.onerror = null;
396
+ window.onunhandledrejection = null;
397
+ sessionStateListeners.clear();
398
+ stateListeners.clear();
399
+ pendingValidations.clear();
400
+ pendingSubmissions.clear();
401
+ currentStateSnapshot = null;
402
+ },
403
+ switchPlayer: (playerId) => {
404
+ window.parent.postMessage({ type: "switch-player", playerId }, "*");
405
+ },
406
+ restoreHistory: (entryId) => {
407
+ window.parent.postMessage({ type: "restore-history", entryId }, "*");
408
+ },
409
+ _subscribeToSessionState: (listener) => {
410
+ sessionStateListeners.add(listener);
411
+ listener({ ...sessionState });
412
+ return () => {
413
+ sessionStateListeners.delete(listener);
414
+ };
415
+ },
416
+ };
417
+ globalThis[PLUGIN_RUNTIME_SINGLETON_KEY] = runtime;
418
+ return runtime;
419
+ }
@@ -0,0 +1,98 @@
1
+ import { type CSSProperties, type ReactNode } from "react";
2
+ import { type Theme, type ThemeOverride } from "./tokens.js";
3
+ /**
4
+ * The set of preset theme ids ships in `@dreamboard/ui-sdk`. Authors
5
+ * may register additional ones by passing a fully-resolved {@link Theme}
6
+ * instead of an id.
7
+ */
8
+ export type ThemePresetId = "tabletop" | "arcade" | "studio";
9
+ /**
10
+ * Resolve a preset id (or a full theme) into a {@link Theme}.
11
+ *
12
+ * Used internally by {@link ThemeProvider}; exported so authors who
13
+ * compose at the call site (e.g. for a side-by-side preview) can do
14
+ * the same resolution without mounting a provider.
15
+ */
16
+ export declare function resolveTheme(input: ThemePresetId | Theme | undefined): Theme;
17
+ /** Look up a registered preset by id. Returns `undefined` if missing. */
18
+ export declare function getThemePreset(id: ThemePresetId): Theme | undefined;
19
+ /**
20
+ * Subset returned by {@link useTheme} — the resolved theme plus a
21
+ * stable copy of its CSS-variable map so consumers can reuse it (for
22
+ * example, to apply tokens to a portal that escapes the provider).
23
+ */
24
+ export interface ThemeContextValue {
25
+ /** Fully resolved theme. */
26
+ readonly theme: Theme;
27
+ /** CSS-var map consumable as `style` on a wrapper. */
28
+ readonly cssVars: CSSProperties;
29
+ }
30
+ declare const ThemeCtx: import("react").Context<ThemeContextValue | null>;
31
+ export interface ThemeProviderProps {
32
+ /**
33
+ * Preset id, a full {@link Theme}, or omitted to use the `tabletop`
34
+ * default.
35
+ */
36
+ theme?: ThemePresetId | Theme;
37
+ /**
38
+ * Deep-partial overrides merged onto the resolved base theme. Use
39
+ * for one-off tweaks (e.g. swapping the player palette per game)
40
+ * without writing a full theme.
41
+ */
42
+ override?: ThemeOverride;
43
+ /**
44
+ * Render mode for reduced-motion enforcement.
45
+ *
46
+ * - `auto` (default): respect the user's OS-level preference via
47
+ * `prefers-reduced-motion: reduce`.
48
+ * - `force`: force `motion.reducedMotion = "true"` regardless of the
49
+ * OS preference (useful for screenshot CI).
50
+ * - `ignore`: never override motion (use only when the embedding
51
+ * shell already decides motion behaviour).
52
+ */
53
+ reducedMotion?: "auto" | "force" | "ignore";
54
+ /**
55
+ * Where to mount the wrapper. `block` (default) renders a `div` and
56
+ * applies the CSS variables to it. `none` skips the wrapper entirely
57
+ * — only useful when the parent is already a Dreamboard provider and
58
+ * just needs to publish a different sub-tree (rare).
59
+ */
60
+ as?: "block" | "none";
61
+ /** Extra style merged after the CSS-var map. */
62
+ style?: CSSProperties;
63
+ /** Extra className for the wrapper. */
64
+ className?: string;
65
+ children: ReactNode;
66
+ }
67
+ /**
68
+ * Mounts a Dreamboard theme. The provider:
69
+ *
70
+ * 1. Resolves the preset (or accepts a full theme).
71
+ * 2. Applies any deep-partial `override`.
72
+ * 3. Optionally clamps `motion.reducedMotion` to `"true"` based on the
73
+ * OS-level preference (or a forced override).
74
+ * 4. Serialises the resolved theme into CSS variables on a wrapper
75
+ * element so descendants can read tokens via `useTheme()` *or* via
76
+ * `var(--db-...)` references in their own CSS / Tailwind.
77
+ *
78
+ * Defaults such as `<GameLayout>` mount this for you with the configured
79
+ * theme; mount it manually only when you need a sub-tree to use a different
80
+ * theme (e.g. a dark sheet over a light layout).
81
+ */
82
+ export declare function ThemeProvider({ theme: themeInput, override, reducedMotion, as, style, className, children, }: ThemeProviderProps): import("react/jsx-runtime").JSX.Element;
83
+ /**
84
+ * Read the active {@link Theme} from a {@link ThemeProvider} ancestor.
85
+ *
86
+ * Falls back to the `tabletop` preset (with no overrides) when called
87
+ * outside a provider — components remain renderable in isolated tests
88
+ * and Storybook/Cosmos fixtures without a wrapping shell.
89
+ */
90
+ export declare function useTheme(): Theme;
91
+ /**
92
+ * Read the {@link CSSProperties} that publish the active theme as CSS
93
+ * variables. Useful when porting a theme into a portal or drawer that
94
+ * is rendered outside the provider's DOM subtree.
95
+ */
96
+ export declare function useThemeCssVars(): CSSProperties;
97
+ export { ThemeCtx };
98
+ //# sourceMappingURL=ThemeProvider.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ThemeProvider.d.ts","sourceRoot":"","sources":["../../src/theme/ThemeProvider.tsx"],"names":[],"mappings":"AAAA,OAAO,EAML,KAAK,aAAa,EAClB,KAAK,SAAS,EACf,MAAM,OAAO,CAAC;AAKf,OAAO,EAAc,KAAK,KAAK,EAAE,KAAK,aAAa,EAAE,MAAM,aAAa,CAAC;AAEzE;;;;GAIG;AACH,MAAM,MAAM,aAAa,GAAG,UAAU,GAAG,QAAQ,GAAG,QAAQ,CAAC;AAQ7D;;;;;;GAMG;AACH,wBAAgB,YAAY,CAAC,KAAK,EAAE,aAAa,GAAG,KAAK,GAAG,SAAS,GAAG,KAAK,CAc5E;AAED,yEAAyE;AACzE,wBAAgB,cAAc,CAAC,EAAE,EAAE,aAAa,GAAG,KAAK,GAAG,SAAS,CAEnE;AAED;;;;GAIG;AACH,MAAM,WAAW,iBAAiB;IAChC,4BAA4B;IAC5B,QAAQ,CAAC,KAAK,EAAE,KAAK,CAAC;IACtB,sDAAsD;IACtD,QAAQ,CAAC,OAAO,EAAE,aAAa,CAAC;CACjC;AAED,QAAA,MAAM,QAAQ,mDAAgD,CAAC;AAE/D,MAAM,WAAW,kBAAkB;IACjC;;;OAGG;IACH,KAAK,CAAC,EAAE,aAAa,GAAG,KAAK,CAAC;IAC9B;;;;OAIG;IACH,QAAQ,CAAC,EAAE,aAAa,CAAC;IACzB;;;;;;;;;OASG;IACH,aAAa,CAAC,EAAE,MAAM,GAAG,OAAO,GAAG,QAAQ,CAAC;IAC5C;;;;;OAKG;IACH,EAAE,CAAC,EAAE,OAAO,GAAG,MAAM,CAAC;IACtB,gDAAgD;IAChD,KAAK,CAAC,EAAE,aAAa,CAAC;IACtB,uCAAuC;IACvC,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,SAAS,CAAC;CACrB;AAeD;;;;;;;;;;;;;;GAcG;AACH,wBAAgB,aAAa,CAAC,EAC5B,KAAK,EAAE,UAAU,EACjB,QAAQ,EACR,aAAsB,EACtB,EAAY,EACZ,KAAK,EACL,SAAS,EACT,QAAQ,GACT,EAAE,kBAAkB,2CAgDpB;AAED;;;;;;GAMG;AACH,wBAAgB,QAAQ,IAAI,KAAK,CAGhC;AAED;;;;GAIG;AACH,wBAAgB,eAAe,IAAI,aAAa,CAQ/C;AAED,OAAO,EAAE,QAAQ,EAAE,CAAC"}
@@ -0,0 +1,148 @@
1
+ import { jsx as _jsx } from "react/jsx-runtime";
2
+ import { createContext, useContext, useEffect, useMemo, useState, } from "react";
3
+ import { themeToCssVars } from "./css-vars.js";
4
+ import { arcadeTheme } from "./presets/arcade.js";
5
+ import { studioTheme } from "./presets/studio.js";
6
+ import { tabletopTheme } from "./presets/tabletop.js";
7
+ import { mergeTheme } from "./tokens.js";
8
+ const PRESETS = {
9
+ tabletop: tabletopTheme,
10
+ arcade: arcadeTheme,
11
+ studio: studioTheme,
12
+ };
13
+ /**
14
+ * Resolve a preset id (or a full theme) into a {@link Theme}.
15
+ *
16
+ * Used internally by {@link ThemeProvider}; exported so authors who
17
+ * compose at the call site (e.g. for a side-by-side preview) can do
18
+ * the same resolution without mounting a provider.
19
+ */
20
+ export function resolveTheme(input) {
21
+ if (!input)
22
+ return tabletopTheme;
23
+ if (typeof input === "string") {
24
+ const preset = PRESETS[input];
25
+ if (!preset) {
26
+ throw new Error(`[ui-sdk] Unknown theme preset "${input}". Pass a full Theme object or one of: ${Object.keys(PRESETS).join(", ")}.`);
27
+ }
28
+ return preset;
29
+ }
30
+ return input;
31
+ }
32
+ /** Look up a registered preset by id. Returns `undefined` if missing. */
33
+ export function getThemePreset(id) {
34
+ return PRESETS[id];
35
+ }
36
+ const ThemeCtx = createContext(null);
37
+ function usePrefersReducedMotion() {
38
+ const [prefers, setPrefers] = useState(false);
39
+ useEffect(() => {
40
+ if (typeof window === "undefined" || !window.matchMedia)
41
+ return;
42
+ const media = window.matchMedia("(prefers-reduced-motion: reduce)");
43
+ setPrefers(media.matches);
44
+ const handle = (event) => setPrefers(event.matches);
45
+ media.addEventListener("change", handle);
46
+ return () => media.removeEventListener("change", handle);
47
+ }, []);
48
+ return prefers;
49
+ }
50
+ /**
51
+ * Mounts a Dreamboard theme. The provider:
52
+ *
53
+ * 1. Resolves the preset (or accepts a full theme).
54
+ * 2. Applies any deep-partial `override`.
55
+ * 3. Optionally clamps `motion.reducedMotion` to `"true"` based on the
56
+ * OS-level preference (or a forced override).
57
+ * 4. Serialises the resolved theme into CSS variables on a wrapper
58
+ * element so descendants can read tokens via `useTheme()` *or* via
59
+ * `var(--db-...)` references in their own CSS / Tailwind.
60
+ *
61
+ * Defaults such as `<GameLayout>` mount this for you with the configured
62
+ * theme; mount it manually only when you need a sub-tree to use a different
63
+ * theme (e.g. a dark sheet over a light layout).
64
+ */
65
+ export function ThemeProvider({ theme: themeInput, override, reducedMotion = "auto", as = "block", style, className, children, }) {
66
+ const prefersReduced = usePrefersReducedMotion();
67
+ const value = useMemo(() => {
68
+ const base = resolveTheme(themeInput);
69
+ const merged = mergeTheme(base, override);
70
+ const motionResolved = reducedMotion === "force"
71
+ ? "true"
72
+ : reducedMotion === "ignore"
73
+ ? merged.motion.reducedMotion
74
+ : prefersReduced
75
+ ? "true"
76
+ : merged.motion.reducedMotion;
77
+ const final = motionResolved === merged.motion.reducedMotion
78
+ ? merged
79
+ : {
80
+ ...merged,
81
+ motion: { ...merged.motion, reducedMotion: motionResolved },
82
+ };
83
+ return { theme: final, cssVars: themeToCssVars(final) };
84
+ }, [themeInput, override, reducedMotion, prefersReduced]);
85
+ if (as === "none") {
86
+ return _jsx(ThemeCtx.Provider, { value: value, children: children });
87
+ }
88
+ const wrapperStyle = {
89
+ ...themeToShadcnVars(value.theme),
90
+ ...value.cssVars,
91
+ fontFamily: value.theme.typography.fontFamily.body,
92
+ ...style,
93
+ };
94
+ return (_jsx(ThemeCtx.Provider, { value: value, children: _jsx("div", { "data-dreamboard-theme": value.theme.meta.id, "data-dreamboard-mode": value.theme.meta.mode, "data-dreamboard-reduced-motion": value.theme.motion.reducedMotion, className: className, style: wrapperStyle, children: children }) }));
95
+ }
96
+ /**
97
+ * Read the active {@link Theme} from a {@link ThemeProvider} ancestor.
98
+ *
99
+ * Falls back to the `tabletop` preset (with no overrides) when called
100
+ * outside a provider — components remain renderable in isolated tests
101
+ * and Storybook/Cosmos fixtures without a wrapping shell.
102
+ */
103
+ export function useTheme() {
104
+ const ctx = useContext(ThemeCtx);
105
+ return ctx?.theme ?? tabletopTheme;
106
+ }
107
+ /**
108
+ * Read the {@link CSSProperties} that publish the active theme as CSS
109
+ * variables. Useful when porting a theme into a portal or drawer that
110
+ * is rendered outside the provider's DOM subtree.
111
+ */
112
+ export function useThemeCssVars() {
113
+ const ctx = useContext(ThemeCtx);
114
+ if (ctx)
115
+ return { ...themeToShadcnVars(ctx.theme), ...ctx.cssVars };
116
+ // Match the fallback theme exposed by `useTheme()`.
117
+ return {
118
+ ...themeToShadcnVars(tabletopTheme),
119
+ ...themeToCssVars(tabletopTheme),
120
+ };
121
+ }
122
+ export { ThemeCtx };
123
+ function themeToShadcnVars(theme) {
124
+ const vars = {
125
+ "--background": theme.semantic.surface.app,
126
+ "--foreground": theme.semantic.text.primary,
127
+ "--card": theme.semantic.surface.card,
128
+ "--card-foreground": theme.semantic.text.primary,
129
+ "--popover": theme.semantic.surface.sheet,
130
+ "--popover-foreground": theme.semantic.text.primary,
131
+ "--primary": theme.semantic.intent.primary.solid,
132
+ "--primary-foreground": theme.semantic.intent.primary.on,
133
+ "--secondary": theme.semantic.intent.secondary.solid,
134
+ "--secondary-foreground": theme.semantic.intent.secondary.on,
135
+ "--muted": theme.semantic.surface.inset,
136
+ "--muted-foreground": theme.semantic.text.muted,
137
+ "--accent": theme.semantic.intent.info.soft,
138
+ "--accent-foreground": theme.semantic.intent.info.onSoft,
139
+ "--destructive": theme.semantic.intent.danger.solid,
140
+ "--destructive-foreground": theme.semantic.intent.danger.on,
141
+ "--border": theme.semantic.border.default,
142
+ "--input": theme.semantic.border.default,
143
+ "--ring": theme.semantic.border.focus,
144
+ "--font-sans": theme.typography.fontFamily.body,
145
+ "--font-display": theme.typography.fontFamily.display,
146
+ };
147
+ return vars;
148
+ }