@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,1294 @@
1
+ /**
2
+ * SVG-based hex grid for hex-based games (Catan, wargames, Hive, Twilight Imperium).
3
+ * Supports tiles, edges (roads), vertices (settlements), and interactive placement overlays.
4
+ * Pan/zoom enabled on mobile via @use-gesture.
5
+ */
6
+
7
+ import { useMemo, useState, type ReactNode } from "react";
8
+ import { clsx } from "clsx";
9
+ import { usePanZoom, calculateViewBox } from "../../hooks/usePanZoom.js";
10
+ import { useIsMobile } from "../../hooks/useIsMobile.js";
11
+ import { handleKeyboardActivation } from "./interaction-accessibility.js";
12
+ import {
13
+ interactiveTargetRenderState,
14
+ isInteractiveTargetSelectable,
15
+ type InteractiveTargetLayer,
16
+ type InteractiveTargetRenderState,
17
+ } from "./target-layer.js";
18
+ import {
19
+ type AuthoredHexBoardInput,
20
+ type AnyHexBoardInput,
21
+ type BoardSpaceIdOf,
22
+ type GeneratedHexBoardInput,
23
+ type NormalizedHexBoard,
24
+ type NormalizedHexEdgeOf,
25
+ type NormalizedHexTileOf,
26
+ type NormalizedHexVertexOf,
27
+ normalizeHexBoardInput,
28
+ } from "../../types/tiled-board.js";
29
+
30
+ export type {
31
+ InteractiveTargetLayer,
32
+ InteractiveTargetRenderState,
33
+ } from "./target-layer.js";
34
+
35
+ // ============================================================================
36
+ // Types
37
+ // ============================================================================
38
+
39
+ export type HexOrientation = "pointy-top" | "flat-top";
40
+
41
+ /**
42
+ * Geometry context passed to `renderTile`.
43
+ *
44
+ * `corners`, `points`, and `bounds` are expressed in tile-local
45
+ * coordinates because each tile is rendered inside a `<g>` translated
46
+ * to its center. Use `position` if you need the resolved center in the
47
+ * board's absolute SVG coordinates.
48
+ *
49
+ * The `inset` option shrinks the polygon toward the center by that many
50
+ * pixels, which is useful for layered effects such as borders, frames,
51
+ * or inner highlights without re-deriving the hex math yourself.
52
+ */
53
+ export interface HexTileGeometry {
54
+ size: number;
55
+ orientation: HexOrientation;
56
+ center: { x: 0; y: 0 };
57
+ position: { x: number; y: number };
58
+ corners: (options?: { inset?: number }) => Array<{ x: number; y: number }>;
59
+ points: (options?: { inset?: number }) => string;
60
+ bounds: {
61
+ minX: number;
62
+ minY: number;
63
+ maxX: number;
64
+ maxY: number;
65
+ width: number;
66
+ height: number;
67
+ };
68
+ }
69
+
70
+ export interface EdgePosition {
71
+ /** Absolute SVG start point of the visible edge line. */
72
+ x1: number;
73
+ y1: number;
74
+ /** Absolute SVG end point of the visible edge line. */
75
+ x2: number;
76
+ y2: number;
77
+ /** Absolute SVG midpoint of the edge. */
78
+ midX: number;
79
+ midY: number;
80
+ /**
81
+ * Angle in degrees from hex1 center to hex2 center.
82
+ * This is perpendicular to the visible edge line.
83
+ */
84
+ centerAngle: number;
85
+ /** Angle in degrees of the visible edge line itself. */
86
+ edgeAngle: number;
87
+ }
88
+
89
+ export interface InteractiveHexVertex<
90
+ TBoard extends AnyHexBoardInput = AnyHexBoardInput,
91
+ > extends NormalizedHexVertexOf<TBoard> {
92
+ position: { x: number; y: number };
93
+ spaceIds: ReadonlyArray<BoardSpaceIdOf<TBoard>>;
94
+ }
95
+
96
+ export interface InteractiveHexEdge<
97
+ TBoard extends AnyHexBoardInput = AnyHexBoardInput,
98
+ > extends NormalizedHexEdgeOf<TBoard> {
99
+ position: EdgePosition;
100
+ spaceIds: ReadonlyArray<BoardSpaceIdOf<TBoard>>;
101
+ }
102
+
103
+ export type InteractiveHexSpace<
104
+ TBoard extends AnyHexBoardInput = AnyHexBoardInput,
105
+ > = NormalizedHexTileOf<TBoard>;
106
+
107
+ interface HexGeneratedGridInputProps {
108
+ id?: string;
109
+ layout?: "hex";
110
+ orientation?: HexOrientation;
111
+ spaces: Extract<AnyHexBoardInput, { spaces: unknown }>["spaces"];
112
+ edges?: AnyHexBoardInput["edges"];
113
+ vertices?: AnyHexBoardInput["vertices"];
114
+ }
115
+
116
+ interface HexAuthoredGridInputProps {
117
+ id?: string;
118
+ layout?: "hex";
119
+ orientation?: HexOrientation;
120
+ tiles: Extract<AnyHexBoardInput, { tiles: unknown }>["tiles"];
121
+ edges?: AnyHexBoardInput["edges"];
122
+ vertices?: AnyHexBoardInput["vertices"];
123
+ }
124
+
125
+ type HexGridInputProps = HexGeneratedGridInputProps | HexAuthoredGridInputProps;
126
+
127
+ type ResolvedArrayProp<Value> =
128
+ Exclude<Value, undefined> extends readonly unknown[]
129
+ ? Exclude<Value, undefined>
130
+ : readonly [];
131
+
132
+ type HexBoardLikeOfProps<TProps extends HexGridInputProps> = TProps extends {
133
+ id?: infer Id;
134
+ layout?: infer Layout;
135
+ orientation?: infer Orientation;
136
+ spaces: infer Spaces;
137
+ edges?: infer Edges;
138
+ vertices?: infer Vertices;
139
+ }
140
+ ? {
141
+ id: Extract<Id, string> extends never ? string : Extract<Id, string>;
142
+ layout?: Extract<Layout, "hex">;
143
+ orientation?: Extract<Orientation, HexOrientation>;
144
+ spaces: Spaces;
145
+ edges: ResolvedArrayProp<Edges>;
146
+ vertices: ResolvedArrayProp<Vertices>;
147
+ } & GeneratedHexBoardInput
148
+ : TProps extends {
149
+ id?: infer Id;
150
+ layout?: infer Layout;
151
+ orientation?: infer Orientation;
152
+ tiles: infer Tiles;
153
+ edges?: infer Edges;
154
+ vertices?: infer Vertices;
155
+ }
156
+ ? {
157
+ id: Extract<Id, string> extends never ? string : Extract<Id, string>;
158
+ layout?: Extract<Layout, "hex">;
159
+ orientation?: Extract<Orientation, HexOrientation>;
160
+ tiles: Tiles;
161
+ edges: ResolvedArrayProp<Edges>;
162
+ vertices: ResolvedArrayProp<Vertices>;
163
+ } & AuthoredHexBoardInput
164
+ : never;
165
+
166
+ export type HexGridProps<TProps extends HexGridInputProps = HexGridInputProps> =
167
+ TProps & {
168
+ orientation?: HexOrientation;
169
+ /** Hex radius in pixels */
170
+ hexSize?: number;
171
+ /**
172
+ * Receives tile data centered at (0,0) plus a `HexTileGeometry`
173
+ * helper. Use `geometry.points({ inset })` to draw custom polygons
174
+ * without duplicating `hexSize` / orientation in the consumer.
175
+ */
176
+ renderTile: (
177
+ tile: NormalizedHexTileOf<NoInfer<HexBoardLikeOfProps<TProps>>>,
178
+ geometry: HexTileGeometry,
179
+ ) => ReactNode;
180
+ /**
181
+ * Receives edge geometry in absolute SVG coordinates.
182
+ * Use `position.edgeAngle` to align artwork with the visible edge.
183
+ */
184
+ renderEdge: (
185
+ edge: NormalizedHexEdgeOf<NoInfer<HexBoardLikeOfProps<TProps>>>,
186
+ position: EdgePosition,
187
+ ) => ReactNode;
188
+ renderVertex: (
189
+ vertex: NormalizedHexVertexOf<NoInfer<HexBoardLikeOfProps<TProps>>>,
190
+ position: { x: number; y: number },
191
+ ) => ReactNode;
192
+ width?: number | string;
193
+ height?: number | string;
194
+ enablePanZoom?: boolean;
195
+ initialZoom?: number;
196
+ minZoom?: number;
197
+ maxZoom?: number;
198
+ className?: string;
199
+
200
+ // Interactive board target layers.
201
+
202
+ /** Reducer-aware space target layer from `board.targetLayers.space(...)`. */
203
+ interactiveSpaces?: InteractiveTargetLayer;
204
+ /** Reducer-aware vertex target layer from `board.targetLayers.vertex(...)`. */
205
+ interactiveVertices?: InteractiveTargetLayer;
206
+ /** Reducer-aware edge target layer from `board.targetLayers.edge(...)`. */
207
+ interactiveEdges?: InteractiveTargetLayer;
208
+ /** Receives space geometry centered at (0,0). */
209
+ renderInteractiveSpace?: (
210
+ space: InteractiveHexSpace<NoInfer<HexBoardLikeOfProps<TProps>>>,
211
+ state: InteractiveTargetRenderState,
212
+ ) => ReactNode;
213
+ /**
214
+ * Receives vertex geometry in absolute SVG coordinates.
215
+ */
216
+ renderInteractiveVertex?: (
217
+ vertex: InteractiveHexVertex<NoInfer<HexBoardLikeOfProps<TProps>>>,
218
+ position: { x: number; y: number },
219
+ state: InteractiveTargetRenderState,
220
+ ) => ReactNode;
221
+ /**
222
+ * Receives edge geometry in the same absolute SVG coordinates as `renderEdge`.
223
+ */
224
+ renderInteractiveEdge?: (
225
+ edge: InteractiveHexEdge<NoInfer<HexBoardLikeOfProps<TProps>>>,
226
+ position: EdgePosition,
227
+ state: InteractiveTargetRenderState,
228
+ ) => ReactNode;
229
+ interactiveVertexSize?: number;
230
+ interactiveEdgeSize?: number;
231
+ };
232
+
233
+ export interface HexGridBoardProps<
234
+ TBoard extends AnyHexBoardInput = AnyHexBoardInput,
235
+ > {
236
+ board: TBoard;
237
+ orientation?: HexOrientation;
238
+ hexSize?: number;
239
+ /**
240
+ * Receives tile data centered at (0,0) plus a `HexTileGeometry`
241
+ * helper. Use `geometry.points({ inset })` to draw custom polygons
242
+ * without duplicating `hexSize` / orientation in the consumer.
243
+ */
244
+ renderTile: (
245
+ tile: NormalizedHexTileOf<NoInfer<TBoard>>,
246
+ geometry: HexTileGeometry,
247
+ ) => ReactNode;
248
+ /**
249
+ * Receives edge geometry in absolute SVG coordinates.
250
+ * Use `position.edgeAngle` to align artwork with the visible edge.
251
+ */
252
+ renderEdge: (
253
+ edge: NormalizedHexEdgeOf<NoInfer<TBoard>>,
254
+ position: EdgePosition,
255
+ ) => ReactNode;
256
+ renderVertex: (
257
+ vertex: NormalizedHexVertexOf<NoInfer<TBoard>>,
258
+ position: { x: number; y: number },
259
+ ) => ReactNode;
260
+ width?: number | string;
261
+ height?: number | string;
262
+ enablePanZoom?: boolean;
263
+ initialZoom?: number;
264
+ minZoom?: number;
265
+ maxZoom?: number;
266
+ className?: string;
267
+ interactiveSpaces?: InteractiveTargetLayer;
268
+ interactiveVertices?: InteractiveTargetLayer;
269
+ interactiveEdges?: InteractiveTargetLayer;
270
+ /** Receives space geometry centered at (0,0). */
271
+ renderInteractiveSpace?: (
272
+ space: InteractiveHexSpace<NoInfer<TBoard>>,
273
+ state: InteractiveTargetRenderState,
274
+ ) => ReactNode;
275
+ /**
276
+ * Receives vertex geometry in absolute SVG coordinates.
277
+ */
278
+ renderInteractiveVertex?: (
279
+ vertex: InteractiveHexVertex<NoInfer<TBoard>>,
280
+ position: { x: number; y: number },
281
+ state: InteractiveTargetRenderState,
282
+ ) => ReactNode;
283
+ /**
284
+ * Receives edge geometry in the same absolute SVG coordinates as `renderEdge`.
285
+ */
286
+ renderInteractiveEdge?: (
287
+ edge: InteractiveHexEdge<NoInfer<TBoard>>,
288
+ position: EdgePosition,
289
+ state: InteractiveTargetRenderState,
290
+ ) => ReactNode;
291
+ interactiveVertexSize?: number;
292
+ interactiveEdgeSize?: number;
293
+ }
294
+
295
+ // ============================================================================
296
+ // Pre-built Helper Components
297
+ // ============================================================================
298
+
299
+ export interface DefaultHexTileProps {
300
+ /** Should match hexSize from HexGrid */
301
+ size: number;
302
+ fill: string;
303
+ stroke?: string;
304
+ strokeWidth?: number;
305
+ isSelected?: boolean;
306
+ isHighlighted?: boolean;
307
+ label?: string;
308
+ showCoordinates?: boolean;
309
+ coordinates?: { q: number; r: number };
310
+ orientation?: HexOrientation;
311
+ onClick?: () => void;
312
+ onPointerEnter?: () => void;
313
+ onPointerLeave?: () => void;
314
+ className?: string;
315
+ }
316
+
317
+ /** Pre-built hexagon tile for use in `renderTile`. */
318
+ export function DefaultHexTile({
319
+ size,
320
+ fill,
321
+ stroke = "#1e293b",
322
+ strokeWidth = 1.5,
323
+ isSelected = false,
324
+ isHighlighted = false,
325
+ label,
326
+ showCoordinates = false,
327
+ coordinates,
328
+ orientation = "pointy-top",
329
+ onClick,
330
+ onPointerEnter,
331
+ onPointerLeave,
332
+ className,
333
+ }: DefaultHexTileProps) {
334
+ const effectiveFill = isSelected
335
+ ? "rgba(59, 130, 246, 0.5)"
336
+ : isHighlighted
337
+ ? "rgba(34, 197, 94, 0.4)"
338
+ : fill;
339
+
340
+ const effectiveStroke = isSelected
341
+ ? "#3b82f6"
342
+ : isHighlighted
343
+ ? "#22c55e"
344
+ : stroke;
345
+
346
+ const effectiveStrokeWidth = isSelected || isHighlighted ? 3 : strokeWidth;
347
+
348
+ const points = hexUtils.getHexPoints(0, 0, size * 0.95, orientation);
349
+
350
+ return (
351
+ <g
352
+ onClick={onClick}
353
+ onPointerEnter={onPointerEnter}
354
+ onPointerLeave={onPointerLeave}
355
+ onKeyDown={(event) => handleKeyboardActivation(event, onClick)}
356
+ className={clsx(
357
+ "transition-all duration-150",
358
+ onClick && "cursor-pointer hover:brightness-110",
359
+ className,
360
+ )}
361
+ role={onClick ? "button" : undefined}
362
+ tabIndex={onClick ? 0 : undefined}
363
+ aria-label={onClick ? (label ?? "Hex tile") : undefined}
364
+ >
365
+ <polygon
366
+ points={points}
367
+ fill={effectiveFill}
368
+ stroke={effectiveStroke}
369
+ strokeWidth={effectiveStrokeWidth}
370
+ filter="url(#hexShadow)"
371
+ />
372
+
373
+ {label && (
374
+ <text
375
+ x={0}
376
+ y={showCoordinates ? -8 : 0}
377
+ textAnchor="middle"
378
+ dominantBaseline="middle"
379
+ fill="white"
380
+ fontSize={size * 0.28}
381
+ fontWeight="bold"
382
+ style={{ textShadow: "1px 1px 2px rgba(0,0,0,0.8)" }}
383
+ pointerEvents="none"
384
+ >
385
+ {label}
386
+ </text>
387
+ )}
388
+
389
+ {showCoordinates && coordinates && (
390
+ <text
391
+ x={0}
392
+ y={label ? 10 : 0}
393
+ textAnchor="middle"
394
+ dominantBaseline="middle"
395
+ fill="rgba(255,255,255,0.7)"
396
+ fontSize={size * 0.2}
397
+ pointerEvents="none"
398
+ >
399
+ {coordinates.q},{coordinates.r}
400
+ </text>
401
+ )}
402
+ </g>
403
+ );
404
+ }
405
+
406
+ export interface DefaultHexEdgeProps {
407
+ position: EdgePosition;
408
+ color: string;
409
+ hasOwner?: boolean;
410
+ strokeWidth?: number;
411
+ touchTargetSize?: number;
412
+ onClick?: () => void;
413
+ className?: string;
414
+ }
415
+
416
+ /** Pre-built edge/road component for use in `renderEdge`. */
417
+ export function DefaultHexEdge({
418
+ position,
419
+ color,
420
+ hasOwner = true,
421
+ strokeWidth = 6,
422
+ touchTargetSize = 20,
423
+ onClick,
424
+ className,
425
+ }: DefaultHexEdgeProps) {
426
+ const touchTargetLength = Math.hypot(
427
+ position.x2 - position.x1,
428
+ position.y2 - position.y1,
429
+ );
430
+
431
+ return (
432
+ <g
433
+ onClick={onClick}
434
+ onKeyDown={(event) => handleKeyboardActivation(event, onClick)}
435
+ className={clsx(
436
+ "transition-all duration-150",
437
+ onClick && "cursor-pointer",
438
+ className,
439
+ )}
440
+ role={onClick ? "button" : undefined}
441
+ tabIndex={onClick ? 0 : undefined}
442
+ aria-label={onClick ? "Hex edge" : undefined}
443
+ >
444
+ {/* Invisible touch target */}
445
+ <rect
446
+ x={position.midX - touchTargetLength / 2}
447
+ y={position.midY - touchTargetSize / 2}
448
+ width={touchTargetLength}
449
+ height={touchTargetSize}
450
+ rx={touchTargetSize / 2}
451
+ fill="rgba(255,255,255,0.001)"
452
+ transform={`rotate(${position.edgeAngle} ${position.midX} ${position.midY})`}
453
+ pointerEvents="all"
454
+ />
455
+ {/* Visible edge */}
456
+ <line
457
+ x1={position.x1}
458
+ y1={position.y1}
459
+ x2={position.x2}
460
+ y2={position.y2}
461
+ stroke={color}
462
+ strokeWidth={hasOwner ? strokeWidth : strokeWidth / 2}
463
+ strokeLinecap="round"
464
+ className={hasOwner ? "" : "opacity-30"}
465
+ />
466
+ </g>
467
+ );
468
+ }
469
+
470
+ export interface DefaultHexVertexProps {
471
+ position: { x: number; y: number };
472
+ color: string;
473
+ stroke?: string;
474
+ strokeWidth?: number;
475
+ hasOwner?: boolean;
476
+ isSelected?: boolean;
477
+ isHighlighted?: boolean;
478
+ size?: number;
479
+ touchTargetSize?: number;
480
+ shape?: "circle" | "square";
481
+ onClick?: () => void;
482
+ onPointerEnter?: () => void;
483
+ onPointerLeave?: () => void;
484
+ className?: string;
485
+ }
486
+
487
+ /** Pre-built vertex/settlement component for use in `renderVertex`. */
488
+ export function DefaultHexVertex({
489
+ position,
490
+ color,
491
+ stroke = "#1e293b",
492
+ strokeWidth = 1.5,
493
+ hasOwner = true,
494
+ isSelected = false,
495
+ isHighlighted = false,
496
+ size = 10,
497
+ touchTargetSize = 22,
498
+ shape = "circle",
499
+ onClick,
500
+ onPointerEnter,
501
+ onPointerLeave,
502
+ className,
503
+ }: DefaultHexVertexProps) {
504
+ const effectiveColor = isSelected
505
+ ? "rgba(59, 130, 246, 0.8)"
506
+ : isHighlighted
507
+ ? "rgba(34, 197, 94, 0.8)"
508
+ : color;
509
+
510
+ const effectiveStroke = isSelected
511
+ ? "#3b82f6"
512
+ : isHighlighted
513
+ ? "#22c55e"
514
+ : stroke;
515
+
516
+ const effectiveStrokeWidth = isSelected || isHighlighted ? 3 : strokeWidth;
517
+
518
+ return (
519
+ <g
520
+ onClick={onClick}
521
+ onPointerEnter={onPointerEnter}
522
+ onPointerLeave={onPointerLeave}
523
+ onKeyDown={(event) => handleKeyboardActivation(event, onClick)}
524
+ className={clsx(
525
+ "transition-all duration-150",
526
+ onClick && "cursor-pointer hover:scale-110",
527
+ className,
528
+ )}
529
+ style={{ transformOrigin: `${position.x}px ${position.y}px` }}
530
+ role={onClick ? "button" : undefined}
531
+ tabIndex={onClick ? 0 : undefined}
532
+ aria-label={onClick ? "Hex vertex" : undefined}
533
+ >
534
+ {/* Invisible touch target */}
535
+ <circle
536
+ cx={position.x}
537
+ cy={position.y}
538
+ r={touchTargetSize}
539
+ fill="rgba(255,255,255,0.001)"
540
+ pointerEvents="all"
541
+ />
542
+ {/* Visible vertex */}
543
+ {shape === "square" ? (
544
+ <rect
545
+ x={position.x - size}
546
+ y={position.y - size}
547
+ width={size * 2}
548
+ height={size * 2}
549
+ fill={effectiveColor}
550
+ stroke={effectiveStroke}
551
+ strokeWidth={effectiveStrokeWidth}
552
+ className={hasOwner ? "" : "opacity-30"}
553
+ />
554
+ ) : (
555
+ <circle
556
+ cx={position.x}
557
+ cy={position.y}
558
+ r={hasOwner ? size : size * 0.5}
559
+ fill={effectiveColor}
560
+ stroke={effectiveStroke}
561
+ strokeWidth={effectiveStrokeWidth}
562
+ className={hasOwner ? "" : "opacity-30"}
563
+ />
564
+ )}
565
+ </g>
566
+ );
567
+ }
568
+
569
+ // ============================================================================
570
+ // Interactive Helper Components (for placement UI)
571
+ // ============================================================================
572
+
573
+ export interface DefaultInteractiveVertexProps {
574
+ position: { x: number; y: number };
575
+ isHovered: boolean;
576
+ size?: number;
577
+ color?: string;
578
+ hoverColor?: string;
579
+ className?: string;
580
+ }
581
+ export function DefaultInteractiveVertex({
582
+ position,
583
+ isHovered,
584
+ size = 8,
585
+ color = "rgba(255, 255, 255, 0.2)",
586
+ hoverColor = "rgba(34, 197, 94, 0.8)",
587
+ className,
588
+ }: DefaultInteractiveVertexProps) {
589
+ return (
590
+ <circle
591
+ cx={position.x}
592
+ cy={position.y}
593
+ r={isHovered ? size * 1.5 : size}
594
+ fill={isHovered ? hoverColor : color}
595
+ stroke={isHovered ? "#22c55e" : "rgba(255,255,255,0.4)"}
596
+ strokeWidth={isHovered ? 2 : 1}
597
+ className={clsx("transition-all duration-150", className)}
598
+ />
599
+ );
600
+ }
601
+
602
+ export interface DefaultInteractiveEdgeProps {
603
+ position: EdgePosition;
604
+ isHovered: boolean;
605
+ strokeWidth?: number;
606
+ color?: string;
607
+ hoverColor?: string;
608
+ className?: string;
609
+ }
610
+ export function DefaultInteractiveEdge({
611
+ position,
612
+ isHovered,
613
+ strokeWidth = 4,
614
+ color = "rgba(255, 255, 255, 0.15)",
615
+ hoverColor = "rgba(251, 146, 60, 0.8)",
616
+ className,
617
+ }: DefaultInteractiveEdgeProps) {
618
+ return (
619
+ <line
620
+ x1={position.x1}
621
+ y1={position.y1}
622
+ x2={position.x2}
623
+ y2={position.y2}
624
+ stroke={isHovered ? hoverColor : color}
625
+ strokeWidth={isHovered ? strokeWidth * 1.5 : strokeWidth}
626
+ strokeLinecap="round"
627
+ className={clsx("transition-all duration-150", className)}
628
+ />
629
+ );
630
+ }
631
+
632
+ // ============================================================================
633
+ // Hex Math Utilities
634
+ // ============================================================================
635
+
636
+ export const hexUtils = {
637
+ /** Convert axial coordinates to pixel position. */
638
+ axialToPixel(
639
+ q: number,
640
+ r: number,
641
+ size: number,
642
+ orientation: HexOrientation,
643
+ ): { x: number; y: number } {
644
+ if (orientation === "pointy-top") {
645
+ const x = size * (Math.sqrt(3) * q + (Math.sqrt(3) / 2) * r);
646
+ const y = size * ((3 / 2) * r);
647
+ return { x, y };
648
+ } else {
649
+ const x = size * ((3 / 2) * q);
650
+ const y = size * ((Math.sqrt(3) / 2) * q + Math.sqrt(3) * r);
651
+ return { x, y };
652
+ }
653
+ },
654
+
655
+ getNeighbors(q: number, r: number): Array<{ q: number; r: number }> {
656
+ return [
657
+ { q: q + 1, r: r },
658
+ { q: q + 1, r: r - 1 },
659
+ { q: q, r: r - 1 },
660
+ { q: q - 1, r: r },
661
+ { q: q - 1, r: r + 1 },
662
+ { q: q, r: r + 1 },
663
+ ];
664
+ },
665
+
666
+ getDistance(q1: number, r1: number, q2: number, r2: number): number {
667
+ return (
668
+ (Math.abs(q1 - q2) + Math.abs(q1 + r1 - q2 - r2) + Math.abs(r1 - r2)) / 2
669
+ );
670
+ },
671
+
672
+ getHexCorners(
673
+ centerX: number,
674
+ centerY: number,
675
+ size: number,
676
+ orientation: HexOrientation,
677
+ ): Array<{ x: number; y: number }> {
678
+ const corners: Array<{ x: number; y: number }> = [];
679
+ const startAngle = orientation === "pointy-top" ? 30 : 0;
680
+
681
+ for (let i = 0; i < 6; i++) {
682
+ const angleDeg = startAngle + 60 * i;
683
+ const angleRad = (Math.PI / 180) * angleDeg;
684
+ corners.push({
685
+ x: centerX + size * Math.cos(angleRad),
686
+ y: centerY + size * Math.sin(angleRad),
687
+ });
688
+ }
689
+ return corners;
690
+ },
691
+
692
+ getHexPoints(
693
+ centerX: number,
694
+ centerY: number,
695
+ size: number,
696
+ orientation: HexOrientation,
697
+ ): string {
698
+ const corners = this.getHexCorners(centerX, centerY, size, orientation);
699
+ return corners.map((c) => `${c.x},${c.y}`).join(" ");
700
+ },
701
+
702
+ getEdgePosition(
703
+ hex1Pos: { x: number; y: number },
704
+ hex2Pos: { x: number; y: number },
705
+ size: number,
706
+ ): EdgePosition {
707
+ const midX = (hex1Pos.x + hex2Pos.x) / 2;
708
+ const midY = (hex1Pos.y + hex2Pos.y) / 2;
709
+ const centerAngleRad = Math.atan2(
710
+ hex2Pos.y - hex1Pos.y,
711
+ hex2Pos.x - hex1Pos.x,
712
+ );
713
+
714
+ // Calculate edge endpoints perpendicular to the line between hex centers
715
+ const edgeAngleRad = centerAngleRad + Math.PI / 2;
716
+ const edgeLength = size * 0.8;
717
+ const centerAngle = (centerAngleRad * 180) / Math.PI;
718
+ const edgeAngle = (edgeAngleRad * 180) / Math.PI;
719
+
720
+ return {
721
+ x1: midX - (edgeLength / 2) * Math.cos(edgeAngleRad),
722
+ y1: midY - (edgeLength / 2) * Math.sin(edgeAngleRad),
723
+ x2: midX + (edgeLength / 2) * Math.cos(edgeAngleRad),
724
+ y2: midY + (edgeLength / 2) * Math.sin(edgeAngleRad),
725
+ midX,
726
+ midY,
727
+ centerAngle,
728
+ edgeAngle,
729
+ };
730
+ },
731
+
732
+ getVertexPosition(
733
+ hex1Pos: { x: number; y: number },
734
+ hex2Pos: { x: number; y: number },
735
+ hex3Pos: { x: number; y: number },
736
+ ): { x: number; y: number } {
737
+ return {
738
+ x: (hex1Pos.x + hex2Pos.x + hex3Pos.x) / 3,
739
+ y: (hex1Pos.y + hex2Pos.y + hex3Pos.y) / 3,
740
+ };
741
+ },
742
+ };
743
+
744
+ // ============================================================================
745
+ // Component
746
+ // ============================================================================
747
+
748
+ export interface HexGridComponent {
749
+ <const TBoard extends AnyHexBoardInput>(
750
+ props: HexGridBoardProps<TBoard>,
751
+ ): ReactNode;
752
+ <const TProps extends HexGeneratedGridInputProps>(
753
+ props: HexGridProps<TProps>,
754
+ ): ReactNode;
755
+ <const TProps extends HexAuthoredGridInputProps>(
756
+ props: HexGridProps<TProps>,
757
+ ): ReactNode;
758
+ }
759
+
760
+ function HexGridImpl(
761
+ props: HexGridBoardProps<AnyHexBoardInput> | HexGridProps<HexGridInputProps>,
762
+ ) {
763
+ const {
764
+ orientation = "pointy-top",
765
+ hexSize = 50,
766
+ renderTile,
767
+ renderEdge,
768
+ renderVertex,
769
+ width = 800,
770
+ height = 600,
771
+ enablePanZoom = true,
772
+ initialZoom = 1,
773
+ minZoom = 0.5,
774
+ maxZoom = 3,
775
+ className,
776
+ interactiveSpaces,
777
+ interactiveVertices,
778
+ interactiveEdges,
779
+ renderInteractiveSpace,
780
+ renderInteractiveVertex,
781
+ renderInteractiveEdge,
782
+ interactiveVertexSize = 12,
783
+ interactiveEdgeSize = 10,
784
+ } = props;
785
+ const board =
786
+ "board" in props
787
+ ? props.board
788
+ : (("spaces" in props
789
+ ? {
790
+ id: "__hex-grid__",
791
+ orientation,
792
+ spaces: props.spaces,
793
+ edges: props.edges ?? [],
794
+ vertices: props.vertices ?? [],
795
+ }
796
+ : {
797
+ id: "__hex-grid__",
798
+ orientation,
799
+ tiles: props.tiles,
800
+ edges: props.edges ?? [],
801
+ vertices: props.vertices ?? [],
802
+ }) satisfies AnyHexBoardInput);
803
+ // Pan/zoom is only enabled on mobile devices when the prop is true
804
+ const isMobile = useIsMobile();
805
+ const effectivePanZoom = enablePanZoom && isMobile;
806
+ const normalizedBoard = useMemo<NormalizedHexBoard<AnyHexBoardInput>>(
807
+ () => normalizeHexBoardInput(board),
808
+ [board],
809
+ );
810
+ const resolvedTiles = normalizedBoard.tiles;
811
+ const resolvedEdges = normalizedBoard.edges;
812
+ const resolvedVertices = normalizedBoard.vertices;
813
+ const resolvedOrientation = normalizedBoard.orientation ?? orientation;
814
+
815
+ // Hover state for interactive elements
816
+ const [hoveredSpaceId, setHoveredSpaceId] = useState<string | null>(null);
817
+ const [hoveredVertexId, setHoveredVertexId] = useState<string | null>(null);
818
+ const [hoveredEdgeId, setHoveredEdgeId] = useState<string | null>(null);
819
+
820
+ // Use the unified pan/zoom hook
821
+ const { transform, bind, isDragging } = usePanZoom({
822
+ enabled: effectivePanZoom,
823
+ initialZoom,
824
+ minZoom,
825
+ maxZoom,
826
+ mode: "viewbox",
827
+ });
828
+
829
+ // Pre-compute tile positions
830
+ const tilePositions = useMemo(() => {
831
+ const positions = new Map<string, { x: number; y: number }>();
832
+ resolvedTiles.forEach((tile) => {
833
+ positions.set(
834
+ tile.id,
835
+ hexUtils.axialToPixel(tile.q, tile.r, hexSize, resolvedOrientation),
836
+ );
837
+ });
838
+ return positions;
839
+ }, [resolvedTiles, hexSize, resolvedOrientation]);
840
+
841
+ // Build a `HexTileGeometry` for a tile centered at `position`.
842
+ //
843
+ // The closures intentionally re-derive corners on demand so callers
844
+ // can pass a per-call `inset` without the grid pre-computing every
845
+ // possible inset. Hex math is cheap (six trig calls).
846
+ const buildTileGeometry = useMemo(
847
+ () =>
848
+ (position: { x: number; y: number }): HexTileGeometry => {
849
+ const corners = (options?: { inset?: number }) => {
850
+ const inset = options?.inset ?? 0;
851
+ const radius = Math.max(0, hexSize - inset);
852
+ return hexUtils.getHexCorners(0, 0, radius, resolvedOrientation);
853
+ };
854
+ const points = (options?: { inset?: number }) =>
855
+ corners(options)
856
+ .map((corner) => `${corner.x},${corner.y}`)
857
+ .join(" ");
858
+ const outer = corners();
859
+ const xs = outer.map((corner) => corner.x);
860
+ const ys = outer.map((corner) => corner.y);
861
+ const minX = Math.min(...xs);
862
+ const maxX = Math.max(...xs);
863
+ const minY = Math.min(...ys);
864
+ const maxY = Math.max(...ys);
865
+ return {
866
+ size: hexSize,
867
+ orientation: resolvedOrientation,
868
+ center: { x: 0, y: 0 },
869
+ position,
870
+ corners,
871
+ points,
872
+ bounds: {
873
+ minX,
874
+ minY,
875
+ maxX,
876
+ maxY,
877
+ width: maxX - minX,
878
+ height: maxY - minY,
879
+ },
880
+ };
881
+ },
882
+ [hexSize, resolvedOrientation],
883
+ );
884
+
885
+ const resolvedEdgePositions = useMemo(
886
+ () =>
887
+ resolvedEdges.flatMap((edge) => {
888
+ const pos1 = tilePositions.get(edge.hex1);
889
+ const pos2 = tilePositions.get(edge.hex2);
890
+ if (!pos1 || !pos2) {
891
+ return [];
892
+ }
893
+ return [
894
+ {
895
+ edge,
896
+ interactiveEdge: {
897
+ ...edge,
898
+ spaceIds: [edge.hex1, edge.hex2] as const,
899
+ position: hexUtils.getEdgePosition(pos1, pos2, hexSize),
900
+ } as InteractiveHexEdge<AnyHexBoardInput>,
901
+ },
902
+ ];
903
+ }),
904
+ [hexSize, resolvedEdges, tilePositions],
905
+ );
906
+
907
+ const resolvedVertexPositions = useMemo(
908
+ () =>
909
+ resolvedVertices.flatMap((vertex) => {
910
+ const [hex0, hex1, hex2] = vertex.hexes;
911
+ if (!hex0 || !hex1 || !hex2) {
912
+ return [];
913
+ }
914
+
915
+ const pos0 = tilePositions.get(hex0);
916
+ const pos1 = tilePositions.get(hex1);
917
+ const pos2 = tilePositions.get(hex2);
918
+ if (!pos0 || !pos1 || !pos2) {
919
+ return [];
920
+ }
921
+
922
+ return [
923
+ {
924
+ vertex,
925
+ interactiveVertex: {
926
+ ...vertex,
927
+ spaceIds: vertex.hexes,
928
+ position: hexUtils.getVertexPosition(pos0, pos1, pos2),
929
+ } as InteractiveHexVertex<AnyHexBoardInput>,
930
+ },
931
+ ];
932
+ }),
933
+ [resolvedVertices, tilePositions],
934
+ );
935
+
936
+ // Calculate bounds for viewBox
937
+ const bounds = useMemo(() => {
938
+ if (resolvedTiles.length === 0) {
939
+ return { minX: 0, minY: 0, width: 400, height: 300 };
940
+ }
941
+
942
+ let minX = Infinity,
943
+ minY = Infinity,
944
+ maxX = -Infinity,
945
+ maxY = -Infinity;
946
+ resolvedTiles.forEach((tile) => {
947
+ const pos = tilePositions.get(tile.id);
948
+ if (pos) {
949
+ minX = Math.min(minX, pos.x - hexSize);
950
+ minY = Math.min(minY, pos.y - hexSize);
951
+ maxX = Math.max(maxX, pos.x + hexSize);
952
+ maxY = Math.max(maxY, pos.y + hexSize);
953
+ }
954
+ });
955
+
956
+ const padding = hexSize;
957
+ return {
958
+ minX: minX - padding,
959
+ minY: minY - padding,
960
+ width: maxX - minX + padding * 2,
961
+ height: maxY - minY + padding * 2,
962
+ };
963
+ }, [resolvedTiles, tilePositions, hexSize]);
964
+
965
+ // Calculate viewBox with pan and zoom
966
+ const viewBox = calculateViewBox(bounds, transform);
967
+
968
+ // Parse viewBox for zoom indicator positioning
969
+ const viewBoxParts = viewBox.split(" ").map(Number);
970
+ const viewBoxX = viewBoxParts[0] ?? 0;
971
+ const viewBoxY = viewBoxParts[1] ?? 0;
972
+ const viewBoxHeight = viewBoxParts[3] ?? 0;
973
+
974
+ return (
975
+ <svg
976
+ width={width}
977
+ height={height}
978
+ viewBox={viewBox}
979
+ className={clsx(
980
+ "hex-grid",
981
+ effectivePanZoom && "touch-none",
982
+ isDragging && "cursor-grabbing",
983
+ effectivePanZoom && !isDragging && "cursor-grab",
984
+ className,
985
+ )}
986
+ {...bind()}
987
+ role="img"
988
+ aria-label="Hex grid game board"
989
+ >
990
+ <defs>
991
+ {/* Gradient for ocean tiles */}
992
+ <linearGradient id="oceanGradient" x1="0%" y1="0%" x2="100%" y2="100%">
993
+ <stop offset="0%" stopColor="#0ea5e9" />
994
+ <stop offset="100%" stopColor="#0284c7" />
995
+ </linearGradient>
996
+ {/* Drop shadow filter */}
997
+ <filter id="hexShadow" x="-20%" y="-20%" width="140%" height="140%">
998
+ <feDropShadow dx="1" dy="1" stdDeviation="2" floodOpacity="0.3" />
999
+ </filter>
1000
+ </defs>
1001
+
1002
+ {/* Tiles layer */}
1003
+ <g className="tiles" role="list" aria-label="Hex tiles">
1004
+ {resolvedTiles.map((tile) => {
1005
+ const pos = tilePositions.get(tile.id);
1006
+ if (!pos) return null;
1007
+
1008
+ const geometry = buildTileGeometry(pos);
1009
+ return (
1010
+ <g
1011
+ key={tile.id}
1012
+ transform={`translate(${pos.x}, ${pos.y})`}
1013
+ role="listitem"
1014
+ aria-label={tile.label ?? `Tile ${tile.id}`}
1015
+ >
1016
+ {renderTile(tile, geometry)}
1017
+ </g>
1018
+ );
1019
+ })}
1020
+ </g>
1021
+
1022
+ {/* Interactive spaces layer */}
1023
+ {interactiveSpaces && resolvedTiles.length > 0 && (
1024
+ <g
1025
+ className="interactive-spaces"
1026
+ role="list"
1027
+ aria-label="Interactive spaces"
1028
+ >
1029
+ {resolvedTiles.map((space) => {
1030
+ const pos = tilePositions.get(space.id);
1031
+ if (!pos) return null;
1032
+ const state = interactiveTargetRenderState(
1033
+ interactiveSpaces,
1034
+ space.id,
1035
+ hoveredSpaceId === space.id,
1036
+ );
1037
+ const isSelectable = isInteractiveTargetSelectable(
1038
+ interactiveSpaces,
1039
+ state,
1040
+ );
1041
+ return (
1042
+ <g
1043
+ key={space.id}
1044
+ transform={`translate(${pos.x}, ${pos.y})`}
1045
+ role={isSelectable ? "button" : undefined}
1046
+ className={clsx(isSelectable && "cursor-pointer")}
1047
+ onPointerEnter={() => setHoveredSpaceId(space.id)}
1048
+ onPointerLeave={() =>
1049
+ setHoveredSpaceId((currentId) =>
1050
+ currentId === space.id ? null : currentId,
1051
+ )
1052
+ }
1053
+ onClick={
1054
+ isSelectable
1055
+ ? () => {
1056
+ void interactiveSpaces.selectTargetId?.(space.id);
1057
+ }
1058
+ : undefined
1059
+ }
1060
+ onKeyDown={(event) =>
1061
+ handleKeyboardActivation(
1062
+ event,
1063
+ isSelectable
1064
+ ? () => {
1065
+ void interactiveSpaces.selectTargetId?.(space.id);
1066
+ }
1067
+ : undefined,
1068
+ )
1069
+ }
1070
+ tabIndex={isSelectable ? 0 : undefined}
1071
+ aria-label={
1072
+ isSelectable ? `Select space ${space.id}` : undefined
1073
+ }
1074
+ >
1075
+ {isSelectable && (
1076
+ <polygon
1077
+ points={buildTileGeometry(pos).points({
1078
+ inset: hexSize * 0.05,
1079
+ })}
1080
+ fill="rgba(255,255,255,0.001)"
1081
+ pointerEvents="all"
1082
+ />
1083
+ )}
1084
+ {renderInteractiveSpace
1085
+ ? renderInteractiveSpace(space, state)
1086
+ : null}
1087
+ </g>
1088
+ );
1089
+ })}
1090
+ </g>
1091
+ )}
1092
+
1093
+ {/* Edges layer (for roads) */}
1094
+ {resolvedEdges.length > 0 && (
1095
+ <g className="edges" role="list" aria-label="Hex edges">
1096
+ {resolvedEdgePositions.map(({ edge, interactiveEdge }) => {
1097
+ return (
1098
+ <g key={edge.id} role="listitem">
1099
+ {renderEdge(edge, interactiveEdge.position)}
1100
+ </g>
1101
+ );
1102
+ })}
1103
+ </g>
1104
+ )}
1105
+
1106
+ {/* Vertices layer (for settlements) */}
1107
+ {resolvedVertices.length > 0 && (
1108
+ <g className="vertices" role="list" aria-label="Hex vertices">
1109
+ {resolvedVertexPositions.map(({ vertex, interactiveVertex }) => {
1110
+ return (
1111
+ <g key={vertex.id} role="listitem">
1112
+ {renderVertex(vertex, interactiveVertex.position)}
1113
+ </g>
1114
+ );
1115
+ })}
1116
+ </g>
1117
+ )}
1118
+
1119
+ {/* Interactive edges layer (for road placement) */}
1120
+ {interactiveEdges && resolvedEdgePositions.length > 0 && (
1121
+ <g
1122
+ className="interactive-edges"
1123
+ role="list"
1124
+ aria-label="Interactive edges for placement"
1125
+ >
1126
+ {resolvedEdgePositions.map(({ interactiveEdge: edge }) => {
1127
+ const state = interactiveTargetRenderState(
1128
+ interactiveEdges,
1129
+ edge.id,
1130
+ hoveredEdgeId === edge.id,
1131
+ );
1132
+ const isSelectable = isInteractiveTargetSelectable(
1133
+ interactiveEdges,
1134
+ state,
1135
+ );
1136
+ const touchTargetLength = Math.hypot(
1137
+ edge.position.x2 - edge.position.x1,
1138
+ edge.position.y2 - edge.position.y1,
1139
+ );
1140
+ return (
1141
+ <g
1142
+ key={edge.id}
1143
+ role={isSelectable ? "button" : undefined}
1144
+ className={clsx(isSelectable && "cursor-pointer")}
1145
+ onPointerEnter={() => setHoveredEdgeId(edge.id)}
1146
+ onPointerLeave={() =>
1147
+ setHoveredEdgeId((currentId) =>
1148
+ currentId === edge.id ? null : currentId,
1149
+ )
1150
+ }
1151
+ onClick={
1152
+ isSelectable
1153
+ ? () => {
1154
+ void interactiveEdges.selectTargetId?.(edge.id);
1155
+ }
1156
+ : undefined
1157
+ }
1158
+ onKeyDown={(event) =>
1159
+ handleKeyboardActivation(
1160
+ event,
1161
+ isSelectable
1162
+ ? () => {
1163
+ void interactiveEdges.selectTargetId?.(edge.id);
1164
+ }
1165
+ : undefined,
1166
+ )
1167
+ }
1168
+ tabIndex={isSelectable ? 0 : undefined}
1169
+ aria-label={isSelectable ? `Select edge ${edge.id}` : undefined}
1170
+ >
1171
+ {isSelectable && (
1172
+ <rect
1173
+ x={edge.position.midX - touchTargetLength / 2}
1174
+ y={edge.position.midY - interactiveEdgeSize}
1175
+ width={touchTargetLength}
1176
+ height={interactiveEdgeSize * 2}
1177
+ rx={interactiveEdgeSize}
1178
+ fill="rgba(255,255,255,0.001)"
1179
+ transform={`rotate(${edge.position.edgeAngle} ${edge.position.midX} ${edge.position.midY})`}
1180
+ pointerEvents="all"
1181
+ />
1182
+ )}
1183
+ {renderInteractiveEdge ? (
1184
+ renderInteractiveEdge(edge, edge.position, state)
1185
+ ) : state.isEnabled && state.isEligible ? (
1186
+ <DefaultInteractiveEdge
1187
+ position={edge.position}
1188
+ isHovered={state.isHovered}
1189
+ strokeWidth={interactiveEdgeSize * 0.6}
1190
+ />
1191
+ ) : null}
1192
+ </g>
1193
+ );
1194
+ })}
1195
+ </g>
1196
+ )}
1197
+
1198
+ {/* Interactive vertices layer (for settlement placement) */}
1199
+ {interactiveVertices && resolvedVertexPositions.length > 0 && (
1200
+ <g
1201
+ className="interactive-vertices"
1202
+ role="list"
1203
+ aria-label="Interactive vertices for placement"
1204
+ >
1205
+ {resolvedVertexPositions.map(({ interactiveVertex: vertex }) => {
1206
+ const state = interactiveTargetRenderState(
1207
+ interactiveVertices,
1208
+ vertex.id,
1209
+ hoveredVertexId === vertex.id,
1210
+ );
1211
+ const isSelectable = isInteractiveTargetSelectable(
1212
+ interactiveVertices,
1213
+ state,
1214
+ );
1215
+ return (
1216
+ <g
1217
+ key={vertex.id}
1218
+ role={isSelectable ? "button" : undefined}
1219
+ className={clsx(isSelectable && "cursor-pointer")}
1220
+ onPointerEnter={() => setHoveredVertexId(vertex.id)}
1221
+ onPointerLeave={() =>
1222
+ setHoveredVertexId((currentId) =>
1223
+ currentId === vertex.id ? null : currentId,
1224
+ )
1225
+ }
1226
+ onClick={
1227
+ isSelectable
1228
+ ? () => {
1229
+ void interactiveVertices.selectTargetId?.(vertex.id);
1230
+ }
1231
+ : undefined
1232
+ }
1233
+ onKeyDown={(event) =>
1234
+ handleKeyboardActivation(
1235
+ event,
1236
+ isSelectable
1237
+ ? () => {
1238
+ void interactiveVertices.selectTargetId?.(vertex.id);
1239
+ }
1240
+ : undefined,
1241
+ )
1242
+ }
1243
+ tabIndex={isSelectable ? 0 : undefined}
1244
+ aria-label={
1245
+ isSelectable ? `Select vertex ${vertex.id}` : undefined
1246
+ }
1247
+ >
1248
+ {isSelectable && (
1249
+ <circle
1250
+ cx={vertex.position.x}
1251
+ cy={vertex.position.y}
1252
+ r={interactiveVertexSize * 1.5}
1253
+ fill="rgba(255,255,255,0.001)"
1254
+ pointerEvents="all"
1255
+ />
1256
+ )}
1257
+ {renderInteractiveVertex ? (
1258
+ renderInteractiveVertex(vertex, vertex.position, state)
1259
+ ) : state.isEnabled && state.isEligible ? (
1260
+ <DefaultInteractiveVertex
1261
+ position={vertex.position}
1262
+ isHovered={state.isHovered}
1263
+ size={interactiveVertexSize * 0.6}
1264
+ />
1265
+ ) : null}
1266
+ </g>
1267
+ );
1268
+ })}
1269
+ </g>
1270
+ )}
1271
+
1272
+ {/* Zoom indicator (for mobile) */}
1273
+ {effectivePanZoom && transform.zoom !== 1 && (
1274
+ <g
1275
+ transform={`translate(${viewBoxX + 10}, ${viewBoxY + viewBoxHeight - 30})`}
1276
+ >
1277
+ <rect
1278
+ x={0}
1279
+ y={0}
1280
+ width={60}
1281
+ height={20}
1282
+ rx={4}
1283
+ fill="rgba(0,0,0,0.6)"
1284
+ />
1285
+ <text x={30} y={14} textAnchor="middle" fill="white" fontSize={12}>
1286
+ {Math.round(transform.zoom * 100)}%
1287
+ </text>
1288
+ </g>
1289
+ )}
1290
+ </svg>
1291
+ );
1292
+ }
1293
+
1294
+ export const HexGrid = HexGridImpl as HexGridComponent;