@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,204 @@
1
+ import { createContext, useContext, useMemo, type ReactNode } from "react";
2
+ import { useStore } from "zustand";
3
+ import { createStore, type StoreApi } from "zustand/vanilla";
4
+ import { useShallow } from "zustand/shallow";
5
+
6
+ type Draft = Readonly<Record<string, unknown>>;
7
+
8
+ const EMPTY_DRAFT: Draft = Object.freeze({});
9
+
10
+ interface DraftState {
11
+ drafts: Readonly<Record<string, Draft>>;
12
+ arms: Readonly<Record<string, string>>;
13
+ submitting: Readonly<Record<string, true>>;
14
+ }
15
+
16
+ /**
17
+ * Imperative API exposed to `useInteractionHandle` and `<BoardSurface>`.
18
+ * Intentionally small; the vanilla zustand store underneath powers
19
+ * fine-grained subscriptions via {@link useDraft} and {@link useArmed}.
20
+ */
21
+ export interface InteractionUiStore {
22
+ /** Read the current draft for an interaction id. Never undefined. */
23
+ getDraft(interactionId: string): Draft;
24
+ /** Merge a single input key into the draft. Creates the draft if needed. */
25
+ setInput(interactionId: string, key: string, value: unknown): void;
26
+ /** Clear a single input, or the whole draft if `key` is omitted. */
27
+ clearInput(interactionId: string, key?: string): void;
28
+ /** Clear every draft and arming state. */
29
+ clearAll(): void;
30
+ /** Which interaction (if any) is currently armed on the given surface. */
31
+ getArmed(surface: string): string | null;
32
+ /** Arm a specific interaction on a surface. Pass `null` to disarm. */
33
+ arm(surface: string, interactionId: string | null): void;
34
+ /** True while a local submission is in flight before the host echo arrives. */
35
+ isSubmitting(interactionId: string): boolean;
36
+ /** Mark or clear a local submission in flight. */
37
+ setSubmitting(interactionId: string, submitting: boolean): void;
38
+ }
39
+
40
+ /** Vanilla zustand store implementing {@link InteractionUiStore}. */
41
+ export type InteractionUiStoreApi = StoreApi<DraftState> & InteractionUiStore;
42
+
43
+ export function createInteractionUiStore(): InteractionUiStoreApi {
44
+ const store = createStore<DraftState>()(() => ({
45
+ drafts: {},
46
+ arms: {},
47
+ submitting: {},
48
+ }));
49
+
50
+ const api: InteractionUiStore = {
51
+ getDraft(interactionId) {
52
+ return store.getState().drafts[interactionId] ?? EMPTY_DRAFT;
53
+ },
54
+ setInput(interactionId, key, value) {
55
+ store.setState((prev) => {
56
+ const current = prev.drafts[interactionId];
57
+ if (current && current[key] === value) return prev;
58
+ return {
59
+ ...prev,
60
+ drafts: {
61
+ ...prev.drafts,
62
+ [interactionId]: { ...(current ?? {}), [key]: value },
63
+ },
64
+ };
65
+ });
66
+ },
67
+ clearInput(interactionId, key) {
68
+ store.setState((prev) => {
69
+ const current = prev.drafts[interactionId];
70
+ if (!current) return prev;
71
+ if (key === undefined) {
72
+ const { [interactionId]: _omit, ...rest } = prev.drafts;
73
+ return { ...prev, drafts: rest };
74
+ }
75
+ if (!(key in current)) return prev;
76
+ const { [key]: _omitKey, ...remainingKeys } = current;
77
+ if (Object.keys(remainingKeys).length === 0) {
78
+ const { [interactionId]: _omit, ...rest } = prev.drafts;
79
+ return { ...prev, drafts: rest };
80
+ }
81
+ return {
82
+ ...prev,
83
+ drafts: { ...prev.drafts, [interactionId]: remainingKeys },
84
+ };
85
+ });
86
+ },
87
+ clearAll() {
88
+ store.setState((prev) => {
89
+ if (
90
+ Object.keys(prev.drafts).length === 0 &&
91
+ Object.keys(prev.arms).length === 0 &&
92
+ Object.keys(prev.submitting).length === 0
93
+ ) {
94
+ return prev;
95
+ }
96
+ return { drafts: {}, arms: {}, submitting: {} };
97
+ });
98
+ },
99
+ getArmed(surface) {
100
+ return store.getState().arms[surface] ?? null;
101
+ },
102
+ arm(surface, interactionId) {
103
+ store.setState((prev) => {
104
+ const current = prev.arms[surface] ?? null;
105
+ if (current === interactionId) return prev;
106
+ if (interactionId === null) {
107
+ const { [surface]: _omit, ...rest } = prev.arms;
108
+ return { ...prev, arms: rest };
109
+ }
110
+ return { ...prev, arms: { ...prev.arms, [surface]: interactionId } };
111
+ });
112
+ },
113
+ isSubmitting(interactionId) {
114
+ return store.getState().submitting[interactionId] === true;
115
+ },
116
+ setSubmitting(interactionId, submitting) {
117
+ store.setState((prev) => {
118
+ const current = prev.submitting[interactionId] === true;
119
+ if (current === submitting) return prev;
120
+ if (!submitting) {
121
+ const { [interactionId]: _omit, ...rest } = prev.submitting;
122
+ return { ...prev, submitting: rest };
123
+ }
124
+ return {
125
+ ...prev,
126
+ submitting: { ...prev.submitting, [interactionId]: true },
127
+ };
128
+ });
129
+ },
130
+ };
131
+
132
+ return Object.assign(store, api);
133
+ }
134
+
135
+ const InteractionUiCtx = createContext<InteractionUiStoreApi | null>(null);
136
+
137
+ /**
138
+ * React provider that holds draft input state shared across every surface
139
+ * inside the tree. Auto-installed by `<PluginRuntime>`; authors rarely
140
+ * mount it directly. Mount manually when rendering surface components in
141
+ * isolation (e.g., Storybook, snapshot tests).
142
+ *
143
+ * ```tsx
144
+ * <InteractionUiProvider>
145
+ * <PanelSurface />
146
+ * <BoardSurface>{(ctx) => <Board ctx={ctx} />}</BoardSurface>
147
+ * </InteractionUiProvider>
148
+ * ```
149
+ */
150
+ export function InteractionUiProvider({
151
+ children,
152
+ store,
153
+ }: {
154
+ children: ReactNode;
155
+ store?: InteractionUiStoreApi;
156
+ }) {
157
+ const ownedStore = useMemo(() => createInteractionUiStore(), []);
158
+ return (
159
+ <InteractionUiCtx.Provider value={store ?? ownedStore}>
160
+ {children}
161
+ </InteractionUiCtx.Provider>
162
+ );
163
+ }
164
+
165
+ /**
166
+ * Access the active draft store. Falls back to an inert in-memory store
167
+ * when no provider is mounted, so surface hooks remain callable in bare
168
+ * test harnesses without crashing — draft state simply isn't shared
169
+ * across components in that case.
170
+ */
171
+ export function useInteractionUiStore(): InteractionUiStoreApi {
172
+ const ctx = useContext(InteractionUiCtx);
173
+ const fallback = useMemo(() => createInteractionUiStore(), []);
174
+ return ctx ?? fallback;
175
+ }
176
+
177
+ /**
178
+ * Subscribe to the draft for a single interaction id with per-slice
179
+ * re-renders. Returns a stable empty object when the draft is unset.
180
+ */
181
+ export function useInteractionDraft(interactionId: string): Draft {
182
+ const store = useInteractionUiStore();
183
+ return useStore(
184
+ store,
185
+ useShallow(
186
+ (state: DraftState) => state.drafts[interactionId] ?? EMPTY_DRAFT,
187
+ ),
188
+ );
189
+ }
190
+
191
+ /** Subscribe to the armed interaction id on a given surface. */
192
+ export function useArmedInteraction(surface: string): string | null {
193
+ const store = useInteractionUiStore();
194
+ return useStore(store, (state: DraftState) => state.arms[surface] ?? null);
195
+ }
196
+
197
+ /** Subscribe to local submitting status for a single interaction key. */
198
+ export function useInteractionSubmitting(interactionId: string): boolean {
199
+ const store = useInteractionUiStore();
200
+ return useStore(
201
+ store,
202
+ (state: DraftState) => state.submitting[interactionId] === true,
203
+ );
204
+ }
@@ -0,0 +1,47 @@
1
+ import { createContext, useContext } from "react";
2
+ import type { PluginSessionState } from "../types/runtime-api";
3
+
4
+ /**
5
+ * Context for plugin session metadata.
6
+ * This context is provided by the RuntimeContext after receiving init message from parent.
7
+ */
8
+ export const PluginSessionContext = createContext<PluginSessionState | null>(
9
+ null,
10
+ );
11
+
12
+ /**
13
+ * Hook to access plugin session metadata.
14
+ * Returns session initialization status and IDs.
15
+ *
16
+ * @returns Plugin session state with status, sessionId, controllablePlayerIds, and controllingPlayerId
17
+ *
18
+ * @example
19
+ * ```tsx
20
+ * function MyPluginComponent() {
21
+ * const { status, sessionId, controllablePlayerIds, controllingPlayerId } = usePluginSession();
22
+ *
23
+ * if (status === "loading") {
24
+ * return <div>Initializing...</div>;
25
+ * }
26
+ *
27
+ * return (
28
+ * <div>
29
+ * <p>Session: {sessionId}</p>
30
+ * <p>Can control: {controllablePlayerIds.join(", ")}</p>
31
+ * <p>Currently controlling: {controllingPlayerId}</p>
32
+ * </div>
33
+ * );
34
+ * }
35
+ * ```
36
+ */
37
+ export function usePluginSession(): PluginSessionState {
38
+ const context = useContext(PluginSessionContext);
39
+
40
+ if (context === null) {
41
+ throw new Error(
42
+ "usePluginSession must be used within a PluginSessionContext.Provider (provided by RuntimeContext)",
43
+ );
44
+ }
45
+
46
+ return context;
47
+ }
@@ -0,0 +1,254 @@
1
+ import {
2
+ default as React,
3
+ createContext,
4
+ useContext,
5
+ useState,
6
+ useEffect,
7
+ useMemo,
8
+ useSyncExternalStore,
9
+ } from "react";
10
+ import type { PluginStateSnapshot } from "../types/plugin-state.js";
11
+ import { useRuntimeContext } from "./RuntimeContext.js";
12
+ import type { PluginRuntimeAPI } from "../runtime/createPluginRuntimeAPI.js";
13
+
14
+ /**
15
+ * React Context for providing plugin state from state-sync messages.
16
+ * This is the new architecture where the host app maintains state and syncs to plugin.
17
+ */
18
+ const PluginStateContext = createContext<PluginStateSnapshot | null>(null);
19
+
20
+ /**
21
+ * Loading component shown while waiting for initial state
22
+ */
23
+ function DefaultLoadingScreen() {
24
+ return (
25
+ <div
26
+ style={{
27
+ display: "flex",
28
+ alignItems: "center",
29
+ justifyContent: "center",
30
+ height: "100%",
31
+ width: "100%",
32
+ color: "#666",
33
+ }}
34
+ >
35
+ <div style={{ textAlign: "center" }}>
36
+ <div
37
+ style={{
38
+ width: "40px",
39
+ height: "40px",
40
+ border: "3px solid #e0e0e0",
41
+ borderTopColor: "#666",
42
+ borderRadius: "50%",
43
+ animation: "spin 1s linear infinite",
44
+ margin: "0 auto 12px",
45
+ }}
46
+ />
47
+ <style>{`@keyframes spin { to { transform: rotate(360deg); } }`}</style>
48
+ <p>Loading game view...</p>
49
+ </div>
50
+ </div>
51
+ );
52
+ }
53
+
54
+ export interface PluginStateProviderProps {
55
+ children: React.ReactNode;
56
+ /**
57
+ * Custom loading component to show while waiting for state
58
+ * @default DefaultLoadingScreen
59
+ */
60
+ loadingComponent?: React.ReactNode;
61
+ }
62
+
63
+ /**
64
+ * PluginStateProvider subscribes to state-sync messages from the host
65
+ * and provides the state to child components via context.
66
+ *
67
+ * In the new architecture:
68
+ * - Host only renders authored game UI after state-sync begins
69
+ * - Plugin receives complete state in first state-sync message
70
+ * - No buffering or waiting needed - state is immediately available
71
+ *
72
+ * This replaces the complex usePluginRuntime hook which had:
73
+ * - waitForGameStart() promise handling
74
+ * - finishSetup() coordination with queueMicrotask
75
+ * - Error timeout handling
76
+ *
77
+ * @example
78
+ * ```tsx
79
+ * function PluginRoot() {
80
+ * return (
81
+ * <RuntimeProvider runtime={runtime}>
82
+ * <PluginStateProvider>
83
+ * <App />
84
+ * </PluginStateProvider>
85
+ * </RuntimeProvider>
86
+ * );
87
+ * }
88
+ * ```
89
+ */
90
+ export function PluginStateProvider({
91
+ children,
92
+ loadingComponent = <DefaultLoadingScreen />,
93
+ }: PluginStateProviderProps) {
94
+ const runtime = useRuntimeContext() as PluginRuntimeAPI;
95
+ const [state, setState] = useState<PluginStateSnapshot | null>(
96
+ () => runtime.getSnapshot?.() ?? null,
97
+ );
98
+
99
+ useEffect(() => {
100
+ // Get initial state if available
101
+ const initialState = runtime.getSnapshot?.();
102
+ if (initialState) {
103
+ setState(initialState);
104
+ }
105
+
106
+ // Subscribe to state changes
107
+ if (runtime.subscribeToState) {
108
+ return runtime.subscribeToState((newState) => {
109
+ setState(newState);
110
+ });
111
+ }
112
+
113
+ return () => {};
114
+ }, [runtime]);
115
+
116
+ // Don't render children until state is available
117
+ // In the new architecture, host guarantees state exists before rendering plugin
118
+ if (!state) {
119
+ return <>{loadingComponent}</>;
120
+ }
121
+
122
+ return (
123
+ <PluginStateContext.Provider value={state}>
124
+ {children}
125
+ </PluginStateContext.Provider>
126
+ );
127
+ }
128
+
129
+ /**
130
+ * Hook to access the full plugin state snapshot.
131
+ *
132
+ * @throws Error if used outside of PluginStateProvider
133
+ * @returns Current plugin state snapshot
134
+ *
135
+ * @example
136
+ * ```typescript
137
+ * function MyComponent() {
138
+ * const state = usePluginStateSnapshot();
139
+ * console.log('Current phase:', state.gameplay.currentPhase);
140
+ * }
141
+ * ```
142
+ */
143
+ export function usePluginStateSnapshot(): PluginStateSnapshot {
144
+ const state = useContext(PluginStateContext);
145
+
146
+ if (!state) {
147
+ throw new Error(
148
+ "usePluginStateSnapshot must be used within PluginStateProvider. " +
149
+ "Make sure you have wrapped your app with <PluginStateProvider>.",
150
+ );
151
+ }
152
+
153
+ return state;
154
+ }
155
+
156
+ /**
157
+ * Hook to select a specific part of the plugin state.
158
+ * Uses useSyncExternalStore for optimal performance - only re-renders
159
+ * when the selected value changes (using reference equality).
160
+ *
161
+ * @param selector - Function to select a part of the state
162
+ * @returns Selected value from state
163
+ *
164
+ * @example
165
+ * ```typescript
166
+ * // Only re-renders when gameplay.currentPhase changes
167
+ * function CurrentStateDisplay() {
168
+ * const currentState = usePluginState((s) => s.gameplay.currentPhase);
169
+ * return <div>State: {currentState}</div>;
170
+ * }
171
+ * ```
172
+ */
173
+ export function usePluginState<T>(
174
+ selector: (state: PluginStateSnapshot) => T,
175
+ ): T {
176
+ const runtime = useRuntimeContext() as PluginRuntimeAPI;
177
+
178
+ const subscribe = (onStoreChange: () => void) => {
179
+ if (!runtime.subscribeToState) {
180
+ return () => {};
181
+ }
182
+ return runtime.subscribeToState(() => {
183
+ onStoreChange();
184
+ });
185
+ };
186
+
187
+ const getSnapshot = () => {
188
+ const state = runtime.getSnapshot?.();
189
+ if (!state) {
190
+ throw new Error(
191
+ "usePluginState: No state available. " +
192
+ "Make sure you have wrapped your app with <PluginStateProvider>.",
193
+ );
194
+ }
195
+ return state;
196
+ };
197
+
198
+ const state = useSyncExternalStore(subscribe, getSnapshot, getSnapshot);
199
+ return useMemo(() => selector(state), [selector, state]);
200
+ }
201
+
202
+ /**
203
+ * Hook to access plugin actions (like marking notifications read).
204
+ *
205
+ * @returns Object with action functions
206
+ *
207
+ * @example
208
+ * ```typescript
209
+ * function NotificationsList() {
210
+ * const notifications = usePluginState((s) => s.notifications);
211
+ * const { markNotificationRead, switchPlayer } = usePluginActions();
212
+ *
213
+ * return (
214
+ * <ul>
215
+ * {notifications.map((n) => (
216
+ * <li key={n.id} onClick={() => markNotificationRead(n.id)}>
217
+ * {n.type}
218
+ * </li>
219
+ * ))}
220
+ * </ul>
221
+ * );
222
+ * }
223
+ * ```
224
+ */
225
+ export function usePluginActions() {
226
+ const runtime = useRuntimeContext() as PluginRuntimeAPI;
227
+
228
+ return {
229
+ /**
230
+ * Mark a notification as read.
231
+ * Sends message to host which updates the notification in GameSessionStore.
232
+ */
233
+ markNotificationRead: (notificationId: string) => {
234
+ window.parent.postMessage(
235
+ { type: "mark-notification-read", notificationId },
236
+ "*",
237
+ );
238
+ },
239
+
240
+ /**
241
+ * Switch to a different player (for users controlling multiple seats).
242
+ */
243
+ switchPlayer: (playerId: string) => {
244
+ runtime.switchPlayer?.(playerId);
245
+ },
246
+
247
+ /**
248
+ * Submit a player interaction (action or prompt response).
249
+ */
250
+ submitInteraction: runtime.submitInteraction,
251
+ };
252
+ }
253
+
254
+ export { PluginStateContext };
@@ -0,0 +1,96 @@
1
+ import React, { createContext, useContext, useState, useEffect } from "react";
2
+ import type { RuntimeAPI, PluginSessionState } from "../types/runtime-api.js";
3
+ import { PluginSessionContext } from "./PluginSessionContext.js";
4
+
5
+ /**
6
+ * React Context for providing RuntimeAPI to plugin components.
7
+ * This context must be provided by the plugin wrapper, not by the plugin code itself.
8
+ */
9
+ export const RuntimeContext = createContext<RuntimeAPI | null>(null);
10
+
11
+ /**
12
+ * Hook to access the RuntimeAPI from context.
13
+ *
14
+ * @throws Error if used outside of RuntimeContext.Provider
15
+ * @returns RuntimeAPI instance
16
+ *
17
+ * @example
18
+ * ```typescript
19
+ * function MyPluginComponent() {
20
+ * const runtime = useRuntimeContext();
21
+ * return (
22
+ * <button onClick={() => runtime.submitInteraction("player-1", "pass", {})}>
23
+ * Pass
24
+ * </button>
25
+ * );
26
+ * }
27
+ * ```
28
+ */
29
+ export function useRuntimeContext(): RuntimeAPI {
30
+ const context = useContext(RuntimeContext);
31
+
32
+ if (!context) {
33
+ throw new Error(
34
+ "useRuntimeContext must be used within a RuntimeContext.Provider",
35
+ );
36
+ }
37
+
38
+ return context;
39
+ }
40
+
41
+ /**
42
+ * RuntimeProvider component that provides both RuntimeAPI and PluginSessionContext.
43
+ * This component subscribes to session state changes from the RuntimeAPI and provides
44
+ * them through PluginSessionContext.
45
+ *
46
+ * @example
47
+ * ```tsx
48
+ * function PluginRoot() {
49
+ * const runtime = createPluginRuntimeAPI();
50
+ *
51
+ * return (
52
+ * <RuntimeProvider runtime={runtime}>
53
+ * <App />
54
+ * </RuntimeProvider>
55
+ * );
56
+ * }
57
+ * ```
58
+ */
59
+ export function RuntimeProvider({
60
+ runtime,
61
+ children,
62
+ }: {
63
+ runtime: RuntimeAPI;
64
+ children: React.ReactNode;
65
+ }) {
66
+ // Subscribe to session state changes
67
+ const [sessionState, setSessionState] = useState<PluginSessionState>(() =>
68
+ runtime.getSessionState(),
69
+ );
70
+
71
+ useEffect(() => {
72
+ // Subscribe to session state changes via internal API
73
+ const runtimeWithInternal = runtime as RuntimeAPI & {
74
+ _subscribeToSessionState?: (
75
+ listener: (state: PluginSessionState) => void,
76
+ ) => () => void;
77
+ };
78
+
79
+ if (runtimeWithInternal._subscribeToSessionState) {
80
+ return runtimeWithInternal._subscribeToSessionState((state) => {
81
+ setSessionState(state);
82
+ });
83
+ }
84
+
85
+ // Fallback: no session state subscription available
86
+ return () => {};
87
+ }, [runtime]);
88
+
89
+ return (
90
+ <RuntimeContext.Provider value={runtime}>
91
+ <PluginSessionContext.Provider value={sessionState}>
92
+ {children}
93
+ </PluginSessionContext.Provider>
94
+ </RuntimeContext.Provider>
95
+ );
96
+ }