@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,605 @@
1
+ import { z } from "zod";
2
+ import type { PlayerId } from "@dreamboard/manifest-contract";
3
+ import type {
4
+ RuntimeAPI,
5
+ PluginSessionState,
6
+ ValidationResult,
7
+ } from "../types/runtime-api.js";
8
+ import type { PluginStateSnapshot } from "../types/plugin-state.js";
9
+
10
+ /**
11
+ * Message schemas from main app to plugin
12
+ * We define them here to avoid circular dependencies with apps/web
13
+ */
14
+
15
+ // Main → Plugin: Initialize plugin with session info
16
+ const InitMessageSchema = z.object({
17
+ type: z.literal("init"),
18
+ sessionId: z.string(),
19
+ controllablePlayerIds: z.array(z.string()),
20
+ controllingPlayerId: z.string(),
21
+ userId: z.string().nullable(),
22
+ });
23
+
24
+ // Main → Plugin: Health check ping
25
+ const PingMessageSchema = z.object({
26
+ type: z.literal("ping"),
27
+ });
28
+
29
+ // Main → Plugin: State sync - sends complete state snapshot
30
+ const StateSyncMessageSchema = z.object({
31
+ type: z.literal("state-sync"),
32
+ syncId: z.number(),
33
+ state: z.custom<PluginStateSnapshot>((data: unknown) => {
34
+ return (
35
+ typeof data === "object" &&
36
+ data !== null &&
37
+ "session" in data &&
38
+ "notifications" in data
39
+ );
40
+ }),
41
+ });
42
+
43
+ // Main → Plugin: Validation result response
44
+ const ValidateInteractionResultMessageSchema = z.object({
45
+ type: z.literal("validate-interaction-result"),
46
+ messageId: z.string(),
47
+ result: z.object({
48
+ valid: z.boolean(),
49
+ errorCode: z.string().nullable().optional(),
50
+ message: z.string().nullable().optional(),
51
+ }),
52
+ });
53
+
54
+ const SubmitResultMessageSchema = z.object({
55
+ type: z.literal("submit-result"),
56
+ messageId: z.string(),
57
+ accepted: z.boolean(),
58
+ errorCode: z.string().nullable().optional(),
59
+ message: z.string().nullable().optional(),
60
+ });
61
+
62
+ // Union of all messages from main → plugin
63
+ const MainToPluginMessageSchema = z.discriminatedUnion("type", [
64
+ InitMessageSchema,
65
+ PingMessageSchema,
66
+ StateSyncMessageSchema,
67
+ ValidateInteractionResultMessageSchema,
68
+ SubmitResultMessageSchema,
69
+ ]);
70
+
71
+ /**
72
+ * Extended RuntimeAPI with plugin-specific methods for state-sync architecture.
73
+ */
74
+ export interface PluginRuntimeAPI extends RuntimeAPI {
75
+ /**
76
+ * Get the current state snapshot.
77
+ * Returns null if no state-sync has been received yet.
78
+ *
79
+ * @example
80
+ * ```typescript
81
+ * const snapshot = runtime.getSnapshot();
82
+ * if (snapshot?.view) {
83
+ * console.log('Current view:', snapshot.view);
84
+ * }
85
+ * ```
86
+ */
87
+ getSnapshot: () => PluginStateSnapshot | null;
88
+
89
+ /**
90
+ * Subscribe to state changes from state-sync messages.
91
+ * Called whenever the host sends a new state-sync.
92
+ *
93
+ * @param listener - Callback invoked with new state snapshot
94
+ * @returns Unsubscribe function
95
+ *
96
+ * @example
97
+ * ```typescript
98
+ * const unsubscribe = runtime.subscribeToState((state) => {
99
+ * console.log('New phase:', state.gameplay.currentPhase);
100
+ * });
101
+ * ```
102
+ */
103
+ subscribeToState: (
104
+ listener: (state: PluginStateSnapshot) => void,
105
+ ) => () => void;
106
+
107
+ /** Internal API for RuntimeContext to subscribe to session state changes */
108
+ _subscribeToSessionState: (
109
+ listener: (state: PluginSessionState) => void,
110
+ ) => () => void;
111
+
112
+ /**
113
+ * Request to restore game state to a previous history entry.
114
+ * Only works if the user is the host.
115
+ *
116
+ * @param entryId - ID of the history entry to restore to
117
+ *
118
+ * @example
119
+ * ```typescript
120
+ * // Restore to a previous state
121
+ * runtime.restoreHistory?.('entry-abc-123');
122
+ * ```
123
+ */
124
+ restoreHistory?: (entryId: string) => void;
125
+ }
126
+
127
+ /**
128
+ * Mint a client-side correlation id for a single submitted interaction.
129
+ * This id flows plugin -> host gateway -> backend HTTP header
130
+ * (`X-Dreamboard-Client-Action-Id`), and back to the host via the recorded
131
+ * `version -> actionId` map so the full t0..t8 latency trace can be
132
+ * assembled for Tier-0 input-latency observability. Falls back to a
133
+ * timestamp-seeded pseudo-uuid on environments without `crypto.randomUUID`
134
+ * (older sandboxed browsers in tests) so we never crash the plugin.
135
+ */
136
+ const mintClientActionId = (): string => {
137
+ const cryptoLike = (
138
+ globalThis as typeof globalThis & {
139
+ crypto?: { randomUUID?: () => string };
140
+ }
141
+ ).crypto;
142
+ if (cryptoLike?.randomUUID) {
143
+ try {
144
+ return cryptoLike.randomUUID();
145
+ } catch {
146
+ // fall through to fallback
147
+ }
148
+ }
149
+ const rand = Math.random().toString(16).slice(2);
150
+ return `cid-${Date.now().toString(16)}-${rand}`;
151
+ };
152
+
153
+ const PLUGIN_RUNTIME_SINGLETON_KEY = "__dreamboardPluginRuntimeApi";
154
+
155
+ type PluginRuntimeGlobal = typeof globalThis & {
156
+ [PLUGIN_RUNTIME_SINGLETON_KEY]?: PluginRuntimeAPI;
157
+ };
158
+
159
+ /**
160
+ * Creates a RuntimeAPI implementation for plugin iframes.
161
+ *
162
+ * Architecture (state-sync):
163
+ * - Host maintains all state in GameSessionStore
164
+ * - Host sends complete state snapshots via state-sync messages
165
+ * - Plugin stores received state and notifies subscribers
166
+ * - No buffering needed - plugin only renders when state exists
167
+ *
168
+ * Security:
169
+ * - Plugin runs in sandboxed iframe (no network access, no same-origin)
170
+ * - All backend communication goes through main app
171
+ *
172
+ * @returns PluginRuntimeAPI instance
173
+ */
174
+ export function createPluginRuntimeAPI(): PluginRuntimeAPI {
175
+ const existingRuntime = (globalThis as PluginRuntimeGlobal)[
176
+ PLUGIN_RUNTIME_SINGLETON_KEY
177
+ ];
178
+ if (existingRuntime) {
179
+ return existingRuntime;
180
+ }
181
+
182
+ // State-sync state
183
+ let currentStateSnapshot: PluginStateSnapshot | null = null;
184
+ const stateListeners = new Set<(state: PluginStateSnapshot) => void>();
185
+
186
+ // Session state
187
+ const sessionState: PluginSessionState = {
188
+ status: "loading",
189
+ sessionId: null,
190
+ controllablePlayerIds: [],
191
+ controllingPlayerId: null,
192
+ userId: null,
193
+ };
194
+ const sessionStateListeners = new Set<(state: PluginSessionState) => void>();
195
+
196
+ // Pending validation requests
197
+ const pendingValidations = new Map<
198
+ string,
199
+ (result: ValidationResult) => void
200
+ >();
201
+ let validationIdCounter = 0;
202
+ const pendingSubmissions = new Map<
203
+ string,
204
+ {
205
+ resolve: () => void;
206
+ reject: (error: Error & { errorCode?: string }) => void;
207
+ }
208
+ >();
209
+ let submitIdCounter = 0;
210
+
211
+ // Helper functions
212
+ const notifySessionStateChange = () => {
213
+ sessionStateListeners.forEach((listener) => {
214
+ try {
215
+ listener({ ...sessionState });
216
+ } catch {
217
+ // Silently catch listener errors
218
+ }
219
+ });
220
+ };
221
+
222
+ const notifyStateListeners = () => {
223
+ if (!currentStateSnapshot) return;
224
+ const snapshot = currentStateSnapshot;
225
+ stateListeners.forEach((listener) => {
226
+ try {
227
+ listener(snapshot);
228
+ } catch {
229
+ // Silently catch listener errors
230
+ }
231
+ });
232
+ };
233
+
234
+ const createSubmissionError = (
235
+ errorCode?: string,
236
+ message?: string,
237
+ ): Error & { errorCode?: string } => {
238
+ const error = new Error(message ?? "Submission failed") as Error & {
239
+ errorCode?: string;
240
+ name: string;
241
+ };
242
+ error.name = "SubmissionError";
243
+ error.errorCode = errorCode;
244
+ return error;
245
+ };
246
+
247
+ const submitViaParent = (payload: {
248
+ type: "interaction";
249
+ playerId: PlayerId;
250
+ interactionId: string;
251
+ params: unknown;
252
+ clientActionId?: string;
253
+ }): Promise<void> =>
254
+ new Promise((resolve, reject) => {
255
+ const messageId = `submit-${++submitIdCounter}`;
256
+ pendingSubmissions.set(messageId, { resolve, reject });
257
+
258
+ // Plugin-iframe `Date.now()` ships alongside the postMessage
259
+ // as the `t0_click` timestamp for Tier-0 input-latency
260
+ // observability. Date.now() (not performance.now()) is
261
+ // intentional: the iframe and the host share a wall-clock
262
+ // base but not a `performance.now()` origin.
263
+ const clientSubmittedAtMs = Date.now();
264
+
265
+ if (payload.clientActionId && typeof performance !== "undefined") {
266
+ try {
267
+ performance.mark(`dreamboard.t0_click.${payload.clientActionId}`, {
268
+ detail: { clientActionId: payload.clientActionId },
269
+ });
270
+ } catch {
271
+ // performance.mark detail arg not supported in older browsers; ignore
272
+ }
273
+ }
274
+
275
+ window.parent.postMessage(
276
+ { ...payload, messageId, clientSubmittedAtMs },
277
+ "*",
278
+ );
279
+
280
+ setTimeout(() => {
281
+ const pending = pendingSubmissions.get(messageId);
282
+ if (!pending) {
283
+ return;
284
+ }
285
+ pendingSubmissions.delete(messageId);
286
+ pending.reject(
287
+ createSubmissionError(
288
+ "submission-timeout",
289
+ "Submission request timed out",
290
+ ),
291
+ );
292
+ }, 10000);
293
+ });
294
+
295
+ // Message handler
296
+ const handleMessage = (event: MessageEvent) => {
297
+ const rawMessage = event.data;
298
+
299
+ const parseResult = MainToPluginMessageSchema.safeParse(rawMessage);
300
+ if (!parseResult.success) {
301
+ // Only warn for messages that look like they're meant for us
302
+ if (rawMessage?.type && typeof rawMessage.type === "string") {
303
+ // eslint-disable-next-line no-console
304
+ console.warn(
305
+ "[Plugin RuntimeAPI] Invalid message received:",
306
+ rawMessage.type,
307
+ );
308
+ }
309
+ return;
310
+ }
311
+
312
+ const message = parseResult.data;
313
+
314
+ switch (message.type) {
315
+ case "init": {
316
+ // eslint-disable-next-line no-console
317
+ console.log("[Plugin RuntimeAPI] Received init message");
318
+
319
+ sessionState.status = "ready";
320
+ sessionState.sessionId = message.sessionId;
321
+ sessionState.controllablePlayerIds =
322
+ message.controllablePlayerIds as PlayerId[];
323
+ sessionState.controllingPlayerId =
324
+ message.controllingPlayerId as PlayerId;
325
+ sessionState.userId = message.userId;
326
+
327
+ notifySessionStateChange();
328
+ window.parent.postMessage({ type: "ready" }, "*");
329
+ break;
330
+ }
331
+
332
+ case "ping": {
333
+ window.parent.postMessage({ type: "pong" }, "*");
334
+ break;
335
+ }
336
+
337
+ case "state-sync": {
338
+ // Handle state-sync from host
339
+ // eslint-disable-next-line no-console
340
+ console.log(
341
+ "[Plugin RuntimeAPI] Received state-sync, syncId:",
342
+ message.syncId,
343
+ );
344
+
345
+ // Tier-0 perf: capture `t7_state_sync_received` wall-clock
346
+ // timestamp up-front so the host can stitch it onto the
347
+ // perf HUD via the outgoing state-ack message.
348
+ const clientReceivedAtMs = Date.now();
349
+ if (typeof performance !== "undefined") {
350
+ try {
351
+ performance.mark(
352
+ `dreamboard.t7_state_sync_received.sync-${message.syncId}`,
353
+ { detail: { syncId: message.syncId } },
354
+ );
355
+ } catch {
356
+ // performance.mark detail arg not supported; ignore
357
+ }
358
+ }
359
+
360
+ currentStateSnapshot = message.state;
361
+
362
+ // Update session state from snapshot
363
+ if (message.state.session) {
364
+ sessionState.sessionId = message.state.session.sessionId;
365
+ sessionState.controllablePlayerIds =
366
+ message.state.session.controllablePlayerIds;
367
+ sessionState.controllingPlayerId =
368
+ message.state.session.controllingPlayerId;
369
+ sessionState.userId = message.state.session.userId;
370
+ sessionState.status = "ready";
371
+ notifySessionStateChange();
372
+ }
373
+
374
+ // Notify state listeners
375
+ notifyStateListeners();
376
+
377
+ // Send acknowledgment (carrying t7 timestamp for the host
378
+ // perf HUD; host ignores it when perf is disabled).
379
+ window.parent.postMessage(
380
+ {
381
+ type: "state-ack",
382
+ syncId: message.syncId,
383
+ clientReceivedAtMs,
384
+ },
385
+ "*",
386
+ );
387
+
388
+ // Tier-0 perf: after notifyStateListeners has kicked React's
389
+ // render, schedule a post-commit microtask + rAF chain so the
390
+ // follow-up `state-rendered` message lands close to when the
391
+ // plugin's DOM would have been painted. `queueMicrotask` is
392
+ // used first because most React listeners finish synchronously;
393
+ // `requestAnimationFrame` then bounces to the next paint tick.
394
+ const schedulePostRender = () => {
395
+ const send = () => {
396
+ const clientRenderedAtMs = Date.now();
397
+ if (typeof performance !== "undefined") {
398
+ try {
399
+ performance.mark(
400
+ `dreamboard.t8_render_commit.sync-${message.syncId}`,
401
+ { detail: { syncId: message.syncId } },
402
+ );
403
+ } catch {
404
+ // ignore
405
+ }
406
+ }
407
+ window.parent.postMessage(
408
+ {
409
+ type: "state-rendered",
410
+ syncId: message.syncId,
411
+ clientReceivedAtMs,
412
+ clientRenderedAtMs,
413
+ },
414
+ "*",
415
+ );
416
+ };
417
+ if (typeof requestAnimationFrame === "function") {
418
+ requestAnimationFrame(send);
419
+ } else {
420
+ queueMicrotask(send);
421
+ }
422
+ };
423
+ queueMicrotask(schedulePostRender);
424
+
425
+ break;
426
+ }
427
+
428
+ case "validate-interaction-result": {
429
+ const resolver = pendingValidations.get(message.messageId);
430
+ if (resolver) {
431
+ pendingValidations.delete(message.messageId);
432
+ resolver({
433
+ valid: message.result.valid,
434
+ errorCode: message.result.errorCode ?? undefined,
435
+ message: message.result.message ?? undefined,
436
+ });
437
+ }
438
+ break;
439
+ }
440
+
441
+ case "submit-result": {
442
+ const pending = pendingSubmissions.get(message.messageId);
443
+ if (!pending) {
444
+ break;
445
+ }
446
+
447
+ if (typeof performance !== "undefined") {
448
+ try {
449
+ performance.mark(`dreamboard.t3b_ack.${message.messageId}`, {
450
+ detail: { messageId: message.messageId },
451
+ });
452
+ } catch {
453
+ // ignore
454
+ }
455
+ }
456
+
457
+ pendingSubmissions.delete(message.messageId);
458
+ if (message.accepted) {
459
+ pending.resolve();
460
+ } else {
461
+ pending.reject(
462
+ createSubmissionError(
463
+ message.errorCode ?? undefined,
464
+ message.message ?? undefined,
465
+ ),
466
+ );
467
+ }
468
+ break;
469
+ }
470
+ }
471
+ };
472
+
473
+ window.addEventListener("message", handleMessage);
474
+
475
+ // Error handlers
476
+ const sendErrorToParent = (message: string, code: string, stack?: string) => {
477
+ // eslint-disable-next-line no-console
478
+ console.error(`[Plugin RuntimeAPI] ${code}:`, message, stack || "");
479
+ window.parent.postMessage(
480
+ {
481
+ type: "error",
482
+ message: stack ? `${message}\n${stack}` : message,
483
+ code,
484
+ },
485
+ "*",
486
+ );
487
+ };
488
+
489
+ window.onerror = (message, source, lineno, colno, error) => {
490
+ const errorMessage =
491
+ typeof message === "string" ? message : error?.message || "Unknown error";
492
+ const location = source ? ` at ${source}:${lineno}:${colno}` : "";
493
+ sendErrorToParent(errorMessage + location, "UNCAUGHT_ERROR", error?.stack);
494
+ return false;
495
+ };
496
+
497
+ window.onunhandledrejection = (event: PromiseRejectionEvent) => {
498
+ const reason = event.reason;
499
+ const message =
500
+ reason instanceof Error
501
+ ? reason.message
502
+ : typeof reason === "string"
503
+ ? reason
504
+ : JSON.stringify(reason);
505
+ const stack = reason instanceof Error ? reason.stack : undefined;
506
+ sendErrorToParent(message, "UNHANDLED_REJECTION", stack);
507
+ };
508
+
509
+ // eslint-disable-next-line no-console
510
+ console.log("[Plugin RuntimeAPI] ✅ Initialized (state-sync architecture)");
511
+
512
+ const runtime: PluginRuntimeAPI = {
513
+ // State-sync methods
514
+ getSnapshot: () => currentStateSnapshot,
515
+
516
+ subscribeToState: (listener) => {
517
+ stateListeners.add(listener);
518
+ // Immediately notify with current state if available
519
+ if (currentStateSnapshot) {
520
+ try {
521
+ listener(currentStateSnapshot);
522
+ } catch {
523
+ // Silently catch listener errors
524
+ }
525
+ }
526
+ return () => {
527
+ stateListeners.delete(listener);
528
+ };
529
+ },
530
+
531
+ validateInteraction: async (playerId, interactionId, params) => {
532
+ return new Promise((resolve) => {
533
+ const messageId = `validate-${++validationIdCounter}`;
534
+ pendingValidations.set(messageId, resolve);
535
+
536
+ window.parent.postMessage(
537
+ {
538
+ type: "validate-interaction",
539
+ playerId,
540
+ interactionId,
541
+ params,
542
+ messageId,
543
+ },
544
+ "*",
545
+ );
546
+
547
+ // Timeout after 10 seconds to avoid hanging forever
548
+ setTimeout(() => {
549
+ if (pendingValidations.has(messageId)) {
550
+ pendingValidations.delete(messageId);
551
+ resolve({
552
+ valid: false,
553
+ errorCode: "validation-timeout",
554
+ message: "Validation request timed out",
555
+ });
556
+ }
557
+ }, 10000);
558
+ });
559
+ },
560
+
561
+ submitInteraction: async (playerId, interactionId, params) =>
562
+ submitViaParent({
563
+ type: "interaction",
564
+ playerId,
565
+ interactionId,
566
+ params,
567
+ clientActionId: mintClientActionId(),
568
+ }),
569
+
570
+ getSessionState: () => ({ ...sessionState }),
571
+
572
+ disconnect: () => {
573
+ window.removeEventListener("message", handleMessage);
574
+ window.onerror = null;
575
+ window.onunhandledrejection = null;
576
+ sessionStateListeners.clear();
577
+ stateListeners.clear();
578
+ pendingValidations.clear();
579
+ pendingSubmissions.clear();
580
+ currentStateSnapshot = null;
581
+ },
582
+
583
+ switchPlayer: (playerId: PlayerId) => {
584
+ window.parent.postMessage({ type: "switch-player", playerId }, "*");
585
+ },
586
+
587
+ restoreHistory: (entryId: string) => {
588
+ window.parent.postMessage({ type: "restore-history", entryId }, "*");
589
+ },
590
+
591
+ _subscribeToSessionState: (
592
+ listener: (state: PluginSessionState) => void,
593
+ ) => {
594
+ sessionStateListeners.add(listener);
595
+ listener({ ...sessionState });
596
+ return () => {
597
+ sessionStateListeners.delete(listener);
598
+ };
599
+ },
600
+ };
601
+
602
+ (globalThis as PluginRuntimeGlobal)[PLUGIN_RUNTIME_SINGLETON_KEY] = runtime;
603
+
604
+ return runtime;
605
+ }
@@ -0,0 +1,36 @@
1
+ import { expect, test } from "bun:test";
2
+ import { renderToString } from "react-dom/server";
3
+ import { ThemeProvider } from "./ThemeProvider.js";
4
+
5
+ test("ThemeProvider publishes shadcn vars and body font on the themed subtree", () => {
6
+ const html = renderToString(
7
+ <ThemeProvider
8
+ override={{
9
+ typography: {
10
+ fontFamily: {
11
+ body: '"Test Body", system-ui, sans-serif',
12
+ display: '"Test Display", serif',
13
+ },
14
+ },
15
+ semantic: {
16
+ intent: {
17
+ primary: {
18
+ solid: "#123456",
19
+ },
20
+ },
21
+ },
22
+ }}
23
+ >
24
+ <span>content</span>
25
+ </ThemeProvider>,
26
+ );
27
+
28
+ expect(html).toContain(
29
+ "--font-sans:&quot;Test Body&quot;, system-ui, sans-serif",
30
+ );
31
+ expect(html).toContain("--font-display:&quot;Test Display&quot;, serif");
32
+ expect(html).toContain("--primary:#123456");
33
+ expect(html).toContain(
34
+ "font-family:&quot;Test Body&quot;, system-ui, sans-serif",
35
+ );
36
+ });