@pie-lib/graphing 4.0.5-next.31 → 4.0.5-next.34

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 (453) hide show
  1. package/CHANGELOG.json +1 -0
  2. package/CHANGELOG.md +1469 -0
  3. package/LICENSE.md +5 -0
  4. package/lib/axis/arrow.js +79 -0
  5. package/lib/axis/arrow.js.map +1 -0
  6. package/lib/axis/axes.js +329 -0
  7. package/lib/axis/axes.js.map +1 -0
  8. package/lib/axis/index.js +21 -0
  9. package/lib/axis/index.js.map +1 -0
  10. package/lib/bg.js +114 -0
  11. package/lib/bg.js.map +1 -0
  12. package/lib/container/actions.js +18 -0
  13. package/lib/container/actions.js.map +1 -0
  14. package/lib/container/index.js +127 -0
  15. package/lib/container/index.js.map +1 -0
  16. package/lib/container/marks.js +22 -0
  17. package/lib/container/marks.js.map +1 -0
  18. package/lib/container/middleware.js +19 -0
  19. package/lib/container/middleware.js.map +1 -0
  20. package/lib/container/reducer.js +18 -0
  21. package/lib/container/reducer.js.map +1 -0
  22. package/lib/coordinates-label.js +79 -0
  23. package/lib/coordinates-label.js.map +1 -0
  24. package/lib/graph-with-controls.js +294 -0
  25. package/lib/graph-with-controls.js.map +1 -0
  26. package/lib/graph.js +327 -0
  27. package/lib/graph.js.map +1 -0
  28. package/lib/grid-setup.js +414 -0
  29. package/lib/grid-setup.js.map +1 -0
  30. package/lib/grid.js +131 -0
  31. package/lib/grid.js.map +1 -0
  32. package/lib/index.js +47 -0
  33. package/lib/index.js.map +1 -0
  34. package/lib/key-legend.js +201 -0
  35. package/lib/key-legend.js.map +1 -0
  36. package/lib/label-svg-icon.js +50 -0
  37. package/lib/label-svg-icon.js.map +1 -0
  38. package/lib/labels.js +243 -0
  39. package/lib/labels.js.map +1 -0
  40. package/lib/mark-label.js +285 -0
  41. package/lib/mark-label.js.map +1 -0
  42. package/lib/toggle-bar.js +252 -0
  43. package/lib/toggle-bar.js.map +1 -0
  44. package/lib/tool-menu.js +78 -0
  45. package/lib/tool-menu.js.map +1 -0
  46. package/lib/tools/absolute/component.js +29 -0
  47. package/lib/tools/absolute/component.js.map +1 -0
  48. package/lib/tools/absolute/index.js +50 -0
  49. package/lib/tools/absolute/index.js.map +1 -0
  50. package/lib/tools/circle/bg-circle.js +123 -0
  51. package/lib/tools/circle/bg-circle.js.map +1 -0
  52. package/lib/tools/circle/component.js +321 -0
  53. package/lib/tools/circle/component.js.map +1 -0
  54. package/lib/tools/circle/index.js +41 -0
  55. package/lib/tools/circle/index.js.map +1 -0
  56. package/lib/tools/exponential/component.js +28 -0
  57. package/lib/tools/exponential/component.js.map +1 -0
  58. package/lib/tools/exponential/index.js +56 -0
  59. package/lib/tools/exponential/index.js.map +1 -0
  60. package/lib/tools/index.js +86 -0
  61. package/lib/tools/index.js.map +1 -0
  62. package/lib/tools/line/component.js +97 -0
  63. package/lib/tools/line/component.js.map +1 -0
  64. package/lib/tools/line/index.js +11 -0
  65. package/lib/tools/line/index.js.map +1 -0
  66. package/lib/tools/parabola/component.js +28 -0
  67. package/lib/tools/parabola/component.js.map +1 -0
  68. package/lib/tools/parabola/index.js +50 -0
  69. package/lib/tools/parabola/index.js.map +1 -0
  70. package/lib/tools/point/component.js +148 -0
  71. package/lib/tools/point/component.js.map +1 -0
  72. package/lib/tools/point/index.js +24 -0
  73. package/lib/tools/point/index.js.map +1 -0
  74. package/lib/tools/polygon/component.js +437 -0
  75. package/lib/tools/polygon/component.js.map +1 -0
  76. package/lib/tools/polygon/index.js +89 -0
  77. package/lib/tools/polygon/index.js.map +1 -0
  78. package/lib/tools/polygon/line.js +112 -0
  79. package/lib/tools/polygon/line.js.map +1 -0
  80. package/lib/tools/polygon/polygon.js +130 -0
  81. package/lib/tools/polygon/polygon.js.map +1 -0
  82. package/lib/tools/ray/component.js +95 -0
  83. package/lib/tools/ray/component.js.map +1 -0
  84. package/lib/tools/ray/index.js +11 -0
  85. package/lib/tools/ray/index.js.map +1 -0
  86. package/lib/tools/segment/component.js +71 -0
  87. package/lib/tools/segment/component.js.map +1 -0
  88. package/lib/tools/segment/index.js +11 -0
  89. package/lib/tools/segment/index.js.map +1 -0
  90. package/lib/tools/shared/arrow-head.js +111 -0
  91. package/lib/tools/shared/arrow-head.js.map +1 -0
  92. package/lib/tools/shared/icons/CorrectSVG.js +40 -0
  93. package/lib/tools/shared/icons/CorrectSVG.js.map +1 -0
  94. package/lib/tools/shared/icons/IncorrectSVG.js +40 -0
  95. package/lib/tools/shared/icons/IncorrectSVG.js.map +1 -0
  96. package/lib/tools/shared/icons/MissingSVG.js +39 -0
  97. package/lib/tools/shared/icons/MissingSVG.js.map +1 -0
  98. package/lib/tools/shared/line/index.js +550 -0
  99. package/lib/tools/shared/line/index.js.map +1 -0
  100. package/lib/tools/shared/line/line-path.js +118 -0
  101. package/lib/tools/shared/line/line-path.js.map +1 -0
  102. package/lib/tools/shared/line/with-root-edge.js +121 -0
  103. package/lib/tools/shared/line/with-root-edge.js.map +1 -0
  104. package/lib/tools/shared/point/arrow-point.js +72 -0
  105. package/lib/tools/shared/point/arrow-point.js.map +1 -0
  106. package/lib/tools/shared/point/arrow.js +67 -0
  107. package/lib/tools/shared/point/arrow.js.map +1 -0
  108. package/lib/tools/shared/point/base-point.js +157 -0
  109. package/lib/tools/shared/point/base-point.js.map +1 -0
  110. package/lib/tools/shared/point/index.js +68 -0
  111. package/lib/tools/shared/point/index.js.map +1 -0
  112. package/lib/tools/shared/styles.js +33 -0
  113. package/lib/tools/shared/styles.js.map +1 -0
  114. package/lib/tools/shared/types.js +16 -0
  115. package/lib/tools/shared/types.js.map +1 -0
  116. package/lib/tools/sine/component.js +40 -0
  117. package/lib/tools/sine/component.js.map +1 -0
  118. package/lib/tools/sine/index.js +50 -0
  119. package/lib/tools/sine/index.js.map +1 -0
  120. package/lib/tools/vector/component.js +68 -0
  121. package/lib/tools/vector/component.js.map +1 -0
  122. package/lib/tools/vector/index.js +11 -0
  123. package/lib/tools/vector/index.js.map +1 -0
  124. package/lib/undo-redo.js +86 -0
  125. package/lib/undo-redo.js.map +1 -0
  126. package/lib/use-debounce.js +25 -0
  127. package/lib/use-debounce.js.map +1 -0
  128. package/lib/utils.js +229 -0
  129. package/lib/utils.js.map +1 -0
  130. package/package.json +33 -45
  131. package/src/__tests__/bg.test.jsx +250 -0
  132. package/src/__tests__/coordinates-label.test.jsx +243 -0
  133. package/src/__tests__/graph-with-controls.test.jsx +198 -0
  134. package/src/__tests__/graph.test.jsx +721 -0
  135. package/src/__tests__/grid-setup.test.jsx +645 -0
  136. package/src/__tests__/grid.test.jsx +22 -0
  137. package/src/__tests__/key-legend.test.jsx +260 -0
  138. package/src/__tests__/label-svg-icon.test.jsx +278 -0
  139. package/src/__tests__/labels.test.jsx +55 -0
  140. package/src/__tests__/mark-label.test.jsx +63 -0
  141. package/src/__tests__/toggle-bar.test.jsx +146 -0
  142. package/src/__tests__/tool-menu.test.jsx +115 -0
  143. package/src/__tests__/undo-redo.test.jsx +24 -0
  144. package/src/__tests__/use-debounce.test.js +21 -0
  145. package/src/__tests__/utils.js +41 -0
  146. package/src/__tests__/utils.test.js +105 -0
  147. package/src/axis/__tests__/arrow.test.jsx +38 -0
  148. package/src/axis/__tests__/axes.test.jsx +216 -0
  149. package/src/axis/arrow.jsx +57 -0
  150. package/src/axis/axes.jsx +285 -0
  151. package/src/axis/index.js +3 -0
  152. package/src/bg.jsx +96 -0
  153. package/src/container/__tests__/actions.test.js +105 -0
  154. package/src/container/__tests__/index.test.jsx +319 -0
  155. package/src/container/__tests__/marks.test.js +172 -0
  156. package/src/container/__tests__/middleware.test.js +235 -0
  157. package/src/container/__tests__/reducer.test.js +324 -0
  158. package/src/container/actions.js +8 -0
  159. package/src/container/index.jsx +91 -0
  160. package/src/container/marks.js +14 -0
  161. package/src/container/middleware.js +7 -0
  162. package/src/container/reducer.js +5 -0
  163. package/src/coordinates-label.jsx +63 -0
  164. package/src/graph-with-controls.jsx +239 -0
  165. package/src/graph.jsx +303 -0
  166. package/src/grid-setup.jsx +432 -0
  167. package/src/grid.jsx +133 -0
  168. package/src/index.js +7 -0
  169. package/src/key-legend.jsx +142 -0
  170. package/src/label-svg-icon.jsx +39 -0
  171. package/src/labels.jsx +207 -0
  172. package/src/mark-label.jsx +244 -0
  173. package/src/toggle-bar.jsx +224 -0
  174. package/src/tool-menu.jsx +49 -0
  175. package/src/tools/absolute/__tests__/component.test.jsx +53 -0
  176. package/src/tools/absolute/component.jsx +23 -0
  177. package/src/tools/absolute/index.js +31 -0
  178. package/src/tools/circle/__tests__/bg-circle.test.jsx +26 -0
  179. package/src/tools/circle/__tests__/component.test.jsx +494 -0
  180. package/src/tools/circle/__tests__/index.test.js +480 -0
  181. package/src/tools/circle/bg-circle.jsx +81 -0
  182. package/src/tools/circle/component.jsx +264 -0
  183. package/src/tools/circle/index.js +25 -0
  184. package/src/tools/exponential/__tests__/component.test.jsx +53 -0
  185. package/src/tools/exponential/__tests__/index.test.js +729 -0
  186. package/src/tools/exponential/component.jsx +23 -0
  187. package/src/tools/exponential/index.js +39 -0
  188. package/src/tools/index.js +48 -0
  189. package/src/tools/line/__tests__/component.test.jsx +37 -0
  190. package/src/tools/line/component.jsx +93 -0
  191. package/src/tools/line/index.js +4 -0
  192. package/src/tools/parabola/__tests__/component.test.jsx +48 -0
  193. package/src/tools/parabola/__tests__/index.test.js +470 -0
  194. package/src/tools/parabola/component.jsx +23 -0
  195. package/src/tools/parabola/index.js +31 -0
  196. package/src/tools/point/__tests__/component.test.jsx +349 -0
  197. package/src/tools/point/__tests__/index.test.js +241 -0
  198. package/src/tools/point/component.jsx +126 -0
  199. package/src/tools/point/index.js +11 -0
  200. package/src/tools/polygon/__tests__/component.test.jsx +471 -0
  201. package/src/tools/polygon/__tests__/index.test.js +294 -0
  202. package/src/tools/polygon/__tests__/line.test.jsx +35 -0
  203. package/src/tools/polygon/__tests__/polygon.test.jsx +61 -0
  204. package/src/tools/polygon/component.jsx +409 -0
  205. package/src/tools/polygon/index.js +52 -0
  206. package/src/tools/polygon/line.jsx +74 -0
  207. package/src/tools/polygon/polygon.jsx +110 -0
  208. package/src/tools/ray/__tests__/component.test.jsx +29 -0
  209. package/src/tools/ray/component.jsx +92 -0
  210. package/src/tools/ray/index.js +4 -0
  211. package/src/tools/segment/__tests__/component.test.jsx +28 -0
  212. package/src/tools/segment/component.jsx +65 -0
  213. package/src/tools/segment/index.js +4 -0
  214. package/src/tools/shared/__tests__/arrow-head.test.jsx +31 -0
  215. package/src/tools/shared/arrow-head.jsx +102 -0
  216. package/src/tools/shared/icons/CorrectSVG.jsx +32 -0
  217. package/src/tools/shared/icons/IncorrectSVG.jsx +30 -0
  218. package/src/tools/shared/icons/MissingSVG.jsx +31 -0
  219. package/src/tools/shared/line/__tests__/index.test.jsx +109 -0
  220. package/src/tools/shared/line/__tests__/line-path.test.jsx +53 -0
  221. package/src/tools/shared/line/__tests__/with-root-edge.test.jsx +73 -0
  222. package/src/tools/shared/line/index.jsx +487 -0
  223. package/src/tools/shared/line/line-path.jsx +80 -0
  224. package/src/tools/shared/line/with-root-edge.jsx +97 -0
  225. package/src/tools/shared/point/__tests__/arrow-point.test.jsx +91 -0
  226. package/src/tools/shared/point/__tests__/base-point.test.jsx +87 -0
  227. package/src/tools/shared/point/arrow-point.jsx +46 -0
  228. package/src/tools/shared/point/arrow.jsx +37 -0
  229. package/src/tools/shared/point/base-point.jsx +121 -0
  230. package/src/tools/shared/point/index.jsx +54 -0
  231. package/src/tools/shared/styles.js +27 -0
  232. package/src/tools/shared/types.js +10 -0
  233. package/src/tools/sine/__tests__/component.test.jsx +51 -0
  234. package/src/tools/sine/component.jsx +32 -0
  235. package/src/tools/sine/index.js +33 -0
  236. package/src/tools/vector/__tests__/component.test.jsx +25 -0
  237. package/src/tools/vector/component.jsx +56 -0
  238. package/src/tools/vector/index.js +4 -0
  239. package/src/undo-redo.jsx +45 -0
  240. package/src/use-debounce.js +13 -0
  241. package/src/utils.js +224 -0
  242. package/dist/_virtual/_rolldown/runtime.js +0 -23
  243. package/dist/autosize-input.d.ts +0 -10
  244. package/dist/autosize-input.js +0 -66
  245. package/dist/axis/arrow.d.ts +0 -13
  246. package/dist/axis/arrow.js +0 -34
  247. package/dist/axis/axes.d.ts +0 -132
  248. package/dist/axis/axes.js +0 -214
  249. package/dist/axis/index.d.ts +0 -10
  250. package/dist/bg.d.ts +0 -51
  251. package/dist/bg.js +0 -44
  252. package/dist/container/actions.d.ts +0 -15
  253. package/dist/container/actions.js +0 -7
  254. package/dist/container/index.d.ts +0 -58
  255. package/dist/container/index.js +0 -48
  256. package/dist/container/marks.d.ts +0 -10
  257. package/dist/container/marks.js +0 -11
  258. package/dist/container/middleware.d.ts +0 -10
  259. package/dist/container/middleware.js +0 -4
  260. package/dist/container/reducer.d.ts +0 -14
  261. package/dist/container/reducer.js +0 -7
  262. package/dist/coordinates-label.d.ts +0 -50
  263. package/dist/coordinates-label.js +0 -46
  264. package/dist/graph-with-controls.d.ts +0 -88
  265. package/dist/graph-with-controls.js +0 -154
  266. package/dist/graph.d.ts +0 -126
  267. package/dist/graph.js +0 -209
  268. package/dist/grid-setup.d.ts +0 -27
  269. package/dist/grid-setup.js +0 -307
  270. package/dist/grid.d.ts +0 -43
  271. package/dist/grid.js +0 -59
  272. package/dist/index.d.ts +0 -15
  273. package/dist/index.js +0 -7
  274. package/dist/key-legend.d.ts +0 -21
  275. package/dist/key-legend.js +0 -231
  276. package/dist/label-svg-icon.d.ts +0 -18
  277. package/dist/label-svg-icon.js +0 -42
  278. package/dist/labels.d.ts +0 -37
  279. package/dist/labels.js +0 -152
  280. package/dist/mark-label.d.ts +0 -58
  281. package/dist/mark-label.js +0 -171
  282. package/dist/node_modules/.bun/@visx_axis@3.12.0_f4eacebf2041cd4f/node_modules/@visx/axis/esm/axis/Axis.js +0 -101
  283. package/dist/node_modules/.bun/@visx_axis@3.12.0_f4eacebf2041cd4f/node_modules/@visx/axis/esm/axis/AxisRenderer.js +0 -63
  284. package/dist/node_modules/.bun/@visx_axis@3.12.0_f4eacebf2041cd4f/node_modules/@visx/axis/esm/axis/Ticks.js +0 -44
  285. package/dist/node_modules/.bun/@visx_axis@3.12.0_f4eacebf2041cd4f/node_modules/@visx/axis/esm/constants/orientation.js +0 -9
  286. package/dist/node_modules/.bun/@visx_axis@3.12.0_f4eacebf2041cd4f/node_modules/@visx/axis/esm/utils/createPoint.js +0 -14
  287. package/dist/node_modules/.bun/@visx_axis@3.12.0_f4eacebf2041cd4f/node_modules/@visx/axis/esm/utils/getAxisRangePaddingConfig.js +0 -21
  288. package/dist/node_modules/.bun/@visx_axis@3.12.0_f4eacebf2041cd4f/node_modules/@visx/axis/esm/utils/getLabelTransform.js +0 -16
  289. package/dist/node_modules/.bun/@visx_axis@3.12.0_f4eacebf2041cd4f/node_modules/@visx/axis/esm/utils/getTickFormatter.js +0 -8
  290. package/dist/node_modules/.bun/@visx_axis@3.12.0_f4eacebf2041cd4f/node_modules/@visx/axis/esm/utils/getTickPosition.js +0 -15
  291. package/dist/node_modules/.bun/@visx_curve@3.12.0/node_modules/@visx/curve/esm/index.js +0 -2
  292. package/dist/node_modules/.bun/@visx_grid@3.12.0_f4eacebf2041cd4f/node_modules/@visx/grid/esm/grids/Grid.js +0 -79
  293. package/dist/node_modules/.bun/@visx_grid@3.12.0_f4eacebf2041cd4f/node_modules/@visx/grid/esm/grids/GridColumns.js +0 -79
  294. package/dist/node_modules/.bun/@visx_grid@3.12.0_f4eacebf2041cd4f/node_modules/@visx/grid/esm/grids/GridRows.js +0 -79
  295. package/dist/node_modules/.bun/@visx_grid@3.12.0_f4eacebf2041cd4f/node_modules/@visx/grid/esm/utils/getScaleBandwidth.js +0 -6
  296. package/dist/node_modules/.bun/@visx_group@3.12.0_f4eacebf2041cd4f/node_modules/@visx/group/esm/Group.js +0 -50
  297. package/dist/node_modules/.bun/@visx_point@3.12.0/node_modules/@visx/point/esm/Point.js +0 -18
  298. package/dist/node_modules/.bun/@visx_scale@3.12.0/node_modules/@visx/scale/esm/utils/coerceNumber.js +0 -10
  299. package/dist/node_modules/.bun/@visx_scale@3.12.0/node_modules/@visx/scale/esm/utils/getTicks.js +0 -9
  300. package/dist/node_modules/.bun/@visx_scale@3.12.0/node_modules/@visx/scale/esm/utils/toString.js +0 -6
  301. package/dist/node_modules/.bun/@visx_shape@3.12.0_f4eacebf2041cd4f/node_modules/@visx/shape/esm/shapes/Line.js +0 -47
  302. package/dist/node_modules/.bun/@visx_shape@3.12.0_f4eacebf2041cd4f/node_modules/@visx/shape/esm/shapes/LinePath.js +0 -50
  303. package/dist/node_modules/.bun/@visx_shape@3.12.0_f4eacebf2041cd4f/node_modules/@visx/shape/esm/util/D3ShapeFactories.js +0 -9
  304. package/dist/node_modules/.bun/@visx_shape@3.12.0_f4eacebf2041cd4f/node_modules/@visx/shape/esm/util/setNumberOrNumberAccessor.js +0 -6
  305. package/dist/node_modules/.bun/@visx_shape@3.12.0_f4eacebf2041cd4f/node_modules/@visx/shape/lib/shapes/Line.js +0 -53
  306. package/dist/node_modules/.bun/@visx_text@3.12.0_f4eacebf2041cd4f/node_modules/@visx/text/esm/Text.js +0 -57
  307. package/dist/node_modules/.bun/@visx_text@3.12.0_f4eacebf2041cd4f/node_modules/@visx/text/esm/hooks/useText.js +0 -91
  308. package/dist/node_modules/.bun/@visx_text@3.12.0_f4eacebf2041cd4f/node_modules/@visx/text/esm/util/getStringWidth.js +0 -21
  309. package/dist/node_modules/.bun/balanced-match@0.4.2/node_modules/balanced-match/index.js +0 -32
  310. package/dist/node_modules/.bun/balanced-match@1.0.2/node_modules/balanced-match/index.js +0 -33
  311. package/dist/node_modules/.bun/classnames@2.5.1/node_modules/classnames/index.js +0 -32
  312. package/dist/node_modules/.bun/clsx@2.1.1/node_modules/clsx/dist/clsx.js +0 -16
  313. package/dist/node_modules/.bun/invariant@2.2.4/node_modules/invariant/browser.js +0 -28
  314. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_Hash.js +0 -21
  315. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_ListCache.js +0 -21
  316. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_Map.js +0 -10
  317. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_MapCache.js +0 -21
  318. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_Symbol.js +0 -9
  319. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_assocIndexOf.js +0 -14
  320. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_baseGetTag.js +0 -15
  321. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_baseIsNative.js +0 -16
  322. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_coreJsData.js +0 -9
  323. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_freeGlobal.js +0 -8
  324. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_getMapData.js +0 -14
  325. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_getNative.js +0 -15
  326. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_getRawTag.js +0 -19
  327. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_getValue.js +0 -11
  328. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_hashClear.js +0 -13
  329. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_hashDelete.js +0 -12
  330. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_hashGet.js +0 -18
  331. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_hashHas.js +0 -14
  332. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_hashSet.js +0 -14
  333. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_isKeyable.js +0 -12
  334. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_isMasked.js +0 -16
  335. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_listCacheClear.js +0 -11
  336. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_listCacheDelete.js +0 -14
  337. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_listCacheGet.js +0 -14
  338. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_listCacheHas.js +0 -13
  339. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_listCacheSet.js +0 -14
  340. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_mapCacheClear.js +0 -19
  341. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_mapCacheDelete.js +0 -14
  342. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_mapCacheGet.js +0 -13
  343. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_mapCacheHas.js +0 -13
  344. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_mapCacheSet.js +0 -14
  345. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_nativeCreate.js +0 -9
  346. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_objectToString.js +0 -12
  347. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_root.js +0 -10
  348. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/_toSource.js +0 -20
  349. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/eq.js +0 -11
  350. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/isFunction.js +0 -16
  351. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/isObject.js +0 -12
  352. package/dist/node_modules/.bun/lodash@4.17.21/node_modules/lodash/memoize.js +0 -20
  353. package/dist/node_modules/.bun/math-expression-evaluator@1.4.0/node_modules/math-expression-evaluator/src/formula_evaluator.js +0 -37
  354. package/dist/node_modules/.bun/math-expression-evaluator@1.4.0/node_modules/math-expression-evaluator/src/lexer.js +0 -509
  355. package/dist/node_modules/.bun/math-expression-evaluator@1.4.0/node_modules/math-expression-evaluator/src/math_function.js +0 -108
  356. package/dist/node_modules/.bun/math-expression-evaluator@1.4.0/node_modules/math-expression-evaluator/src/postfix.js +0 -31
  357. package/dist/node_modules/.bun/math-expression-evaluator@1.4.0/node_modules/math-expression-evaluator/src/postfix_evaluator.js +0 -45
  358. package/dist/node_modules/.bun/react-redux@9.3.0_9e2203c65d1d5fa1/node_modules/react-redux/dist/react-redux.js +0 -471
  359. package/dist/node_modules/.bun/reduce-css-calc@1.3.0/node_modules/reduce-css-calc/index.js +0 -49
  360. package/dist/node_modules/.bun/reduce-function-call@1.0.3/node_modules/reduce-function-call/index.js +0 -34
  361. package/dist/node_modules/.bun/redux-undo@1.1.0/node_modules/redux-undo/dist/redux-undo.js +0 -185
  362. package/dist/node_modules/.bun/redux@5.0.1/node_modules/redux/dist/redux.js +0 -198
  363. package/dist/node_modules/.bun/use-sync-external-store@1.6.0_f4eacebf2041cd4f/node_modules/use-sync-external-store/cjs/use-sync-external-store-with-selector.development.js +0 -53
  364. package/dist/node_modules/.bun/use-sync-external-store@1.6.0_f4eacebf2041cd4f/node_modules/use-sync-external-store/cjs/use-sync-external-store-with-selector.production.js +0 -51
  365. package/dist/node_modules/.bun/use-sync-external-store@1.6.0_f4eacebf2041cd4f/node_modules/use-sync-external-store/with-selector.js +0 -10
  366. package/dist/toggle-bar.d.ts +0 -40
  367. package/dist/toggle-bar.js +0 -155
  368. package/dist/tool-menu.d.ts +0 -29
  369. package/dist/tool-menu.js +0 -41
  370. package/dist/tools/absolute/component.d.ts +0 -16
  371. package/dist/tools/absolute/component.js +0 -15
  372. package/dist/tools/absolute/index.d.ts +0 -20
  373. package/dist/tools/absolute/index.js +0 -32
  374. package/dist/tools/circle/bg-circle.d.ts +0 -116
  375. package/dist/tools/circle/bg-circle.js +0 -72
  376. package/dist/tools/circle/component.d.ts +0 -83
  377. package/dist/tools/circle/component.js +0 -195
  378. package/dist/tools/circle/index.d.ts +0 -20
  379. package/dist/tools/circle/index.js +0 -22
  380. package/dist/tools/exponential/component.d.ts +0 -16
  381. package/dist/tools/exponential/component.js +0 -14
  382. package/dist/tools/exponential/index.d.ts +0 -20
  383. package/dist/tools/exponential/index.js +0 -32
  384. package/dist/tools/index.d.ts +0 -27
  385. package/dist/tools/index.js +0 -55
  386. package/dist/tools/line/component.d.ts +0 -124
  387. package/dist/tools/line/component.js +0 -72
  388. package/dist/tools/line/index.d.ts +0 -13
  389. package/dist/tools/line/index.js +0 -6
  390. package/dist/tools/parabola/component.d.ts +0 -16
  391. package/dist/tools/parabola/component.js +0 -14
  392. package/dist/tools/parabola/index.d.ts +0 -20
  393. package/dist/tools/parabola/index.js +0 -32
  394. package/dist/tools/point/component.d.ts +0 -50
  395. package/dist/tools/point/component.js +0 -85
  396. package/dist/tools/point/index.d.ts +0 -15
  397. package/dist/tools/point/index.js +0 -13
  398. package/dist/tools/polygon/component.d.ts +0 -120
  399. package/dist/tools/polygon/component.js +0 -262
  400. package/dist/tools/polygon/index.d.ts +0 -19
  401. package/dist/tools/polygon/index.js +0 -55
  402. package/dist/tools/polygon/line.d.ts +0 -125
  403. package/dist/tools/polygon/line.js +0 -63
  404. package/dist/tools/polygon/polygon.d.ts +0 -122
  405. package/dist/tools/polygon/polygon.js +0 -74
  406. package/dist/tools/ray/component.d.ts +0 -101
  407. package/dist/tools/ray/component.js +0 -71
  408. package/dist/tools/ray/index.d.ts +0 -13
  409. package/dist/tools/ray/index.js +0 -6
  410. package/dist/tools/segment/component.d.ts +0 -100
  411. package/dist/tools/segment/component.js +0 -52
  412. package/dist/tools/segment/index.d.ts +0 -13
  413. package/dist/tools/segment/index.js +0 -6
  414. package/dist/tools/shared/arrow-head.d.ts +0 -51
  415. package/dist/tools/shared/arrow-head.js +0 -63
  416. package/dist/tools/shared/icons/CorrectSVG.d.ts +0 -26
  417. package/dist/tools/shared/icons/CorrectSVG.js +0 -29
  418. package/dist/tools/shared/icons/IncorrectSVG.d.ts +0 -26
  419. package/dist/tools/shared/icons/IncorrectSVG.js +0 -27
  420. package/dist/tools/shared/icons/MissingSVG.d.ts +0 -26
  421. package/dist/tools/shared/icons/MissingSVG.js +0 -28
  422. package/dist/tools/shared/line/index.d.ts +0 -245
  423. package/dist/tools/shared/line/index.js +0 -319
  424. package/dist/tools/shared/line/line-path.d.ts +0 -57
  425. package/dist/tools/shared/line/line-path.js +0 -71
  426. package/dist/tools/shared/line/with-root-edge.d.ts +0 -142
  427. package/dist/tools/shared/line/with-root-edge.js +0 -73
  428. package/dist/tools/shared/point/arrow-point.d.ts +0 -56
  429. package/dist/tools/shared/point/arrow-point.js +0 -41
  430. package/dist/tools/shared/point/arrow.d.ts +0 -45
  431. package/dist/tools/shared/point/arrow.js +0 -35
  432. package/dist/tools/shared/point/base-point.d.ts +0 -52
  433. package/dist/tools/shared/point/base-point.js +0 -103
  434. package/dist/tools/shared/point/index.d.ts +0 -216
  435. package/dist/tools/shared/point/index.js +0 -45
  436. package/dist/tools/shared/styles.d.ts +0 -29
  437. package/dist/tools/shared/styles.js +0 -20
  438. package/dist/tools/shared/types.d.ts +0 -21
  439. package/dist/tools/shared/types.js +0 -11
  440. package/dist/tools/sine/component.d.ts +0 -16
  441. package/dist/tools/sine/component.js +0 -22
  442. package/dist/tools/sine/index.d.ts +0 -20
  443. package/dist/tools/sine/index.js +0 -32
  444. package/dist/tools/vector/component.d.ts +0 -100
  445. package/dist/tools/vector/component.js +0 -44
  446. package/dist/tools/vector/index.d.ts +0 -13
  447. package/dist/tools/vector/index.js +0 -6
  448. package/dist/undo-redo.d.ts +0 -22
  449. package/dist/undo-redo.js +0 -47
  450. package/dist/use-debounce.d.ts +0 -9
  451. package/dist/use-debounce.js +0 -13
  452. package/dist/utils.d.ts +0 -61
  453. package/dist/utils.js +0 -75
@@ -0,0 +1,235 @@
1
+ import { getLastAction, lastActionMiddleware } from '../middleware';
2
+
3
+ describe('lastActionMiddleware', () => {
4
+ describe('middleware functionality', () => {
5
+ it('creates middleware function', () => {
6
+ const middleware = lastActionMiddleware();
7
+ expect(typeof middleware).toBe('function');
8
+ });
9
+
10
+ it('middleware returns a function', () => {
11
+ const middleware = lastActionMiddleware();
12
+ const next = jest.fn();
13
+ const result = middleware(next);
14
+ expect(typeof result).toBe('function');
15
+ });
16
+
17
+ it('calls next with action', () => {
18
+ const middleware = lastActionMiddleware();
19
+ const next = jest.fn();
20
+ const dispatch = middleware(next);
21
+ const action = { type: 'TEST_ACTION' };
22
+
23
+ dispatch(action);
24
+ expect(next).toHaveBeenCalledWith(action);
25
+ });
26
+
27
+ it('returns result from next', () => {
28
+ const middleware = lastActionMiddleware();
29
+ const expectedResult = { status: 'success' };
30
+ const next = jest.fn(() => expectedResult);
31
+ const dispatch = middleware(next);
32
+ const action = { type: 'TEST_ACTION' };
33
+
34
+ const result = dispatch(action);
35
+ expect(result).toBe(expectedResult);
36
+ });
37
+
38
+ it('stores action before calling next', () => {
39
+ const middleware = lastActionMiddleware();
40
+ const next = jest.fn(() => {
41
+ const lastAction = getLastAction();
42
+ expect(lastAction).toBeDefined();
43
+ expect(lastAction.type).toBe('TEST_ACTION');
44
+ });
45
+ const dispatch = middleware(next);
46
+ const action = { type: 'TEST_ACTION' };
47
+
48
+ dispatch(action);
49
+ expect(next).toHaveBeenCalled();
50
+ });
51
+ });
52
+
53
+ describe('getLastAction', () => {
54
+ it('returns null initially', () => {
55
+ const lastAction = getLastAction();
56
+ expect(lastAction).toBeDefined();
57
+ });
58
+
59
+ it('returns last dispatched action', () => {
60
+ const middleware = lastActionMiddleware();
61
+ const next = jest.fn();
62
+ const dispatch = middleware(next);
63
+ const action = { type: 'MY_ACTION', payload: 'test' };
64
+
65
+ dispatch(action);
66
+ const lastAction = getLastAction();
67
+
68
+ expect(lastAction).toEqual(action);
69
+ expect(lastAction.type).toBe('MY_ACTION');
70
+ expect(lastAction.payload).toBe('test');
71
+ });
72
+
73
+ it('updates when new action is dispatched', () => {
74
+ const middleware = lastActionMiddleware();
75
+ const next = jest.fn();
76
+ const dispatch = middleware(next);
77
+
78
+ const action1 = { type: 'ACTION_1' };
79
+ dispatch(action1);
80
+ expect(getLastAction()).toEqual(action1);
81
+
82
+ const action2 = { type: 'ACTION_2' };
83
+ dispatch(action2);
84
+ expect(getLastAction()).toEqual(action2);
85
+ });
86
+
87
+ it('stores action with payload', () => {
88
+ const middleware = lastActionMiddleware();
89
+ const next = jest.fn();
90
+ const dispatch = middleware(next);
91
+ const action = {
92
+ type: 'CHANGE_MARKS',
93
+ marks: [{ id: 1, type: 'point' }],
94
+ };
95
+
96
+ dispatch(action);
97
+ const lastAction = getLastAction();
98
+
99
+ expect(lastAction.marks).toEqual(action.marks);
100
+ expect(lastAction.marks.length).toBe(1);
101
+ });
102
+
103
+ it('stores action with complex payload', () => {
104
+ const middleware = lastActionMiddleware();
105
+ const next = jest.fn();
106
+ const dispatch = middleware(next);
107
+ const action = {
108
+ type: 'COMPLEX_ACTION',
109
+ data: {
110
+ nested: {
111
+ value: 123,
112
+ array: [1, 2, 3],
113
+ },
114
+ },
115
+ };
116
+
117
+ dispatch(action);
118
+ const lastAction = getLastAction();
119
+
120
+ expect(lastAction.data.nested.value).toBe(123);
121
+ expect(lastAction.data.nested.array).toEqual([1, 2, 3]);
122
+ });
123
+ });
124
+
125
+ describe('action storage', () => {
126
+ it('overwrites previous action', () => {
127
+ const middleware = lastActionMiddleware();
128
+ const next = jest.fn();
129
+ const dispatch = middleware(next);
130
+
131
+ dispatch({ type: 'FIRST' });
132
+ dispatch({ type: 'SECOND' });
133
+ dispatch({ type: 'THIRD' });
134
+
135
+ const lastAction = getLastAction();
136
+ expect(lastAction.type).toBe('THIRD');
137
+ });
138
+
139
+ it('stores reference to action object', () => {
140
+ const middleware = lastActionMiddleware();
141
+ const next = jest.fn();
142
+ const dispatch = middleware(next);
143
+ const action = { type: 'TEST', data: { value: 1 } };
144
+
145
+ dispatch(action);
146
+ const lastAction = getLastAction();
147
+
148
+ // Modify original action
149
+ action.data.value = 2;
150
+
151
+ // Last action should reflect the change (same reference)
152
+ expect(lastAction.data.value).toBe(2);
153
+ });
154
+ });
155
+
156
+ describe('integration scenarios', () => {
157
+ it('works with multiple middleware calls', () => {
158
+ const middleware = lastActionMiddleware();
159
+ const next1 = jest.fn();
160
+ const next2 = jest.fn();
161
+
162
+ const dispatch1 = middleware(next1);
163
+ const dispatch2 = middleware(next2);
164
+
165
+ dispatch1({ type: 'ACTION_1' });
166
+ expect(getLastAction().type).toBe('ACTION_1');
167
+
168
+ dispatch2({ type: 'ACTION_2' });
169
+ expect(getLastAction().type).toBe('ACTION_2');
170
+ });
171
+
172
+ it('preserves action types', () => {
173
+ const middleware = lastActionMiddleware();
174
+ const next = jest.fn();
175
+ const dispatch = middleware(next);
176
+
177
+ const actionTypes = ['ADD_MARK', 'CHANGE_MARKS', 'DELETE_MARK', 'UPDATE_MARK'];
178
+
179
+ actionTypes.forEach((type) => {
180
+ dispatch({ type });
181
+ expect(getLastAction().type).toBe(type);
182
+ });
183
+ });
184
+
185
+ it('handles actions without type', () => {
186
+ const middleware = lastActionMiddleware();
187
+ const next = jest.fn();
188
+ const dispatch = middleware(next);
189
+ const action = { payload: 'data' };
190
+
191
+ dispatch(action);
192
+ const lastAction = getLastAction();
193
+
194
+ expect(lastAction).toEqual(action);
195
+ expect(lastAction.type).toBeUndefined();
196
+ });
197
+
198
+ it('handles empty action object', () => {
199
+ const middleware = lastActionMiddleware();
200
+ const next = jest.fn();
201
+ const dispatch = middleware(next);
202
+ const action = {};
203
+
204
+ dispatch(action);
205
+ const lastAction = getLastAction();
206
+
207
+ expect(lastAction).toEqual(action);
208
+ });
209
+ });
210
+
211
+ describe('error handling', () => {
212
+ it('stores action even if next throws error', () => {
213
+ const middleware = lastActionMiddleware();
214
+ const next = jest.fn(() => {
215
+ throw new Error('Next error');
216
+ });
217
+ const dispatch = middleware(next);
218
+ const action = { type: 'ERROR_ACTION' };
219
+
220
+ expect(() => dispatch(action)).toThrow('Next error');
221
+ expect(getLastAction()).toEqual(action);
222
+ });
223
+
224
+ it('propagates errors from next', () => {
225
+ const middleware = lastActionMiddleware();
226
+ const errorMessage = 'Test error';
227
+ const next = jest.fn(() => {
228
+ throw new Error(errorMessage);
229
+ });
230
+ const dispatch = middleware(next);
231
+
232
+ expect(() => dispatch({ type: 'TEST' })).toThrow(errorMessage);
233
+ });
234
+ });
235
+ });
@@ -0,0 +1,324 @@
1
+ import reducer from '../reducer';
2
+ import { changeMarks } from '../actions';
3
+
4
+ describe('container reducer', () => {
5
+ describe('reducer creation', () => {
6
+ it('creates a reducer function', () => {
7
+ const rootReducer = reducer();
8
+ expect(typeof rootReducer).toBe('function');
9
+ });
10
+
11
+ it('returns an object with marks property', () => {
12
+ const rootReducer = reducer();
13
+ const state = rootReducer(undefined, { type: 'INIT' });
14
+ expect(state).toHaveProperty('marks');
15
+ });
16
+
17
+ it('wraps marks reducer with undoable', () => {
18
+ const rootReducer = reducer();
19
+ const state = rootReducer(undefined, { type: 'INIT' });
20
+
21
+ expect(state.marks).toHaveProperty('past');
22
+ expect(state.marks).toHaveProperty('present');
23
+ expect(state.marks).toHaveProperty('future');
24
+ });
25
+ });
26
+
27
+ describe('initial state', () => {
28
+ it('initializes with empty past', () => {
29
+ const rootReducer = reducer();
30
+ const state = rootReducer(undefined, { type: 'INIT' });
31
+
32
+ expect(state.marks.past).toEqual([]);
33
+ });
34
+
35
+ it('initializes with empty present', () => {
36
+ const rootReducer = reducer();
37
+ const state = rootReducer(undefined, { type: 'INIT' });
38
+
39
+ expect(state.marks.present).toEqual([]);
40
+ });
41
+
42
+ it('initializes with empty future', () => {
43
+ const rootReducer = reducer();
44
+ const state = rootReducer(undefined, { type: 'INIT' });
45
+
46
+ expect(state.marks.future).toEqual([]);
47
+ });
48
+ });
49
+
50
+ describe('CHANGE_MARKS action', () => {
51
+ it('updates present with new marks', () => {
52
+ const rootReducer = reducer();
53
+ const newMarks = [{ id: 1, type: 'point', x: 5, y: 10 }];
54
+ const action = changeMarks(newMarks);
55
+
56
+ let state = rootReducer(undefined, { type: '@@INIT' });
57
+ state = rootReducer(state, action);
58
+
59
+ expect(state.marks.present).toEqual(newMarks);
60
+ });
61
+
62
+ it('adds previous state to past', () => {
63
+ const rootReducer = reducer();
64
+ const firstMarks = [{ id: 1, type: 'point' }];
65
+ const secondMarks = [{ id: 2, type: 'line' }];
66
+
67
+ let state = rootReducer(undefined, { type: '@@INIT' });
68
+ state = rootReducer(state, changeMarks(firstMarks));
69
+ state = rootReducer(state, changeMarks(secondMarks));
70
+
71
+ expect(state.marks.past.length).toBeGreaterThan(0);
72
+ expect(state.marks.present).toEqual(secondMarks);
73
+ });
74
+
75
+ it('clears future on new action', () => {
76
+ const rootReducer = reducer();
77
+ const firstMarks = [{ id: 1, type: 'point' }];
78
+ const secondMarks = [{ id: 2, type: 'line' }];
79
+ const thirdMarks = [{ id: 3, type: 'circle' }];
80
+
81
+ let state = rootReducer(undefined, { type: '@@INIT' });
82
+ state = rootReducer(state, changeMarks(firstMarks));
83
+ state = rootReducer(state, changeMarks(secondMarks));
84
+ state = rootReducer(state, { type: '@@redux-undo/UNDO' });
85
+
86
+ expect(state.marks.future.length).toBeGreaterThan(0);
87
+
88
+ state = rootReducer(state, changeMarks(thirdMarks));
89
+ expect(state.marks.future).toEqual([]);
90
+ });
91
+
92
+ it('handles empty marks array', () => {
93
+ const rootReducer = reducer();
94
+ const action = changeMarks([]);
95
+
96
+ let state = rootReducer(undefined, { type: '@@INIT' });
97
+ state = rootReducer(state, action);
98
+
99
+ expect(state.marks.present).toEqual([]);
100
+ });
101
+
102
+ it('handles multiple marks', () => {
103
+ const rootReducer = reducer();
104
+ const marks = [
105
+ { id: 1, type: 'point', x: 1, y: 2 },
106
+ { id: 2, type: 'line', from: { x: 0, y: 0 }, to: { x: 5, y: 5 } },
107
+ { id: 3, type: 'circle', center: { x: 3, y: 3 }, radius: 2 },
108
+ ];
109
+ const action = changeMarks(marks);
110
+
111
+ let state = rootReducer(undefined, { type: '@@INIT' });
112
+ state = rootReducer(state, action);
113
+
114
+ expect(state.marks.present).toEqual(marks);
115
+ expect(state.marks.present.length).toBe(3);
116
+ });
117
+ });
118
+
119
+ describe('undo functionality', () => {
120
+ it('supports undo action', () => {
121
+ const rootReducer = reducer();
122
+ const firstMarks = [{ id: 1, type: 'point' }];
123
+ const secondMarks = [{ id: 2, type: 'line' }];
124
+
125
+ let state = rootReducer(undefined, { type: '@@INIT' });
126
+ state = rootReducer(state, changeMarks(firstMarks));
127
+ state = rootReducer(state, changeMarks(secondMarks));
128
+
129
+ state = rootReducer(state, { type: '@@redux-undo/UNDO' });
130
+
131
+ expect(state.marks.present).toEqual(firstMarks);
132
+ });
133
+
134
+ it('moves present to future on undo', () => {
135
+ const rootReducer = reducer();
136
+ const firstMarks = [{ id: 1, type: 'point' }];
137
+ const secondMarks = [{ id: 2, type: 'line' }];
138
+
139
+ let state = rootReducer(undefined, { type: '@@INIT' });
140
+ state = rootReducer(state, changeMarks(firstMarks));
141
+ state = rootReducer(state, changeMarks(secondMarks));
142
+ state = rootReducer(state, { type: '@@redux-undo/UNDO' });
143
+
144
+ expect(state.marks.future.length).toBeGreaterThan(0);
145
+ });
146
+
147
+ it('handles multiple undos', () => {
148
+ const rootReducer = reducer();
149
+ const marks1 = [{ id: 1 }];
150
+ const marks2 = [{ id: 2 }];
151
+ const marks3 = [{ id: 3 }];
152
+
153
+ let state = rootReducer(undefined, { type: '@@INIT' });
154
+ state = rootReducer(state, changeMarks(marks1));
155
+ state = rootReducer(state, changeMarks(marks2));
156
+ state = rootReducer(state, changeMarks(marks3));
157
+
158
+ state = rootReducer(state, { type: '@@redux-undo/UNDO' });
159
+ state = rootReducer(state, { type: '@@redux-undo/UNDO' });
160
+
161
+ expect(state.marks.present).toEqual(marks1);
162
+ });
163
+
164
+ it('does not undo beyond initial state', () => {
165
+ const rootReducer = reducer();
166
+ const marks = [{ id: 1, type: 'point' }];
167
+
168
+ let state = rootReducer(undefined, { type: '@@INIT' });
169
+ state = rootReducer(state, changeMarks(marks));
170
+
171
+ state = rootReducer(state, { type: '@@redux-undo/UNDO' });
172
+ state = rootReducer(state, { type: '@@redux-undo/UNDO' });
173
+ state = rootReducer(state, { type: '@@redux-undo/UNDO' });
174
+
175
+ expect(state.marks.present).toEqual([]);
176
+ expect(state.marks.past).toEqual([]);
177
+ });
178
+ });
179
+
180
+ describe('redo functionality', () => {
181
+ it('supports redo action', () => {
182
+ const rootReducer = reducer();
183
+ const firstMarks = [{ id: 1, type: 'point' }];
184
+ const secondMarks = [{ id: 2, type: 'line' }];
185
+
186
+ let state = rootReducer(undefined, { type: '@@INIT' });
187
+ state = rootReducer(state, changeMarks(firstMarks));
188
+ state = rootReducer(state, changeMarks(secondMarks));
189
+ state = rootReducer(state, { type: '@@redux-undo/UNDO' });
190
+
191
+ // Redo
192
+ state = rootReducer(state, { type: '@@redux-undo/REDO' });
193
+
194
+ expect(state.marks.present).toEqual(secondMarks);
195
+ });
196
+
197
+ it('moves future to past on redo', () => {
198
+ const rootReducer = reducer();
199
+ const firstMarks = [{ id: 1, type: 'point' }];
200
+ const secondMarks = [{ id: 2, type: 'line' }];
201
+
202
+ let state = rootReducer(undefined, { type: '@@INIT' });
203
+ state = rootReducer(state, changeMarks(firstMarks));
204
+ state = rootReducer(state, changeMarks(secondMarks));
205
+ state = rootReducer(state, { type: '@@redux-undo/UNDO' });
206
+
207
+ const futureBeforeRedo = state.marks.future.length;
208
+ state = rootReducer(state, { type: '@@redux-undo/REDO' });
209
+
210
+ expect(state.marks.future.length).toBe(futureBeforeRedo - 1);
211
+ });
212
+
213
+ it('handles multiple redos', () => {
214
+ const rootReducer = reducer();
215
+ const marks1 = [{ id: 1 }];
216
+ const marks2 = [{ id: 2 }];
217
+ const marks3 = [{ id: 3 }];
218
+
219
+ let state = rootReducer(undefined, { type: '@@INIT' });
220
+ state = rootReducer(state, changeMarks(marks1));
221
+ state = rootReducer(state, changeMarks(marks2));
222
+ state = rootReducer(state, changeMarks(marks3));
223
+
224
+ // undo twice, then redo twice
225
+ state = rootReducer(state, { type: '@@redux-undo/UNDO' });
226
+ state = rootReducer(state, { type: '@@redux-undo/UNDO' });
227
+ state = rootReducer(state, { type: '@@redux-undo/REDO' });
228
+ state = rootReducer(state, { type: '@@redux-undo/REDO' });
229
+
230
+ expect(state.marks.present).toEqual(marks3);
231
+ });
232
+
233
+ it('does not redo beyond last state', () => {
234
+ const rootReducer = reducer();
235
+ const marks = [{ id: 1, type: 'point' }];
236
+
237
+ let state = rootReducer(undefined, { type: '@@INIT' });
238
+ state = rootReducer(state, changeMarks(marks));
239
+ state = rootReducer(state, { type: '@@redux-undo/UNDO' });
240
+ state = rootReducer(state, { type: '@@redux-undo/REDO' });
241
+
242
+ // try to redo beyond the end
243
+ state = rootReducer(state, { type: '@@redux-undo/REDO' });
244
+ state = rootReducer(state, { type: '@@redux-undo/REDO' });
245
+
246
+ expect(state.marks.present).toEqual(marks);
247
+ expect(state.marks.future).toEqual([]);
248
+ });
249
+ });
250
+
251
+ describe('state structure', () => {
252
+ it('maintains correct undoable structure', () => {
253
+ const rootReducer = reducer();
254
+ const state = rootReducer(undefined, { type: 'INIT' });
255
+
256
+ expect(state.marks).toHaveProperty('past');
257
+ expect(state.marks).toHaveProperty('present');
258
+ expect(state.marks).toHaveProperty('future');
259
+ expect(Array.isArray(state.marks.past)).toBe(true);
260
+ expect(Array.isArray(state.marks.future)).toBe(true);
261
+ });
262
+
263
+ it('preserves mark properties through undo/redo', () => {
264
+ const rootReducer = reducer();
265
+ const marks = [
266
+ {
267
+ id: 1,
268
+ type: 'point',
269
+ x: 10,
270
+ y: 20,
271
+ label: 'A',
272
+ correctness: { value: 'correct' },
273
+ },
274
+ ];
275
+
276
+ let state = rootReducer(undefined, { type: 'INIT' });
277
+ state = rootReducer(state, changeMarks(marks));
278
+ state = rootReducer(state, changeMarks([]));
279
+ state = rootReducer(state, { type: '@@redux-undo/UNDO' });
280
+
281
+ expect(state.marks.present).toEqual(marks);
282
+ expect(state.marks.present[0]).toBeDefined();
283
+ if (state.marks.present[0]) {
284
+ expect(state.marks.present[0]).toEqual(marks[0]);
285
+ expect(state.marks.present[0].correctness).toBeDefined();
286
+ }
287
+ });
288
+ });
289
+
290
+ describe('integration', () => {
291
+ it('handles complex undo/redo sequences', () => {
292
+ const rootReducer = reducer();
293
+ const marks1 = [{ id: 1 }];
294
+ const marks2 = [{ id: 2 }];
295
+ const marks3 = [{ id: 3 }];
296
+ const marks4 = [{ id: 4 }];
297
+
298
+ let state = rootReducer(undefined, { type: '@@INIT' });
299
+ state = rootReducer(state, changeMarks(marks1));
300
+ state = rootReducer(state, changeMarks(marks2));
301
+ state = rootReducer(state, changeMarks(marks3));
302
+ state = rootReducer(state, { type: '@@redux-undo/UNDO' }); // back to marks2
303
+ state = rootReducer(state, { type: '@@redux-undo/UNDO' }); // back to marks1
304
+ state = rootReducer(state, { type: '@@redux-undo/REDO' }); // forward to marks2
305
+ state = rootReducer(state, changeMarks(marks4));
306
+
307
+ expect(state.marks.present).toEqual(marks4);
308
+ expect(state.marks.future).toEqual([]);
309
+ });
310
+
311
+ it('works with empty state transitions', () => {
312
+ const rootReducer = reducer();
313
+ const marks = [{ id: 1 }];
314
+
315
+ let state = rootReducer(undefined, { type: '@@INIT' });
316
+ state = rootReducer(state, changeMarks([]));
317
+ state = rootReducer(state, changeMarks(marks));
318
+ state = rootReducer(state, changeMarks([]));
319
+ state = rootReducer(state, { type: '@@redux-undo/UNDO' });
320
+
321
+ expect(state.marks.present).toEqual(marks);
322
+ });
323
+ });
324
+ });
@@ -0,0 +1,8 @@
1
+ export const addMark = () => ({
2
+ type: 'ADD_MARK',
3
+ });
4
+
5
+ export const changeMarks = (marks) => ({
6
+ type: 'CHANGE_MARKS',
7
+ marks,
8
+ });
@@ -0,0 +1,91 @@
1
+ import { connect, Provider } from 'react-redux';
2
+ import React from 'react';
3
+ import { applyMiddleware, createStore } from 'redux';
4
+ import reducer from './reducer';
5
+ import { changeMarks } from './actions';
6
+ import PropTypes from 'prop-types';
7
+ import { isEqual } from 'lodash-es';
8
+ import { ActionCreators } from 'redux-undo';
9
+ import GraphWithControls from '../graph-with-controls';
10
+ import { lastActionMiddleware } from './middleware';
11
+
12
+ const mapStateToProps = (s) => ({
13
+ marks: s.marks.present,
14
+ });
15
+
16
+ const mapDispatchToProps = (dispatch) => ({
17
+ onChangeMarks: (m) => dispatch(changeMarks(m)),
18
+ onUndo: () => dispatch(ActionCreators.undo()),
19
+ onRedo: () => dispatch(ActionCreators.redo()),
20
+ onReset: () => dispatch(changeMarks([])),
21
+ });
22
+
23
+ export const GraphContainer = connect(mapStateToProps, mapDispatchToProps)(GraphWithControls);
24
+
25
+ /**
26
+ * The graph component entry point with undo/redo
27
+ * Redux is an implementation detail, hide it in the react component.
28
+ */
29
+ class Root extends React.Component {
30
+ static propTypes = {
31
+ onChangeMarks: PropTypes.func,
32
+ marks: PropTypes.array,
33
+ };
34
+
35
+ constructor(props) {
36
+ super(props);
37
+
38
+ const r = reducer();
39
+ this.store = createStore(r, { marks: props.marks }, applyMiddleware(lastActionMiddleware));
40
+
41
+ this.store.subscribe(this.onStoreChange);
42
+ }
43
+
44
+ componentDidUpdate(prevProps) {
45
+ const { marks } = this.props;
46
+ const storeState = this.store.getState();
47
+
48
+ if (isEqual(storeState.marks.present, marks)) {
49
+ return;
50
+ }
51
+
52
+ if (!isEqual(prevProps.marks, marks)) {
53
+ // Don't dispatch marks that have correctness (evaluate mode marks) into the undo history.
54
+ // These marks are already handled in render() via the correctnessSet path and bypass the
55
+ // Redux store entirely. Dispatching them would pollute the undo history and cause the graph
56
+ // to show the correct answer when the user presses undo after returning to gather mode.
57
+ const hasCorrectness = marks && marks.find((m) => m.correctness);
58
+
59
+ if (!hasCorrectness) {
60
+ this.store.dispatch(changeMarks(marks));
61
+ }
62
+ }
63
+ }
64
+
65
+ onStoreChange = () => {
66
+ const { marks, onChangeMarks } = this.props;
67
+ const storeState = this.store.getState();
68
+
69
+ if (!isEqual(storeState.marks.present, marks)) {
70
+ onChangeMarks(storeState.marks.present);
71
+ }
72
+ };
73
+
74
+ render() {
75
+ // eslint-disable-next-line no-unused-vars
76
+ const { onChangeMarks, marks, ...rest } = this.props;
77
+ const correctnessSet = marks && marks.find((m) => m.correctness);
78
+
79
+ if (correctnessSet) {
80
+ return <GraphWithControls {...rest} marks={marks} disabled={correctnessSet} />;
81
+ }
82
+
83
+ return (
84
+ <Provider store={this.store}>
85
+ <GraphContainer {...rest} />
86
+ </Provider>
87
+ );
88
+ }
89
+ }
90
+
91
+ export default Root;
@@ -0,0 +1,14 @@
1
+ const marks = (state = [], action) => {
2
+ switch (action.type) {
3
+ case 'CHANGE_MARKS':
4
+ if (Array.isArray(action.marks)) {
5
+ return action.marks;
6
+ } else {
7
+ throw new Error('marks must be an array');
8
+ }
9
+ default:
10
+ return state;
11
+ }
12
+ };
13
+
14
+ export default marks;
@@ -0,0 +1,7 @@
1
+ let lastAction = null;
2
+ export const getLastAction = () => lastAction;
3
+
4
+ export const lastActionMiddleware = () => (next) => (action) => {
5
+ lastAction = action;
6
+ return next(action);
7
+ };
@@ -0,0 +1,5 @@
1
+ import { combineReducers } from 'redux';
2
+ import marks from './marks';
3
+ import undoable from 'redux-undo';
4
+
5
+ export default () => combineReducers({ marks: undoable(marks, { debug: false }) });