@dreamboard-games/sdk 0.2.0

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 (369) hide show
  1. package/LICENSE.md +96 -0
  2. package/README.md +12 -0
  3. package/dist/HandView-ncJIVLhN.d.ts +193 -0
  4. package/dist/ResourceCounter-CTREyF73.d.ts +102 -0
  5. package/dist/ThemeProvider-fy0_QzgO.d.ts +99 -0
  6. package/dist/bundle-TIZcw8LB.d.ts +281 -0
  7. package/dist/cards-Sl3b40Mv.d.ts +13 -0
  8. package/dist/chunk-7YAHLYBR.js +481 -0
  9. package/dist/chunk-7YAHLYBR.js.map +1 -0
  10. package/dist/chunk-FDNZTDD6.js +8085 -0
  11. package/dist/chunk-FDNZTDD6.js.map +1 -0
  12. package/dist/chunk-GKKBPPSW.js +598 -0
  13. package/dist/chunk-GKKBPPSW.js.map +1 -0
  14. package/dist/chunk-I46YJSOD.js +1 -0
  15. package/dist/chunk-I46YJSOD.js.map +1 -0
  16. package/dist/chunk-KAELH4KC.js +104 -0
  17. package/dist/chunk-KAELH4KC.js.map +1 -0
  18. package/dist/chunk-PZ5AY32C.js +10 -0
  19. package/dist/chunk-PZ5AY32C.js.map +1 -0
  20. package/dist/chunk-T3ZKNUZ7.js +1 -0
  21. package/dist/chunk-T3ZKNUZ7.js.map +1 -0
  22. package/dist/chunk-T52J5RMF.js +1 -0
  23. package/dist/chunk-T52J5RMF.js.map +1 -0
  24. package/dist/chunk-TDSWKVZ4.js +5401 -0
  25. package/dist/chunk-TDSWKVZ4.js.map +1 -0
  26. package/dist/chunk-U5C6BONG.js +34 -0
  27. package/dist/chunk-U5C6BONG.js.map +1 -0
  28. package/dist/chunk-VDXOF4FW.js +69 -0
  29. package/dist/chunk-VDXOF4FW.js.map +1 -0
  30. package/dist/chunk-VFTAA4WO.js +115 -0
  31. package/dist/chunk-VFTAA4WO.js.map +1 -0
  32. package/dist/chunk-WN74KVNY.js +17 -0
  33. package/dist/chunk-WN74KVNY.js.map +1 -0
  34. package/dist/chunk-WYPQ3GG5.js +10990 -0
  35. package/dist/chunk-WYPQ3GG5.js.map +1 -0
  36. package/dist/components-D5ZRE2Hl.d.ts +1451 -0
  37. package/dist/generated/runtime/primitives.d.ts +12 -0
  38. package/dist/generated/runtime/primitives.js +180 -0
  39. package/dist/generated/runtime/primitives.js.map +1 -0
  40. package/dist/generated/runtime-api.d.ts +3 -0
  41. package/dist/generated/runtime-api.js +2 -0
  42. package/dist/generated/runtime-api.js.map +1 -0
  43. package/dist/generated/runtime.d.ts +14 -0
  44. package/dist/generated/runtime.js +18 -0
  45. package/dist/generated/runtime.js.map +1 -0
  46. package/dist/generated/workspace-contract.d.ts +14 -0
  47. package/dist/generated/workspace-contract.js +14 -0
  48. package/dist/generated/workspace-contract.js.map +1 -0
  49. package/dist/hex-board-view-D_07hO6O.d.ts +933 -0
  50. package/dist/hex-color-MhOyuY-o.d.ts +8 -0
  51. package/dist/index-BwqPQtBu.d.ts +1433 -0
  52. package/dist/index.d.ts +1 -0
  53. package/dist/index.js +12 -0
  54. package/dist/index.js.map +1 -0
  55. package/dist/infrastructure/reducer-bundle-abi.d.ts +1083 -0
  56. package/dist/infrastructure/reducer-bundle-abi.js +14 -0
  57. package/dist/infrastructure/reducer-bundle-abi.js.map +1 -0
  58. package/dist/infrastructure/workspace-codegen.d.ts +53 -0
  59. package/dist/infrastructure/workspace-codegen.js +44 -0
  60. package/dist/infrastructure/workspace-codegen.js.map +1 -0
  61. package/dist/manifest-contract-BNHVGFtU.d.ts +9 -0
  62. package/dist/package-set.d.ts +13 -0
  63. package/dist/package-set.js +12 -0
  64. package/dist/package-set.js.map +1 -0
  65. package/dist/primitive-props-DpKs-GCr.d.ts +11 -0
  66. package/dist/reducer.d.ts +3786 -0
  67. package/dist/reducer.js +8131 -0
  68. package/dist/reducer.js.map +1 -0
  69. package/dist/runtime/primitives.d.ts +226 -0
  70. package/dist/runtime/primitives.js +180 -0
  71. package/dist/runtime/primitives.js.map +1 -0
  72. package/dist/runtime/types/runtime-api.d.ts +1 -0
  73. package/dist/runtime/types/runtime-api.js +2 -0
  74. package/dist/runtime/types/runtime-api.js.map +1 -0
  75. package/dist/runtime/workspace-contract.d.ts +172 -0
  76. package/dist/runtime/workspace-contract.js +14 -0
  77. package/dist/runtime/workspace-contract.js.map +1 -0
  78. package/dist/runtime-api-3dshj6kK.d.ts +101 -0
  79. package/dist/runtime-api-DWxvTr-O.d.ts +379 -0
  80. package/dist/runtime.d.ts +58 -0
  81. package/dist/runtime.js +13 -0
  82. package/dist/runtime.js.map +1 -0
  83. package/dist/slots-1GPGihk8.d.ts +8 -0
  84. package/dist/testing.d.ts +149 -0
  85. package/dist/testing.js +513 -0
  86. package/dist/testing.js.map +1 -0
  87. package/dist/types.d.ts +496 -0
  88. package/dist/types.js +28 -0
  89. package/dist/types.js.map +1 -0
  90. package/dist/ui/components.d.ts +16 -0
  91. package/dist/ui/components.js +192 -0
  92. package/dist/ui/components.js.map +1 -0
  93. package/dist/ui/defaults.d.ts +19 -0
  94. package/dist/ui/defaults.js +104 -0
  95. package/dist/ui/defaults.js.map +1 -0
  96. package/dist/ui/plugin-styles.css +250 -0
  97. package/dist/ui/types/player-state.d.ts +365 -0
  98. package/dist/ui/types/player-state.js +1 -0
  99. package/dist/ui/types/player-state.js.map +1 -0
  100. package/dist/ui-contract-iQfTtUSL.d.ts +1161 -0
  101. package/dist/ui.d.ts +320 -0
  102. package/dist/ui.js +253 -0
  103. package/dist/ui.js.map +1 -0
  104. package/package.json +199 -0
  105. package/src/generated/reducer-contract/builders.ts +90 -0
  106. package/src/generated/reducer-contract/version.ts +9 -0
  107. package/src/generated/reducer-contract/wire.ts +100 -0
  108. package/src/generated/reducer-contract/zod.ts +101 -0
  109. package/src/generated/runtime/primitives.ts +2 -0
  110. package/src/generated/runtime-api.ts +5 -0
  111. package/src/generated/runtime.ts +35 -0
  112. package/src/generated/workspace-contract.ts +2 -0
  113. package/src/index.ts +7 -0
  114. package/src/infrastructure/reducer-bundle-abi.ts +8 -0
  115. package/src/infrastructure/reducer-contract/bundle.ts +37 -0
  116. package/src/infrastructure/workspace-codegen/hex-geometry.ts +69 -0
  117. package/src/infrastructure/workspace-codegen/index.ts +64 -0
  118. package/src/infrastructure/workspace-codegen/manifest-contract.ts +6632 -0
  119. package/src/infrastructure/workspace-codegen/manifest-validation.ts +795 -0
  120. package/src/infrastructure/workspace-codegen/ownership.ts +131 -0
  121. package/src/infrastructure/workspace-codegen/preset-card-sets.ts +169 -0
  122. package/src/infrastructure/workspace-codegen/seeds.ts +1705 -0
  123. package/src/infrastructure/workspace-codegen.ts +1 -0
  124. package/src/package-set.ts +19 -0
  125. package/src/reducer/authoring/contract.ts +157 -0
  126. package/src/reducer/authoring/effect.ts +224 -0
  127. package/src/reducer/authoring/game.ts +23 -0
  128. package/src/reducer/authoring/interaction.ts +98 -0
  129. package/src/reducer/authoring/phase.ts +300 -0
  130. package/src/reducer/authoring/types.ts +70 -0
  131. package/src/reducer/authoring/validation.ts +382 -0
  132. package/src/reducer/authoring/view-stage.ts +68 -0
  133. package/src/reducer/authoring.ts +29 -0
  134. package/src/reducer/bundle/ingress-bundle.ts +491 -0
  135. package/src/reducer/bundle/trusted/engine-instruction-resolver.ts +254 -0
  136. package/src/reducer/bundle/trusted/flow-instruction-resolver.ts +73 -0
  137. package/src/reducer/bundle/trusted/instruction-runner.ts +414 -0
  138. package/src/reducer/bundle/trusted/interaction-authorization.ts +137 -0
  139. package/src/reducer/bundle/trusted/interaction-collectors.ts +859 -0
  140. package/src/reducer/bundle/trusted/interaction-decision.ts +747 -0
  141. package/src/reducer/bundle/trusted/interaction-resolver.ts +95 -0
  142. package/src/reducer/bundle/trusted/interaction-types.ts +171 -0
  143. package/src/reducer/bundle/trusted/lifecycle-runner.ts +427 -0
  144. package/src/reducer/bundle/trusted/projection-builder.ts +356 -0
  145. package/src/reducer/bundle/trusted/projection-context.ts +39 -0
  146. package/src/reducer/bundle/trusted/rng-sampler.ts +150 -0
  147. package/src/reducer/bundle/trusted/runtime-registry.ts +120 -0
  148. package/src/reducer/bundle/trusted/runtime-scope.ts +336 -0
  149. package/src/reducer/bundle/trusted/simultaneous-player.ts +97 -0
  150. package/src/reducer/bundle/trusted/stage-resolver.ts +87 -0
  151. package/src/reducer/bundle/trusted/static-projection.ts +116 -0
  152. package/src/reducer/bundle/trusted/trusted-runtime-args.ts +97 -0
  153. package/src/reducer/bundle/trusted/trusted-runtime-result.ts +39 -0
  154. package/src/reducer/bundle/trusted/trusted-setup-profiles.ts +43 -0
  155. package/src/reducer/bundle/trusted/trusted-state-codec.ts +48 -0
  156. package/src/reducer/bundle/trusted-bundle.ts +97 -0
  157. package/src/reducer/bundle/types.ts +171 -0
  158. package/src/reducer/bundle.ts +2 -0
  159. package/src/reducer/client-param-schemas.ts +57 -0
  160. package/src/reducer/compose.ts +34 -0
  161. package/src/reducer/core/runtime-input.ts +30 -0
  162. package/src/reducer/core/runtime-instruction.ts +59 -0
  163. package/src/reducer/core/types.ts +62 -0
  164. package/src/reducer/definition-index.ts +277 -0
  165. package/src/reducer/derived.ts +106 -0
  166. package/src/reducer/effects.ts +92 -0
  167. package/src/reducer/engine/runtime-instruction-engine.ts +155 -0
  168. package/src/reducer/ingress/decode-runtime-input.ts +7 -0
  169. package/src/reducer/ingress/decode-session-state.ts +9 -0
  170. package/src/reducer/ingress/encode-session-state.ts +6 -0
  171. package/src/reducer/ingress/input-codec.ts +18 -0
  172. package/src/reducer/ingress/phase-schemas.ts +62 -0
  173. package/src/reducer/ingress/raw-types.ts +107 -0
  174. package/src/reducer/ingress/runtime-codec.ts +14 -0
  175. package/src/reducer/ingress/runtime-payload.ts +13 -0
  176. package/src/reducer/ingress/session-codec.ts +392 -0
  177. package/src/reducer/ingress/types.ts +6 -0
  178. package/src/reducer/inputs/boardInput.ts +217 -0
  179. package/src/reducer/inputs/boardTarget.ts +190 -0
  180. package/src/reducer/inputs/cardInput.ts +86 -0
  181. package/src/reducer/inputs/cardTarget.ts +101 -0
  182. package/src/reducer/inputs/choiceTarget.ts +104 -0
  183. package/src/reducer/inputs/defineInputs.ts +71 -0
  184. package/src/reducer/inputs/formInput.ts +809 -0
  185. package/src/reducer/inputs/many.ts +120 -0
  186. package/src/reducer/inputs/promptInput.ts +87 -0
  187. package/src/reducer/inputs/rngInput.ts +58 -0
  188. package/src/reducer/inputs/targetRule.ts +123 -0
  189. package/src/reducer/inputs.ts +41 -0
  190. package/src/reducer/model/definition.ts +1072 -0
  191. package/src/reducer/model/extract.ts +745 -0
  192. package/src/reducer/model/manifest.ts +570 -0
  193. package/src/reducer/model/queries.ts +641 -0
  194. package/src/reducer/model/runtime.ts +264 -0
  195. package/src/reducer/model/spec.ts +1386 -0
  196. package/src/reducer/model/table.ts +260 -0
  197. package/src/reducer/model.ts +7 -0
  198. package/src/reducer/ops.ts +1034 -0
  199. package/src/reducer/parse-utils.ts +28 -0
  200. package/src/reducer/per-player.ts +422 -0
  201. package/src/reducer/rng.ts +69 -0
  202. package/src/reducer/schema-helpers.ts +185 -0
  203. package/src/reducer/setup-bootstrap-helpers.ts +171 -0
  204. package/src/reducer/setup-bootstrap.ts +481 -0
  205. package/src/reducer/table-ops.ts +2671 -0
  206. package/src/reducer/table-queries.ts +372 -0
  207. package/src/reducer/transaction.ts +120 -0
  208. package/src/reducer.ts +314 -0
  209. package/src/runtime/primitives.ts +1 -0
  210. package/src/runtime/types/runtime-api.ts +1 -0
  211. package/src/runtime/workspace-contract.ts +32 -0
  212. package/src/runtime-internal/components/InteractionForm.tsx +1309 -0
  213. package/src/runtime-internal/components/PluginRuntime.tsx +103 -0
  214. package/src/runtime-internal/components/board/target-layer.ts +70 -0
  215. package/src/runtime-internal/context/ClientParamSchemaContext.tsx +44 -0
  216. package/src/runtime-internal/context/InteractionDraftContext.tsx +279 -0
  217. package/src/runtime-internal/context/PluginSessionContext.tsx +47 -0
  218. package/src/runtime-internal/context/PluginStateContext.tsx +262 -0
  219. package/src/runtime-internal/context/RuntimeContext.tsx +96 -0
  220. package/src/runtime-internal/defaults/components.tsx +409 -0
  221. package/src/runtime-internal/defaults/index.ts +11 -0
  222. package/src/runtime-internal/errors/ValidationError.ts +29 -0
  223. package/src/runtime-internal/hooks/useActivePlayers.ts +33 -0
  224. package/src/runtime-internal/hooks/useBoardInteractions.ts +665 -0
  225. package/src/runtime-internal/hooks/useGameSelector.ts +105 -0
  226. package/src/runtime-internal/hooks/useGameView.ts +9 -0
  227. package/src/runtime-internal/hooks/useInteractionByKey.ts +354 -0
  228. package/src/runtime-internal/hooks/useInteractionHandle.ts +438 -0
  229. package/src/runtime-internal/hooks/useIsMyTurn.ts +20 -0
  230. package/src/runtime-internal/hooks/useLobby.ts +76 -0
  231. package/src/runtime-internal/hooks/useMe.ts +48 -0
  232. package/src/runtime-internal/hooks/usePlayerInfo.ts +28 -0
  233. package/src/runtime-internal/hooks/usePlayerTurnOrder.ts +23 -0
  234. package/src/runtime-internal/hooks/usePluginRuntime.ts +147 -0
  235. package/src/runtime-internal/hooks/useSeatInbox.ts +61 -0
  236. package/src/runtime-internal/hooks/useSimultaneousPhase.ts +10 -0
  237. package/src/runtime-internal/index.ts +42 -0
  238. package/src/runtime-internal/internal.ts +43 -0
  239. package/src/runtime-internal/plugin-styles.css +250 -0
  240. package/src/runtime-internal/primitives/board.tsx +459 -0
  241. package/src/runtime-internal/primitives/dialog-lifecycle.ts +58 -0
  242. package/src/runtime-internal/primitives/dice.tsx +79 -0
  243. package/src/runtime-internal/primitives/game-ui-provider.tsx +35 -0
  244. package/src/runtime-internal/primitives/game.tsx +387 -0
  245. package/src/runtime-internal/primitives/hand-intent-adapter.ts +147 -0
  246. package/src/runtime-internal/primitives/hand-surface.tsx +594 -0
  247. package/src/runtime-internal/primitives/index.ts +196 -0
  248. package/src/runtime-internal/primitives/interaction-form-binding.tsx +56 -0
  249. package/src/runtime-internal/primitives/interaction-submit.ts +90 -0
  250. package/src/runtime-internal/primitives/interaction.tsx +987 -0
  251. package/src/runtime-internal/primitives/phase.tsx +43 -0
  252. package/src/runtime-internal/primitives/player-roster.tsx +302 -0
  253. package/src/runtime-internal/primitives/primitive-props.tsx +101 -0
  254. package/src/runtime-internal/primitives/prompt.tsx +255 -0
  255. package/src/runtime-internal/primitives/ui.tsx +60 -0
  256. package/src/runtime-internal/primitives/zone.tsx +791 -0
  257. package/src/runtime-internal/reducer.ts +30 -0
  258. package/src/runtime-internal/runtime/createPluginRuntimeAPI.ts +605 -0
  259. package/src/runtime-internal/types/plugin-state.ts +508 -0
  260. package/src/runtime-internal/types/reducer-state.ts +24 -0
  261. package/src/runtime-internal/types/runtime-api.ts +114 -0
  262. package/src/runtime-internal/ui-contract.ts +519 -0
  263. package/src/runtime-internal/utils/card-intent-adapter.ts +546 -0
  264. package/src/runtime-internal/utils/interaction-inputs.ts +492 -0
  265. package/src/runtime-internal/utils/interaction-labels.ts +23 -0
  266. package/src/runtime-internal/utils/interaction-router.ts +273 -0
  267. package/src/runtime-internal/utils/interaction-status.ts +74 -0
  268. package/src/runtime-internal/workspace-contract.ts +1170 -0
  269. package/src/runtime.ts +34 -0
  270. package/src/testing/create-expect-api.ts +352 -0
  271. package/src/testing/create-test-runtime.ts +381 -0
  272. package/src/testing/definitions.ts +127 -0
  273. package/src/testing/index.ts +3 -0
  274. package/src/testing.ts +1 -0
  275. package/src/type-stubs/manifest-contract.d.ts +42 -0
  276. package/src/type-stubs/manifest-contract.js +72 -0
  277. package/src/type-stubs/ui-contract.d.ts +5 -0
  278. package/src/type-stubs/ui-contract.js +1 -0
  279. package/src/types/authoring-card-properties.type-test.ts +266 -0
  280. package/src/types/authoring.ts +1282 -0
  281. package/src/types/cards.ts +19 -0
  282. package/src/types/contracts.ts +1550 -0
  283. package/src/types/generated-helpers.ts +35 -0
  284. package/src/types/index.ts +147 -0
  285. package/src/types/slots.ts +11 -0
  286. package/src/types.ts +1 -0
  287. package/src/ui/components/ActionButton.tsx +97 -0
  288. package/src/ui/components/ActionPanel.tsx +315 -0
  289. package/src/ui/components/Card.tsx +378 -0
  290. package/src/ui/components/CardDragSurface.tsx +1076 -0
  291. package/src/ui/components/ChromeSuppressionContext.tsx +70 -0
  292. package/src/ui/components/CostDisplay.tsx +145 -0
  293. package/src/ui/components/DiceRoller.tsx +581 -0
  294. package/src/ui/components/Drawer.tsx +180 -0
  295. package/src/ui/components/ErrorBoundary.tsx +275 -0
  296. package/src/ui/components/GameEndDisplay.tsx +398 -0
  297. package/src/ui/components/GameSkeleton.tsx +260 -0
  298. package/src/ui/components/Hand.tsx +468 -0
  299. package/src/ui/components/HandDock.tsx +299 -0
  300. package/src/ui/components/HandView.tsx +441 -0
  301. package/src/ui/components/MobileHandTray.tsx +381 -0
  302. package/src/ui/components/MoreActions.tsx +143 -0
  303. package/src/ui/components/PhaseIndicator.tsx +341 -0
  304. package/src/ui/components/PlayArea.tsx +146 -0
  305. package/src/ui/components/PrimaryActionButton.tsx +336 -0
  306. package/src/ui/components/PrimaryButton.tsx +45 -0
  307. package/src/ui/components/ResourceCounter.tsx +270 -0
  308. package/src/ui/components/StagingZone.tsx +134 -0
  309. package/src/ui/components/ThemedButton.tsx +113 -0
  310. package/src/ui/components/Toast.tsx +264 -0
  311. package/src/ui/components/board/HexGrid.tsx +1294 -0
  312. package/src/ui/components/board/NetworkGraph.tsx +476 -0
  313. package/src/ui/components/board/SlotSystem.tsx +388 -0
  314. package/src/ui/components/board/SquareGrid.tsx +1165 -0
  315. package/src/ui/components/board/TrackBoard.tsx +496 -0
  316. package/src/ui/components/board/ZoneMap.tsx +448 -0
  317. package/src/ui/components/board/hex-board-view.ts +123 -0
  318. package/src/ui/components/board/index.ts +142 -0
  319. package/src/ui/components/board/interaction-accessibility.ts +21 -0
  320. package/src/ui/components/board/target-layer.ts +66 -0
  321. package/src/ui/components/card-render-content.type-test.ts +27 -0
  322. package/src/ui/components/hand-layout-math.ts +163 -0
  323. package/src/ui/components/hand-pointer-engine.ts +413 -0
  324. package/src/ui/components/index.ts +245 -0
  325. package/src/ui/components.ts +1 -0
  326. package/src/ui/defaults/components.tsx +106 -0
  327. package/src/ui/defaults/index.ts +8 -0
  328. package/src/ui/defaults.ts +1 -0
  329. package/src/ui/errors/ValidationError.ts +29 -0
  330. package/src/ui/helpers/cards.ts +19 -0
  331. package/src/ui/helpers/track-board.ts +211 -0
  332. package/src/ui/hooks/useBoardTopology.ts +316 -0
  333. package/src/ui/hooks/useCards.ts +10 -0
  334. package/src/ui/hooks/useHandCardPointer.ts +381 -0
  335. package/src/ui/hooks/useHandLayout.ts +378 -0
  336. package/src/ui/hooks/useHandPresentation.ts +121 -0
  337. package/src/ui/hooks/useHexBoard.ts +74 -0
  338. package/src/ui/hooks/useHexGrid.ts +185 -0
  339. package/src/ui/hooks/useIsMobile.ts +35 -0
  340. package/src/ui/hooks/usePanZoom.ts +278 -0
  341. package/src/ui/hooks/useSquareBoard.ts +124 -0
  342. package/src/ui/hooks/useSquareGrid.ts +328 -0
  343. package/src/ui/index.ts +98 -0
  344. package/src/ui/internal/ui/alert.tsx +51 -0
  345. package/src/ui/internal/ui/button.tsx +58 -0
  346. package/src/ui/internal/ui/dialog.tsx +134 -0
  347. package/src/ui/internal/ui/input.tsx +21 -0
  348. package/src/ui/internal/ui/label.tsx +21 -0
  349. package/src/ui/internal/ui/select.tsx +129 -0
  350. package/src/ui/internal/ui/tooltip.tsx +54 -0
  351. package/src/ui/internal/ui/utils.ts +5 -0
  352. package/src/ui/plugin-styles.css +250 -0
  353. package/src/ui/primitives/dialog-lifecycle.ts +58 -0
  354. package/src/ui/primitives/dice.tsx +79 -0
  355. package/src/ui/primitives/primitive-props.tsx +101 -0
  356. package/src/ui/theme/ThemeProvider.tsx +252 -0
  357. package/src/ui/theme/board.ts +61 -0
  358. package/src/ui/theme/css-vars.ts +105 -0
  359. package/src/ui/theme/derive.ts +240 -0
  360. package/src/ui/theme/index.ts +61 -0
  361. package/src/ui/theme/presets/arcade.ts +261 -0
  362. package/src/ui/theme/presets/studio.ts +261 -0
  363. package/src/ui/theme/presets/tabletop.ts +266 -0
  364. package/src/ui/theme/tokens.ts +392 -0
  365. package/src/ui/types/hex-color.ts +20 -0
  366. package/src/ui/types/player-state.ts +463 -0
  367. package/src/ui/types/tiled-board.ts +785 -0
  368. package/src/ui/types/visual-state.ts +137 -0
  369. package/src/ui.ts +1 -0
@@ -0,0 +1,21 @@
1
+ import type { KeyboardEvent } from "react";
2
+
3
+ export function handleKeyboardActivation(
4
+ event: KeyboardEvent,
5
+ onActivate: (() => void) | undefined,
6
+ options?: { stopPropagation?: boolean },
7
+ ): void {
8
+ if (!onActivate) {
9
+ return;
10
+ }
11
+
12
+ if (event.key !== "Enter" && event.key !== " ") {
13
+ return;
14
+ }
15
+
16
+ event.preventDefault();
17
+ if (options?.stopPropagation) {
18
+ event.stopPropagation();
19
+ }
20
+ onActivate();
21
+ }
@@ -0,0 +1,66 @@
1
+ export type BoardTargetKind = "edge" | "vertex" | "space" | "tile";
2
+
3
+ export interface InteractiveTargetState {
4
+ kind?: BoardTargetKind;
5
+ id: string;
6
+ eligible: boolean;
7
+ selectable: boolean;
8
+ hovered: boolean;
9
+ pending: boolean;
10
+ conflict: boolean;
11
+ unavailableReason?: string;
12
+ select?: () => unknown | Promise<unknown>;
13
+ }
14
+
15
+ export interface InteractiveTargetLayer {
16
+ enabled?: boolean;
17
+ eligible?: ReadonlySet<string>;
18
+ selectTargetId?: (targetId: string) => unknown | Promise<unknown>;
19
+ targetState?: (targetId: string) => Partial<InteractiveTargetState>;
20
+ }
21
+
22
+ export interface InteractiveTargetRenderState extends InteractiveTargetState {
23
+ isEnabled: boolean;
24
+ isEligible: boolean;
25
+ isHovered: boolean;
26
+ }
27
+
28
+ export function interactiveTargetRenderState(
29
+ layer: InteractiveTargetLayer,
30
+ targetId: string,
31
+ isHovered: boolean,
32
+ ): InteractiveTargetRenderState {
33
+ const enabled = layer.enabled !== false;
34
+ const eligible = layer.eligible?.has(targetId) ?? true;
35
+ const extra = layer.targetState?.(targetId) ?? {};
36
+ const selectable =
37
+ extra.selectable ?? (enabled && eligible && !!layer.selectTargetId);
38
+ return {
39
+ id: targetId,
40
+ ...extra,
41
+ eligible: extra.eligible ?? eligible,
42
+ selectable,
43
+ hovered: isHovered,
44
+ pending: extra.pending ?? false,
45
+ conflict: extra.conflict ?? false,
46
+ select:
47
+ extra.select ??
48
+ (layer.selectTargetId
49
+ ? () => layer.selectTargetId?.(targetId)
50
+ : undefined),
51
+ isEnabled: enabled,
52
+ isEligible: extra.eligible ?? eligible,
53
+ isHovered,
54
+ };
55
+ }
56
+
57
+ export function isInteractiveTargetSelectable(
58
+ layer: InteractiveTargetLayer,
59
+ state: InteractiveTargetRenderState,
60
+ ): boolean {
61
+ return (
62
+ state.isEnabled &&
63
+ state.selectable &&
64
+ !!(state.select ?? layer.selectTargetId)
65
+ );
66
+ }
@@ -0,0 +1,27 @@
1
+ import type { ViewCard } from "../index.js";
2
+ import type { CardFaceProps } from "./Card.js";
3
+
4
+ type ExpectExtends<Actual extends Expected, Expected> = true;
5
+
6
+ type TypedCard = ViewCard<
7
+ "tech-card-1",
8
+ "breakthrough",
9
+ {
10
+ cost: number;
11
+ label: string;
12
+ }
13
+ >;
14
+
15
+ type RenderContentCard = Parameters<
16
+ NonNullable<CardFaceProps<TypedCard>["renderContent"]>
17
+ >[0];
18
+
19
+ type _CardIdIsPreserved = ExpectExtends<RenderContentCard["id"], "tech-card-1">;
20
+ type _CardTypeIsPreserved = ExpectExtends<
21
+ RenderContentCard["cardType"],
22
+ "breakthrough"
23
+ >;
24
+ type _PropertiesArePreserved = ExpectExtends<
25
+ RenderContentCard["properties"]["cost"],
26
+ number
27
+ >;
@@ -0,0 +1,163 @@
1
+ /**
2
+ * Pure layout math for the controlled `HandView`.
3
+ *
4
+ * The functions here are independent of React and DOM measurement so they can
5
+ * be unit-tested directly. {@link computeFanLayout} matches the algorithm
6
+ * documented in `docs/references/ui-sdk-mobile-hand-and-card-interactions.md`,
7
+ * and {@link chooseHandLayoutMode} translates a measured container width and a
8
+ * caller policy into the actual presentation mode the hand should render.
9
+ */
10
+
11
+ export type HandPresentationMode = "fan" | "compressed-fan" | "strip" | "tray";
12
+
13
+ export interface FanLayoutOptions {
14
+ availableWidth: number;
15
+ cardWidth: number;
16
+ cardHeight: number;
17
+ count: number;
18
+ /** Minimum visible slice for any card (touch readability floor). */
19
+ minVisibleSlice: number;
20
+ /** Maximum tilt of the outermost card, in degrees. */
21
+ maxAngle: number;
22
+ /** How many pixels the outermost cards drop below the center card. */
23
+ arcDepth: number;
24
+ }
25
+
26
+ export interface FanCardPosition {
27
+ left: number;
28
+ rotate: number;
29
+ translateY: number;
30
+ zIndex: number;
31
+ }
32
+
33
+ export interface FanLayoutResult {
34
+ positions: FanCardPosition[];
35
+ /** Step between successive card origins (left offset). */
36
+ step: number;
37
+ /** Minimum bounding width of the laid-out hand. */
38
+ totalWidth: number;
39
+ /** Visible portion of each card after overlap, in pixels. */
40
+ visibleSlice: number;
41
+ }
42
+
43
+ export function computeFanLayout(options: FanLayoutOptions): FanLayoutResult {
44
+ const {
45
+ availableWidth,
46
+ cardWidth,
47
+ count,
48
+ minVisibleSlice,
49
+ maxAngle,
50
+ arcDepth,
51
+ } = options;
52
+ if (count <= 0) {
53
+ return { positions: [], step: 0, totalWidth: 0, visibleSlice: cardWidth };
54
+ }
55
+ if (count === 1) {
56
+ return {
57
+ positions: [{ left: 0, rotate: 0, translateY: 0, zIndex: 1 }],
58
+ step: cardWidth,
59
+ totalWidth: cardWidth,
60
+ visibleSlice: cardWidth,
61
+ };
62
+ }
63
+
64
+ const idealStep =
65
+ availableWidth > cardWidth
66
+ ? (availableWidth - cardWidth) / (count - 1)
67
+ : minVisibleSlice;
68
+ const step = Math.max(minVisibleSlice, Math.min(cardWidth, idealStep));
69
+ const center = (count - 1) / 2;
70
+
71
+ const positions: FanCardPosition[] = Array.from(
72
+ { length: count },
73
+ (_, index) => {
74
+ const t = center === 0 ? 0 : (index - center) / center;
75
+ return {
76
+ left: index * step,
77
+ rotate: t * maxAngle,
78
+ translateY: Math.abs(t) * arcDepth,
79
+ zIndex: index + 1,
80
+ };
81
+ },
82
+ );
83
+
84
+ return {
85
+ positions,
86
+ step,
87
+ totalWidth: cardWidth + (count - 1) * step,
88
+ visibleSlice: step,
89
+ };
90
+ }
91
+
92
+ export interface ChooseHandLayoutModeOptions {
93
+ /** Measured container width in CSS pixels. `0` means unmeasured. */
94
+ containerWidth: number;
95
+ cardCount: number;
96
+ cardWidth: number;
97
+ /**
98
+ * Caller-stated layout preference. The pair represents the desktop/mobile
99
+ * intent — the visible-slice gate decides which one renders.
100
+ */
101
+ desktop: HandPresentationMode;
102
+ mobile: HandPresentationMode;
103
+ /** Minimum visible per-card slice that still feels touch-usable. */
104
+ comfortableSlice?: number;
105
+ /** Floor for compressed fan; below this the mobile fallback wins. */
106
+ compressedSlice?: number;
107
+ }
108
+
109
+ /**
110
+ * Choose the actual presentation mode for the current viewport.
111
+ *
112
+ * The decision is target-exposure based, not breakpoint based:
113
+ *
114
+ * - if the desktop preference fits with a comfortable slice, render it as-is;
115
+ * - if it would only barely fit, use a `compressed-fan` when the desktop side
116
+ * is fan-like;
117
+ * - otherwise switch to the mobile fallback (`tray`, `strip`, or whatever the
118
+ * caller asked for) so playable targets stay usable.
119
+ */
120
+ export function chooseHandLayoutMode({
121
+ containerWidth,
122
+ cardCount,
123
+ cardWidth,
124
+ desktop,
125
+ mobile,
126
+ comfortableSlice = HAND_MODE_GEOMETRY.fan.minVisibleSlice,
127
+ compressedSlice = HAND_MODE_GEOMETRY["compressed-fan"].minVisibleSlice,
128
+ }: ChooseHandLayoutModeOptions): HandPresentationMode {
129
+ if (cardCount <= 1) return desktop;
130
+ if (containerWidth <= 0) return desktop;
131
+
132
+ const idealStep = (containerWidth - cardWidth) / (cardCount - 1);
133
+ const step = Math.max(0, Math.min(cardWidth, idealStep));
134
+
135
+ const desktopIsFanLike = desktop === "fan" || desktop === "compressed-fan";
136
+ if (!desktopIsFanLike) {
137
+ if (containerWidth >= cardWidth * cardCount) return desktop;
138
+ return mobile;
139
+ }
140
+
141
+ if (step >= comfortableSlice) {
142
+ return desktop === "compressed-fan" ? "compressed-fan" : "fan";
143
+ }
144
+ if (step >= compressedSlice) {
145
+ return "compressed-fan";
146
+ }
147
+ return mobile;
148
+ }
149
+
150
+ export interface HandModeGeometry {
151
+ minVisibleSlice: number;
152
+ maxAngle: number;
153
+ arcDepth: number;
154
+ }
155
+
156
+ /** Recommended starting values from the mobile-hand reference. */
157
+ export const HAND_MODE_GEOMETRY: Record<
158
+ Exclude<HandPresentationMode, "tray" | "strip">,
159
+ HandModeGeometry
160
+ > = {
161
+ fan: { minVisibleSlice: 64, maxAngle: 5, arcDepth: 12 },
162
+ "compressed-fan": { minVisibleSlice: 44, maxAngle: 4, arcDepth: 8 },
163
+ };
@@ -0,0 +1,413 @@
1
+ /**
2
+ * Deterministic pointer recognizer for the controlled hand view.
3
+ *
4
+ * The engine is independent from React: it consumes pointer events and a
5
+ * caller-supplied callbacks bag, and produces semantic gesture events. It
6
+ * owns no DOM, emits no `CardIntent`, and does not track the drag-lifecycle
7
+ * `phase` — that responsibility lives in `CardDragSurface`. The engine only
8
+ * recognizes `tap`, `previewStart`/`previewEnd`, `liftStart`/`liftMove`/
9
+ * `liftEnd`/`liftCancel`, and a `horizontalBrowse` informational signal.
10
+ *
11
+ * The thresholds match
12
+ * `docs/references/ui-sdk-mobile-hand-and-card-interactions.md`:
13
+ *
14
+ * | Threshold | Default |
15
+ * | ------------------------- | ----------- |
16
+ * | Press preview delay | `220ms` |
17
+ * | Movement slop | `8px` |
18
+ * | Axis bias | `1.25` |
19
+ * | Drag lift distance | `28px` |
20
+ *
21
+ * Callers wire the engine's `handlePointerDown` / `handlePointerMove` /
22
+ * `handlePointerUp` / `handlePointerCancel` to the matching DOM events on
23
+ * each interactive card.
24
+ */
25
+
26
+ export type HandInteractionPolicy = "direct-activate" | "drag-to-target";
27
+
28
+ /**
29
+ * Internal recognition state. Not exposed to consumers because drag-lifecycle
30
+ * phases (`inspecting`/`dragging`/`settling`/`returning`) live in
31
+ * `CardDragSurface`.
32
+ */
33
+ export type CardPointerState =
34
+ | { kind: "idle" }
35
+ | { kind: "pressing"; pointerId: number; cardId: string; startedAt: number }
36
+ | { kind: "preview"; pointerId: number; cardId: string }
37
+ | { kind: "horizontalBrowse"; pointerId: number; cardId: string }
38
+ | { kind: "lifted"; pointerId: number; cardId: string };
39
+
40
+ export interface PointerEngineThresholds {
41
+ pressPreviewMs: number;
42
+ movementSlopPx: number;
43
+ axisBiasRatio: number;
44
+ dragLiftDistancePx: number;
45
+ }
46
+
47
+ export const DEFAULT_POINTER_THRESHOLDS: PointerEngineThresholds = {
48
+ pressPreviewMs: 220,
49
+ movementSlopPx: 8,
50
+ axisBiasRatio: 1.25,
51
+ dragLiftDistancePx: 28,
52
+ };
53
+
54
+ export type AxisDecision = "undecided" | "horizontal" | "upward";
55
+
56
+ export function resolveAxis(
57
+ dx: number,
58
+ dy: number,
59
+ thresholds: PointerEngineThresholds = DEFAULT_POINTER_THRESHOLDS,
60
+ ): AxisDecision {
61
+ if (
62
+ Math.abs(dx) < thresholds.movementSlopPx &&
63
+ Math.abs(dy) < thresholds.movementSlopPx
64
+ ) {
65
+ return "undecided";
66
+ }
67
+ if (Math.abs(dx) > Math.abs(dy) * thresholds.axisBiasRatio)
68
+ return "horizontal";
69
+ if (-dy > Math.abs(dx) * thresholds.axisBiasRatio) return "upward";
70
+ return "undecided";
71
+ }
72
+
73
+ export interface PointerEngineCardSnapshot {
74
+ cardId: string;
75
+ /** Whether the runtime adapter says this card is interactive. */
76
+ eligible: boolean;
77
+ /** Whether the runtime adapter says this card is disabled. */
78
+ disabled: boolean;
79
+ }
80
+
81
+ export interface LiftStartEvent {
82
+ cardId: string;
83
+ pointerId: number;
84
+ startX: number;
85
+ startY: number;
86
+ pointerX: number;
87
+ pointerY: number;
88
+ grabOffsetX: number;
89
+ grabOffsetY: number;
90
+ }
91
+
92
+ export interface LiftMoveEvent {
93
+ cardId: string;
94
+ pointerId: number;
95
+ pointerX: number;
96
+ pointerY: number;
97
+ }
98
+
99
+ export interface LiftEndEvent {
100
+ cardId: string;
101
+ pointerId: number;
102
+ pointerX: number;
103
+ pointerY: number;
104
+ }
105
+
106
+ export interface PointerEngineCallbacks {
107
+ /**
108
+ * Recognized clean tap (no movement, no preview). Source is `pointer` for
109
+ * pointerup taps; the keyboard activation path is a separate caller method
110
+ * (see `triggerKeyboardActivate`/`triggerKeyboardLift`).
111
+ */
112
+ onTap?: (cardId: string) => void;
113
+ /** Long-press preview started. */
114
+ onPreviewStart?: (cardId: string) => void;
115
+ /** Preview ended (release or axis disqualification). */
116
+ onPreviewEnd?: (cardId: string) => void;
117
+ /** Vertical lift gesture began past the drag threshold. */
118
+ onLiftStart?: (event: LiftStartEvent) => void;
119
+ /** Pointer move while lifted. */
120
+ onLiftMove?: (event: LiftMoveEvent) => void;
121
+ /** Pointer release while lifted. */
122
+ onLiftEnd?: (event: LiftEndEvent) => void;
123
+ /** Lift was cancelled (pointer cancel/lost capture). */
124
+ onLiftCancel?: (cardId: string) => void;
125
+ /** Horizontal browse arbitration (informational). */
126
+ onHorizontalBrowse?: (cardId: string) => void;
127
+ /**
128
+ * Called when the engine has captured the pointer and wants the host to
129
+ * suppress scrolling. The host controls the `touch-action` value.
130
+ */
131
+ onLockScroll?: (locked: boolean) => void;
132
+ /** Called whenever the recognition state transitions, for visual hints. */
133
+ onStateChange?: (next: CardPointerState) => void;
134
+ }
135
+
136
+ interface ActivePointer {
137
+ cardId: string;
138
+ pointerId: number;
139
+ startedAt: number;
140
+ startX: number;
141
+ startY: number;
142
+ lastX: number;
143
+ lastY: number;
144
+ axis: AxisDecision;
145
+ /** Set when the long-press preview timer fires. */
146
+ previewActive: boolean;
147
+ }
148
+
149
+ export class HandPointerEngine {
150
+ private readonly callbacks: PointerEngineCallbacks;
151
+ private readonly thresholds: PointerEngineThresholds;
152
+ private active: ActivePointer | null = null;
153
+ private state: CardPointerState = { kind: "idle" };
154
+ private previewTimer: ReturnType<typeof setTimeout> | null = null;
155
+ private now: () => number;
156
+
157
+ constructor(
158
+ callbacks: PointerEngineCallbacks,
159
+ thresholds: PointerEngineThresholds = DEFAULT_POINTER_THRESHOLDS,
160
+ now: () => number = () => Date.now(),
161
+ ) {
162
+ this.callbacks = callbacks;
163
+ this.thresholds = thresholds;
164
+ this.now = now;
165
+ }
166
+
167
+ /** Returns the most recently emitted recognition state (test helper). */
168
+ getState(): CardPointerState {
169
+ return this.state;
170
+ }
171
+
172
+ dispose(): void {
173
+ this.clearPreviewTimer();
174
+ if (this.state.kind === "lifted") {
175
+ this.callbacks.onLiftCancel?.(this.state.cardId);
176
+ }
177
+ this.active = null;
178
+ this.transition({ kind: "idle" });
179
+ this.callbacks.onLockScroll?.(false);
180
+ }
181
+
182
+ handlePointerDown(
183
+ card: PointerEngineCardSnapshot,
184
+ event: { pointerId: number; clientX: number; clientY: number },
185
+ ): void {
186
+ if (this.active) return;
187
+ if (card.disabled) return;
188
+
189
+ const startedAt = this.now();
190
+ this.active = {
191
+ cardId: card.cardId,
192
+ pointerId: event.pointerId,
193
+ startedAt,
194
+ startX: event.clientX,
195
+ startY: event.clientY,
196
+ lastX: event.clientX,
197
+ lastY: event.clientY,
198
+ axis: "undecided",
199
+ previewActive: false,
200
+ };
201
+ this.transition({
202
+ kind: "pressing",
203
+ pointerId: event.pointerId,
204
+ cardId: card.cardId,
205
+ startedAt,
206
+ });
207
+ this.schedulePreview(card);
208
+ }
209
+
210
+ handlePointerMove(
211
+ card: PointerEngineCardSnapshot,
212
+ event: { pointerId: number; clientX: number; clientY: number },
213
+ ): void {
214
+ const pointer = this.active;
215
+ if (!pointer || pointer.pointerId !== event.pointerId) return;
216
+ if (pointer.cardId !== card.cardId) return;
217
+
218
+ pointer.lastX = event.clientX;
219
+ pointer.lastY = event.clientY;
220
+
221
+ const dx = event.clientX - pointer.startX;
222
+ const dy = event.clientY - pointer.startY;
223
+ const axis = resolveAxis(dx, dy, this.thresholds);
224
+ if (axis !== "undecided") pointer.axis = axis;
225
+
226
+ if (this.state.kind === "lifted") {
227
+ this.callbacks.onLiftMove?.({
228
+ cardId: pointer.cardId,
229
+ pointerId: pointer.pointerId,
230
+ pointerX: event.clientX,
231
+ pointerY: event.clientY,
232
+ });
233
+ return;
234
+ }
235
+
236
+ if (this.state.kind === "pressing") {
237
+ if (axis === "horizontal") {
238
+ this.clearPreviewTimer();
239
+ this.transition({
240
+ kind: "horizontalBrowse",
241
+ pointerId: pointer.pointerId,
242
+ cardId: pointer.cardId,
243
+ });
244
+ this.callbacks.onHorizontalBrowse?.(pointer.cardId);
245
+ return;
246
+ }
247
+ if (
248
+ axis === "upward" &&
249
+ card.eligible &&
250
+ -dy >= this.thresholds.dragLiftDistancePx
251
+ ) {
252
+ this.clearPreviewTimer();
253
+ this.beginLift(pointer, event.clientX, event.clientY);
254
+ return;
255
+ }
256
+ return;
257
+ }
258
+
259
+ if (this.state.kind === "preview") {
260
+ if (
261
+ axis === "upward" &&
262
+ card.eligible &&
263
+ -dy >= this.thresholds.dragLiftDistancePx
264
+ ) {
265
+ this.callbacks.onPreviewEnd?.(card.cardId);
266
+ pointer.previewActive = false;
267
+ this.beginLift(pointer, event.clientX, event.clientY);
268
+ return;
269
+ }
270
+ if (axis === "horizontal") {
271
+ this.callbacks.onPreviewEnd?.(card.cardId);
272
+ pointer.previewActive = false;
273
+ this.transition({
274
+ kind: "horizontalBrowse",
275
+ pointerId: pointer.pointerId,
276
+ cardId: pointer.cardId,
277
+ });
278
+ this.callbacks.onHorizontalBrowse?.(pointer.cardId);
279
+ return;
280
+ }
281
+ return;
282
+ }
283
+
284
+ // horizontalBrowse — remain locked until release.
285
+ }
286
+
287
+ handlePointerUp(
288
+ card: PointerEngineCardSnapshot,
289
+ event: { pointerId: number; clientX: number; clientY: number },
290
+ ): void {
291
+ const pointer = this.active;
292
+ if (!pointer || pointer.pointerId !== event.pointerId) return;
293
+ if (pointer.cardId !== card.cardId) return;
294
+ this.clearPreviewTimer();
295
+
296
+ const dx = event.clientX - pointer.startX;
297
+ const dy = event.clientY - pointer.startY;
298
+ const isTap =
299
+ pointer.axis === "undecided" &&
300
+ !pointer.previewActive &&
301
+ Math.abs(dx) < this.thresholds.movementSlopPx &&
302
+ Math.abs(dy) < this.thresholds.movementSlopPx;
303
+
304
+ if (this.state.kind === "lifted") {
305
+ this.callbacks.onLiftEnd?.({
306
+ cardId: pointer.cardId,
307
+ pointerId: pointer.pointerId,
308
+ pointerX: event.clientX,
309
+ pointerY: event.clientY,
310
+ });
311
+ this.reset();
312
+ return;
313
+ }
314
+
315
+ if (isTap) {
316
+ if (card.eligible && !card.disabled) {
317
+ this.callbacks.onTap?.(card.cardId);
318
+ }
319
+ this.reset();
320
+ return;
321
+ }
322
+
323
+ if (pointer.previewActive) {
324
+ this.callbacks.onPreviewEnd?.(card.cardId);
325
+ }
326
+ this.reset();
327
+ }
328
+
329
+ handlePointerCancel(card: PointerEngineCardSnapshot): void {
330
+ const pointer = this.active;
331
+ if (!pointer || pointer.cardId !== card.cardId) return;
332
+ this.clearPreviewTimer();
333
+ if (this.state.kind === "lifted") {
334
+ this.callbacks.onLiftCancel?.(card.cardId);
335
+ this.reset();
336
+ return;
337
+ }
338
+ if (pointer.previewActive) {
339
+ this.callbacks.onPreviewEnd?.(card.cardId);
340
+ }
341
+ this.reset();
342
+ }
343
+
344
+ /**
345
+ * Programmatic reset of recognition state (e.g. after the surface commits
346
+ * a keyboard lift externally). Does not emit callbacks.
347
+ */
348
+ resetRecognition(): void {
349
+ this.clearPreviewTimer();
350
+ this.active = null;
351
+ this.callbacks.onLockScroll?.(false);
352
+ this.transition({ kind: "idle" });
353
+ }
354
+
355
+ private beginLift(
356
+ pointer: ActivePointer,
357
+ clientX: number,
358
+ clientY: number,
359
+ ): void {
360
+ this.callbacks.onLockScroll?.(true);
361
+ const grabOffsetX = clientX - pointer.startX;
362
+ const grabOffsetY = clientY - pointer.startY;
363
+ this.transition({
364
+ kind: "lifted",
365
+ pointerId: pointer.pointerId,
366
+ cardId: pointer.cardId,
367
+ });
368
+ this.callbacks.onLiftStart?.({
369
+ cardId: pointer.cardId,
370
+ pointerId: pointer.pointerId,
371
+ startX: pointer.startX,
372
+ startY: pointer.startY,
373
+ pointerX: clientX,
374
+ pointerY: clientY,
375
+ grabOffsetX,
376
+ grabOffsetY,
377
+ });
378
+ }
379
+
380
+ private schedulePreview(card: PointerEngineCardSnapshot): void {
381
+ this.clearPreviewTimer();
382
+ this.previewTimer = setTimeout(() => {
383
+ const pointer = this.active;
384
+ if (!pointer || pointer.cardId !== card.cardId) return;
385
+ if (pointer.axis !== "undecided") return;
386
+ pointer.previewActive = true;
387
+ this.callbacks.onPreviewStart?.(card.cardId);
388
+ this.transition({
389
+ kind: "preview",
390
+ pointerId: pointer.pointerId,
391
+ cardId: card.cardId,
392
+ });
393
+ }, this.thresholds.pressPreviewMs);
394
+ }
395
+
396
+ private clearPreviewTimer(): void {
397
+ if (this.previewTimer !== null) {
398
+ clearTimeout(this.previewTimer);
399
+ this.previewTimer = null;
400
+ }
401
+ }
402
+
403
+ private reset(): void {
404
+ this.active = null;
405
+ this.callbacks.onLockScroll?.(false);
406
+ this.transition({ kind: "idle" });
407
+ }
408
+
409
+ private transition(next: CardPointerState): void {
410
+ this.state = next;
411
+ this.callbacks.onStateChange?.(next);
412
+ }
413
+ }