@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,437 @@
1
+ import { useCallback, useEffect, useMemo, useRef } from "react";
2
+ import {
3
+ useArmedInteraction,
4
+ useInteractionDraft,
5
+ useInteractionSubmitting,
6
+ useInteractionUiStore,
7
+ } from "../context/InteractionDraftContext.js";
8
+ import { useClientParamSchema } from "../context/ClientParamSchemaContext.js";
9
+ import { usePluginState } from "../context/PluginStateContext.js";
10
+ import { usePluginSession } from "../context/PluginSessionContext.js";
11
+ import { useRuntimeContext } from "../context/RuntimeContext.js";
12
+ import {
13
+ ValidationError,
14
+ validationErrorFromUnknown,
15
+ } from "../errors/ValidationError.js";
16
+ import type { InteractionDescriptor } from "../types/plugin-state.js";
17
+ import {
18
+ applyInteractionInputDefaults,
19
+ hasInteractionFieldErrors,
20
+ inputByKey,
21
+ isInputValueReady,
22
+ interactionArmScope,
23
+ interactionInputKeys,
24
+ mergeInteractionFieldErrors,
25
+ validateInteractionInputDomains,
26
+ } from "../utils/interaction-inputs.js";
27
+
28
+ /**
29
+ * Anything that can be used as a submit params object. Subset of TS
30
+ * objects so generic defaults and `keyof Params & string` hold.
31
+ */
32
+ export type InteractionParamsShape = Record<string, unknown>;
33
+
34
+ export type InteractionHandleStatus = "open" | "submitting" | "submitted";
35
+
36
+ export type DraftValidation<
37
+ Params extends InteractionParamsShape = InteractionParamsShape,
38
+ > =
39
+ | {
40
+ ok: true;
41
+ params: Params;
42
+ fieldErrors: Partial<Record<keyof Params & string, readonly string[]>>;
43
+ formErrors: readonly string[];
44
+ missing: ReadonlyArray<keyof Params & string>;
45
+ }
46
+ | {
47
+ ok: false;
48
+ fieldErrors: Partial<Record<keyof Params & string, readonly string[]>>;
49
+ formErrors: readonly string[];
50
+ missing: ReadonlyArray<keyof Params & string>;
51
+ };
52
+
53
+ /**
54
+ * Bound handle around an {@link InteractionDescriptor}. Surfaces call into
55
+ * this hook to submit/validate an interaction, track draft input state for
56
+ * multi-step prompts, and arm/disarm themselves on a surface.
57
+ *
58
+ * Availability flags are mirrored from the authoritative descriptor; UI
59
+ * MUST NOT recompute eligibility locally.
60
+ *
61
+ * When the surrounding workspace knows the concrete params shape (e.g.
62
+ * obtained from the generated `InteractionParamsOf<Id>` alias), parameterise
63
+ * this handle on `Params` so `submit`, `draft`, and `setInput` are all
64
+ * statically typed. The default `Record<string, unknown>` preserves the
65
+ * loosely-typed fallback for generic infrastructure.
66
+ */
67
+ export interface InteractionHandle<
68
+ Params extends InteractionParamsShape = InteractionParamsShape,
69
+ DefaultedKeys extends keyof Params & string = never,
70
+ > {
71
+ descriptor: InteractionDescriptor;
72
+ /** Draft commit policy projected by the reducer. */
73
+ commit: InteractionDescriptor["commit"];
74
+ /**
75
+ * Submit the interaction. When `params` is omitted the current draft (as
76
+ * mutated by {@link InteractionHandle.setInput}) is used instead, which is
77
+ * the common case for multi-input prompts.
78
+ */
79
+ submit: (params?: Params) => Promise<void>;
80
+ /** Run server validation using `params` (or the current draft by default). */
81
+ validate: (params?: Params) => Promise<void>;
82
+ /** Run local generated client-schema validation against the current draft. */
83
+ validateDraft: () => DraftValidation<Params>;
84
+ /** Run server validation against the current draft. */
85
+ validateDraftServer: () => Promise<void>;
86
+ /**
87
+ * Validate the current draft locally, submit parsed params, then clear the
88
+ * draft only when submission succeeds.
89
+ */
90
+ submitDraft: () => Promise<void>;
91
+ /** Mirrors `descriptor.available`. */
92
+ available: boolean;
93
+ /** Mirrors `descriptor.unavailableReason`. */
94
+ unavailableReason?: string;
95
+ /** Local/authoritative progress for this interaction. */
96
+ status: InteractionHandleStatus;
97
+
98
+ // --- Draft state --------------------------------------------------------
99
+
100
+ /** Live draft values for this interaction. Empty object when unset. */
101
+ draft: Readonly<Partial<Params>>;
102
+ /**
103
+ * Draft values with authored input defaults applied. Fields with declared
104
+ * defaults are typed as present; other draft fields remain partial.
105
+ */
106
+ values: Readonly<Partial<Params> & Pick<Params, DefaultedKeys>>;
107
+ /** Set a single input on the draft. */
108
+ setInput: <K extends keyof Params & string>(key: K, value: Params[K]) => void;
109
+ /** Clear a single input (or the whole draft when `key` is omitted). */
110
+ clearInput: (key?: keyof Params & string) => void;
111
+ /**
112
+ * True when every key declared on `descriptor.inputs` has a value in
113
+ * the draft. Falls back to `true` when the descriptor declares no inputs.
114
+ */
115
+ isReady: boolean;
116
+ /**
117
+ * True when this interaction is the currently armed one on its surface.
118
+ * Armed interactions are the ones that surfaces like `BoardSurface` use
119
+ * to highlight eligible targets and route clicks.
120
+ */
121
+ isArmed: boolean;
122
+ /** Arm this interaction on its surface (disarms any previously armed). */
123
+ arm: () => void;
124
+ /** Disarm this interaction (if it was armed). */
125
+ disarm: () => void;
126
+ }
127
+
128
+ /**
129
+ * Bind an {@link InteractionDescriptor} to submit/validate helpers plus
130
+ * draft + arming state. Use this from any surface; draft state is shared
131
+ * across components through {@link InteractionUiProvider}, which
132
+ * `PluginRuntime` auto-mounts.
133
+ *
134
+ * Example:
135
+ * ```tsx
136
+ * const handle = useInteractionHandle(placeThing);
137
+ * handle.setInput("cardId", card.id);
138
+ * handle.setInput("ringId", ring.id);
139
+ * if (handle.isReady) await handle.submit();
140
+ * ```
141
+ */
142
+ export function useInteractionHandle<
143
+ Params extends InteractionParamsShape = InteractionParamsShape,
144
+ DefaultedKeys extends keyof Params & string = never,
145
+ >(descriptor: InteractionDescriptor): InteractionHandle<Params, DefaultedKeys> {
146
+ const runtime = useRuntimeContext();
147
+ const { controllingPlayerId } = usePluginSession();
148
+ const store = useInteractionUiStore();
149
+ const submittingRef = useRef(false);
150
+ const autoSubmitSignatureRef = useRef<string | null>(null);
151
+ const simultaneousPhase = usePluginState(
152
+ (state) => state.gameplay.simultaneousPhase,
153
+ );
154
+
155
+ const { interactionId, interactionKey, phaseName } = descriptor;
156
+ const armScope = interactionArmScope(descriptor);
157
+ const inputKeys = useMemo(
158
+ () => interactionInputKeys(descriptor),
159
+ [descriptor],
160
+ );
161
+ const paramsSchema = useClientParamSchema(phaseName, interactionId);
162
+ const draft = useInteractionDraft(interactionKey) as Readonly<
163
+ Partial<Params>
164
+ >;
165
+ const values = useMemo(
166
+ () =>
167
+ applyInteractionInputDefaults<Params>(descriptor, draft) as Readonly<
168
+ Partial<Params> & Pick<Params, DefaultedKeys>
169
+ >,
170
+ [descriptor, draft],
171
+ );
172
+ const armedId = useArmedInteraction(armScope);
173
+ const isArmed = armedId === interactionKey;
174
+ const submitting = useInteractionSubmitting(interactionKey);
175
+ const submitted =
176
+ controllingPlayerId !== null &&
177
+ simultaneousPhase?.phaseName === phaseName &&
178
+ simultaneousPhase.interactionId === interactionId &&
179
+ simultaneousPhase.sealedPlayerIds.includes(controllingPlayerId);
180
+ const status: InteractionHandleStatus = submitted
181
+ ? "submitted"
182
+ : submitting
183
+ ? "submitting"
184
+ : "open";
185
+
186
+ const isReady = useMemo(() => {
187
+ const required = inputKeys;
188
+ if (required.length === 0) return true;
189
+ return required.every((key) => {
190
+ const input = inputByKey(descriptor, key);
191
+ const value = (values as Record<string, unknown>)[key];
192
+ return input
193
+ ? isInputValueReady(input, value)
194
+ : value !== null && value !== undefined;
195
+ });
196
+ }, [descriptor, values, inputKeys]);
197
+
198
+ const requirePlayer = useCallback(() => {
199
+ if (!controllingPlayerId) {
200
+ throw new Error("useInteractionHandle: no controlling player available");
201
+ }
202
+ return controllingPlayerId;
203
+ }, [controllingPlayerId]);
204
+
205
+ const submit = useCallback(
206
+ async (params?: Params) => {
207
+ if (status !== "open" || submittingRef.current) {
208
+ throw new ValidationError(
209
+ status === "submitted" ? "ALREADY_SUBMITTED" : "SUBMITTING",
210
+ status === "submitted"
211
+ ? "Interaction has already been submitted."
212
+ : "Interaction submission is already in progress.",
213
+ );
214
+ }
215
+ submittingRef.current = true;
216
+ store.setSubmitting(interactionKey, true);
217
+ const finalParams = applyInteractionInputDefaults<Params>(
218
+ descriptor,
219
+ params ?? values,
220
+ ) as Params;
221
+ try {
222
+ await runtime.submitInteraction(
223
+ requirePlayer(),
224
+ interactionId,
225
+ finalParams as Record<string, unknown>,
226
+ );
227
+ store.clearInput(interactionKey);
228
+ if (store.getArmed(armScope) === interactionKey) {
229
+ store.arm(armScope, null);
230
+ }
231
+ } catch (error) {
232
+ throw validationErrorFromUnknown(error);
233
+ } finally {
234
+ submittingRef.current = false;
235
+ store.setSubmitting(interactionKey, false);
236
+ }
237
+ },
238
+ [
239
+ descriptor,
240
+ values,
241
+ runtime,
242
+ requirePlayer,
243
+ interactionId,
244
+ interactionKey,
245
+ armScope,
246
+ store,
247
+ status,
248
+ ],
249
+ );
250
+
251
+ const validate = useCallback(
252
+ async (params?: Params) => {
253
+ const finalParams = applyInteractionInputDefaults<Params>(
254
+ descriptor,
255
+ params ?? values,
256
+ ) as Params;
257
+ const result = await runtime.validateInteraction(
258
+ requirePlayer(),
259
+ interactionId,
260
+ finalParams as Record<string, unknown>,
261
+ );
262
+ if (!result.valid) {
263
+ throw new ValidationError(result.errorCode, result.message);
264
+ }
265
+ },
266
+ [descriptor, values, runtime, requirePlayer, interactionId],
267
+ );
268
+
269
+ const validateDraft = useCallback((): DraftValidation<Params> => {
270
+ const rawDraft = { ...values } as Record<string, unknown>;
271
+ const required = inputKeys;
272
+ const missing = required.filter((key) => {
273
+ const input = inputByKey(descriptor, key);
274
+ const value = rawDraft[key];
275
+ return input
276
+ ? !isInputValueReady(input, value)
277
+ : value === null || value === undefined;
278
+ }) as Array<keyof Params & string>;
279
+ const domainFieldErrors = validateInteractionInputDomains(
280
+ descriptor,
281
+ rawDraft,
282
+ ) as Partial<Record<keyof Params & string, readonly string[]>>;
283
+
284
+ if (!paramsSchema) {
285
+ if (missing.length > 0 || hasInteractionFieldErrors(domainFieldErrors)) {
286
+ return {
287
+ ok: false,
288
+ fieldErrors: domainFieldErrors,
289
+ formErrors: [],
290
+ missing,
291
+ };
292
+ }
293
+ return {
294
+ ok: true,
295
+ params: rawDraft as Params,
296
+ fieldErrors: {},
297
+ formErrors: [],
298
+ missing: [],
299
+ };
300
+ }
301
+
302
+ const result = paramsSchema.safeParse(rawDraft);
303
+ if (result.success) {
304
+ if (hasInteractionFieldErrors(domainFieldErrors)) {
305
+ return {
306
+ ok: false,
307
+ fieldErrors: domainFieldErrors,
308
+ formErrors: [],
309
+ missing,
310
+ };
311
+ }
312
+ return {
313
+ ok: true,
314
+ params: result.data as Params,
315
+ fieldErrors: {},
316
+ formErrors: [],
317
+ missing: [],
318
+ };
319
+ }
320
+
321
+ const fieldErrors: Partial<
322
+ Record<keyof Params & string, readonly string[]>
323
+ > = {};
324
+ const formErrors: string[] = [];
325
+ const missingSet = new Set<PropertyKey>(missing);
326
+ for (const issue of result.error.issues) {
327
+ const [first] = issue.path;
328
+ if (typeof first === "string" && required.includes(first)) {
329
+ if (missingSet.has(first)) continue;
330
+ const key = first as keyof Params & string;
331
+ fieldErrors[key] = [...(fieldErrors[key] ?? []), issue.message];
332
+ } else {
333
+ formErrors.push(issue.message);
334
+ }
335
+ }
336
+ return {
337
+ ok: false,
338
+ fieldErrors: mergeInteractionFieldErrors(
339
+ fieldErrors,
340
+ domainFieldErrors,
341
+ ) as Partial<Record<keyof Params & string, readonly string[]>>,
342
+ formErrors,
343
+ missing,
344
+ };
345
+ }, [descriptor, values, inputKeys, paramsSchema]);
346
+
347
+ const validateDraftServer = useCallback(async () => {
348
+ await validate({ ...values } as Params);
349
+ }, [values, validate]);
350
+
351
+ const submitDraft = useCallback(async () => {
352
+ const validation = validateDraft();
353
+ if (!validation.ok) {
354
+ const message =
355
+ validation.formErrors[0] ??
356
+ Object.values(validation.fieldErrors).flat()[0] ??
357
+ (validation.missing.length > 0
358
+ ? "Required inputs are missing"
359
+ : "Draft validation failed");
360
+ throw new ValidationError("INVALID_DRAFT", message);
361
+ }
362
+ await submit(validation.params);
363
+ }, [submit, validateDraft]);
364
+
365
+ useEffect(() => {
366
+ if (descriptor.commit.mode !== "autoWhenReady") {
367
+ autoSubmitSignatureRef.current = null;
368
+ return;
369
+ }
370
+ if (!descriptor.available || status !== "open" || !isReady) {
371
+ if (!isReady) autoSubmitSignatureRef.current = null;
372
+ return;
373
+ }
374
+ const validation = validateDraft();
375
+ if (!validation.ok) return;
376
+ const signature = `${interactionKey}:${JSON.stringify(validation.params)}`;
377
+ if (autoSubmitSignatureRef.current === signature) return;
378
+ autoSubmitSignatureRef.current = signature;
379
+ void submit(validation.params).catch(() => {
380
+ // Runtime error channels surface the failure. Keep the draft intact and
381
+ // suppress repeated attempts until the player changes the draft.
382
+ });
383
+ }, [
384
+ descriptor.available,
385
+ descriptor.commit.mode,
386
+ interactionKey,
387
+ isReady,
388
+ status,
389
+ submit,
390
+ validateDraft,
391
+ ]);
392
+
393
+ const setInput = useCallback(
394
+ <K extends keyof Params & string>(key: K, value: Params[K]) => {
395
+ store.setInput(interactionKey, key, value as unknown);
396
+ },
397
+ [store, interactionKey],
398
+ );
399
+
400
+ const clearInput = useCallback(
401
+ (key?: keyof Params & string) => {
402
+ store.clearInput(interactionKey, key);
403
+ },
404
+ [store, interactionKey],
405
+ );
406
+
407
+ const arm = useCallback(() => {
408
+ store.arm(armScope, interactionKey);
409
+ }, [store, armScope, interactionKey]);
410
+
411
+ const disarm = useCallback(() => {
412
+ if (store.getArmed(armScope) === interactionKey) {
413
+ store.arm(armScope, null);
414
+ }
415
+ }, [store, armScope, interactionKey]);
416
+
417
+ return {
418
+ descriptor,
419
+ commit: descriptor.commit,
420
+ available: descriptor.available,
421
+ unavailableReason: descriptor.unavailableReason,
422
+ status,
423
+ submit,
424
+ validate,
425
+ validateDraft,
426
+ validateDraftServer,
427
+ submitDraft,
428
+ draft,
429
+ values,
430
+ setInput,
431
+ clearInput,
432
+ isReady,
433
+ isArmed,
434
+ arm,
435
+ disarm,
436
+ };
437
+ }
@@ -0,0 +1,35 @@
1
+ import { useState, useEffect } from "react";
2
+
3
+ const MOBILE_BREAKPOINT = 768;
4
+
5
+ /**
6
+ * Hook to detect if the user is on a mobile/small screen device
7
+ * @param breakpoint - The width threshold in pixels (default: 768)
8
+ * @returns boolean indicating if the screen is mobile-sized
9
+ */
10
+ export function useIsMobile(breakpoint: number = MOBILE_BREAKPOINT): boolean {
11
+ const [isMobile, setIsMobile] = useState(() => {
12
+ // SSR-safe: default to false if window is not available
13
+ if (typeof window === "undefined") return false;
14
+ return window.innerWidth < breakpoint;
15
+ });
16
+
17
+ useEffect(() => {
18
+ if (typeof window === "undefined") return;
19
+
20
+ const mediaQuery = window.matchMedia(`(max-width: ${breakpoint - 1}px)`);
21
+
22
+ const handleChange = (e: MediaQueryListEvent | MediaQueryList) => {
23
+ setIsMobile(e.matches);
24
+ };
25
+
26
+ // Set initial value
27
+ handleChange(mediaQuery);
28
+
29
+ // Listen for changes
30
+ mediaQuery.addEventListener("change", handleChange);
31
+ return () => mediaQuery.removeEventListener("change", handleChange);
32
+ }, [breakpoint]);
33
+
34
+ return isMobile;
35
+ }
@@ -0,0 +1,99 @@
1
+ import { expect, test } from "bun:test";
2
+ import { renderToString } from "react-dom/server";
3
+ import { RuntimeContext } from "../context/RuntimeContext.js";
4
+ import { PluginSessionContext } from "../context/PluginSessionContext.js";
5
+ import type { PluginStateSnapshot } from "../types/plugin-state.js";
6
+ import type { PluginSessionState, RuntimeAPI } from "../types/runtime-api.js";
7
+ import { useIsMyTurn } from "./useIsMyTurn.js";
8
+
9
+ function createSessionState(
10
+ overrides: Partial<PluginSessionState> = {},
11
+ ): PluginSessionState {
12
+ return {
13
+ status: "ready",
14
+ sessionId: "session-1",
15
+ controllablePlayerIds: ["player-1"],
16
+ controllingPlayerId: "player-1",
17
+ userId: "user-1",
18
+ ...overrides,
19
+ };
20
+ }
21
+
22
+ function createSnapshot(
23
+ session: PluginSessionState,
24
+ activePlayers: string[],
25
+ ): PluginStateSnapshot {
26
+ return {
27
+ view: null,
28
+ gameplay: {
29
+ currentPhase: "takeTurn",
30
+ currentStage: null,
31
+ activePlayers,
32
+ availableInteractions: [],
33
+ zones: {},
34
+ },
35
+ lobby: null,
36
+ notifications: [],
37
+ session,
38
+ history: null,
39
+ syncId: 1,
40
+ };
41
+ }
42
+
43
+ function createRuntime(snapshot: PluginStateSnapshot): RuntimeAPI & {
44
+ getSnapshot: () => PluginStateSnapshot;
45
+ subscribeToState: (
46
+ listener: (state: PluginStateSnapshot) => void,
47
+ ) => () => void;
48
+ } {
49
+ return {
50
+ validateInteraction: async () => ({ valid: true }),
51
+ submitInteraction: async () => undefined,
52
+ getSessionState: () => snapshot.session,
53
+ disconnect: () => undefined,
54
+ getSnapshot: () => snapshot,
55
+ subscribeToState: () => () => undefined,
56
+ };
57
+ }
58
+
59
+ test("useIsMyTurn returns true when the controlled player is active", () => {
60
+ const session = createSessionState();
61
+ const runtime = createRuntime(createSnapshot(session, ["player-1"]));
62
+ let captured = false;
63
+
64
+ function Harness() {
65
+ captured = useIsMyTurn();
66
+ return null;
67
+ }
68
+
69
+ renderToString(
70
+ <RuntimeContext.Provider value={runtime}>
71
+ <PluginSessionContext.Provider value={session}>
72
+ <Harness />
73
+ </PluginSessionContext.Provider>
74
+ </RuntimeContext.Provider>,
75
+ );
76
+
77
+ expect(captured).toBe(true);
78
+ });
79
+
80
+ test("useIsMyTurn returns false when the controlled player is not active", () => {
81
+ const session = createSessionState();
82
+ const runtime = createRuntime(createSnapshot(session, ["player-2"]));
83
+ let captured = true;
84
+
85
+ function Harness() {
86
+ captured = useIsMyTurn();
87
+ return null;
88
+ }
89
+
90
+ renderToString(
91
+ <RuntimeContext.Provider value={runtime}>
92
+ <PluginSessionContext.Provider value={session}>
93
+ <Harness />
94
+ </PluginSessionContext.Provider>
95
+ </RuntimeContext.Provider>,
96
+ );
97
+
98
+ expect(captured).toBe(false);
99
+ });
@@ -0,0 +1,15 @@
1
+ import { usePluginState } from "../context/PluginStateContext.js";
2
+ import { usePluginSession } from "../context/PluginSessionContext.js";
3
+
4
+ /**
5
+ * Returns whether the currently controlled player is one of the engine-tracked
6
+ * active players for the current gameplay snapshot.
7
+ */
8
+ export function useIsMyTurn(): boolean {
9
+ const { controllingPlayerId } = usePluginSession();
10
+ const activePlayers = usePluginState((state) => state.gameplay.activePlayers);
11
+
12
+ return (
13
+ controllingPlayerId !== null && activePlayers.includes(controllingPlayerId)
14
+ );
15
+ }
@@ -0,0 +1,76 @@
1
+ import { useState, useEffect } from "react";
2
+ import { useRuntimeContext } from "../context/RuntimeContext.js";
3
+ import type { LobbyState } from "../types/plugin-state.js";
4
+ import type { PluginRuntimeAPI } from "../runtime/createPluginRuntimeAPI.js";
5
+
6
+ // Re-export LobbyState for convenience
7
+ export type { LobbyState };
8
+
9
+ /**
10
+ * Subscribes to lobby updates from the plugin runtime snapshot when present.
11
+ * Returns `null` until the host provides lobby state (SSR/tests/minimal runtimes).
12
+ */
13
+ export function useLobbyState(): LobbyState | null {
14
+ const runtime = useRuntimeContext() as PluginRuntimeAPI;
15
+
16
+ const getStateFromSnapshot = (): LobbyState | null => {
17
+ if (!runtime.getSnapshot) return null;
18
+ const snapshot = runtime.getSnapshot();
19
+ if (!snapshot?.lobby) return null;
20
+ return snapshot.lobby;
21
+ };
22
+
23
+ const [lobbyState, setLobbyState] = useState<LobbyState | null>(
24
+ getStateFromSnapshot,
25
+ );
26
+
27
+ useEffect(() => {
28
+ if (!runtime.subscribeToState) {
29
+ return;
30
+ }
31
+
32
+ const initialState = runtime.getSnapshot?.();
33
+ if (initialState?.lobby) {
34
+ setLobbyState(initialState.lobby);
35
+ }
36
+
37
+ return runtime.subscribeToState((snapshot) => {
38
+ if (snapshot.lobby) {
39
+ setLobbyState(snapshot.lobby);
40
+ }
41
+ });
42
+ }, [runtime]);
43
+
44
+ return lobbyState;
45
+ }
46
+
47
+ /**
48
+ * Hook to subscribe to lobby state updates.
49
+ * Returns the latest lobby information from state-sync messages.
50
+ *
51
+ * State is provided by PluginStateProvider from host's state-sync messages.
52
+ * The host transforms raw SSE LOBBY_UPDATE messages into clean LobbyState objects.
53
+ *
54
+ * @returns Current lobby state (never null - throws if not available)
55
+ * @throws Error if lobby state is not available
56
+ *
57
+ * @example
58
+ * ```typescript
59
+ * function LobbyScreen() {
60
+ * const lobby = useLobby();
61
+ * // lobby is guaranteed to be non-null
62
+ * return <div>{lobby.seats.length} seats</div>;
63
+ * }
64
+ * ```
65
+ */
66
+ export function useLobby(): LobbyState {
67
+ const lobbyState = useLobbyState();
68
+ if (lobbyState === null) {
69
+ throw new Error(
70
+ "useLobby: Lobby state not available. " +
71
+ "The host should only render the plugin when lobby state is ready.",
72
+ );
73
+ }
74
+
75
+ return lobbyState;
76
+ }