@pie-lib/graphing 3.1.1-next.0 → 3.1.1-next.1

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 (197) hide show
  1. package/lib/axis/arrow.js +0 -3
  2. package/lib/axis/arrow.js.map +1 -1
  3. package/lib/axis/axes.js +0 -16
  4. package/lib/axis/axes.js.map +1 -1
  5. package/lib/axis/index.js +0 -7
  6. package/lib/axis/index.js.map +1 -1
  7. package/lib/bg.js +0 -6
  8. package/lib/bg.js.map +1 -1
  9. package/lib/container/actions.js +0 -1
  10. package/lib/container/actions.js.map +1 -1
  11. package/lib/container/index.js +4 -8
  12. package/lib/container/index.js.map +1 -1
  13. package/lib/container/marks.js +0 -2
  14. package/lib/container/marks.js.map +1 -1
  15. package/lib/container/middleware.js +0 -1
  16. package/lib/container/middleware.js.map +1 -1
  17. package/lib/container/reducer.js +0 -1
  18. package/lib/container/reducer.js.map +1 -1
  19. package/lib/coordinates-label.js +0 -11
  20. package/lib/coordinates-label.js.map +1 -1
  21. package/lib/graph-with-controls.js +3 -21
  22. package/lib/graph-with-controls.js.map +1 -1
  23. package/lib/graph.js +8 -27
  24. package/lib/graph.js.map +1 -1
  25. package/lib/grid-setup.js +0 -11
  26. package/lib/grid-setup.js.map +1 -1
  27. package/lib/grid.js +0 -22
  28. package/lib/grid.js.map +1 -1
  29. package/lib/index.js +0 -7
  30. package/lib/index.js.map +1 -1
  31. package/lib/key-legend.js +1 -2
  32. package/lib/key-legend.js.map +1 -1
  33. package/lib/label-svg-icon.js +0 -1
  34. package/lib/label-svg-icon.js.map +1 -1
  35. package/lib/labels.js +0 -13
  36. package/lib/labels.js.map +1 -1
  37. package/lib/mark-label.js +0 -15
  38. package/lib/mark-label.js.map +1 -1
  39. package/lib/toggle-bar.js +0 -17
  40. package/lib/toggle-bar.js.map +1 -1
  41. package/lib/tool-menu.js +0 -3
  42. package/lib/tool-menu.js.map +1 -1
  43. package/lib/tools/absolute/component.js +0 -1
  44. package/lib/tools/absolute/component.js.map +1 -1
  45. package/lib/tools/absolute/index.js +0 -10
  46. package/lib/tools/absolute/index.js.map +1 -1
  47. package/lib/tools/circle/bg-circle.js +0 -15
  48. package/lib/tools/circle/bg-circle.js.map +1 -1
  49. package/lib/tools/circle/component.js +2 -15
  50. package/lib/tools/circle/component.js.map +1 -1
  51. package/lib/tools/circle/index.js +0 -10
  52. package/lib/tools/circle/index.js.map +1 -1
  53. package/lib/tools/exponential/component.js +0 -1
  54. package/lib/tools/exponential/component.js.map +1 -1
  55. package/lib/tools/exponential/index.js +0 -10
  56. package/lib/tools/exponential/index.js.map +1 -1
  57. package/lib/tools/index.js +0 -1
  58. package/lib/tools/index.js.map +1 -1
  59. package/lib/tools/line/component.js +0 -2
  60. package/lib/tools/line/component.js.map +1 -1
  61. package/lib/tools/line/index.js +0 -1
  62. package/lib/tools/line/index.js.map +1 -1
  63. package/lib/tools/parabola/component.js +0 -1
  64. package/lib/tools/parabola/component.js.map +1 -1
  65. package/lib/tools/parabola/index.js +0 -10
  66. package/lib/tools/parabola/index.js.map +1 -1
  67. package/lib/tools/point/component.js +3 -15
  68. package/lib/tools/point/component.js.map +1 -1
  69. package/lib/tools/point/index.js +0 -10
  70. package/lib/tools/point/index.js.map +1 -1
  71. package/lib/tools/polygon/component.js +5 -29
  72. package/lib/tools/polygon/component.js.map +1 -1
  73. package/lib/tools/polygon/index.js +0 -12
  74. package/lib/tools/polygon/index.js.map +1 -1
  75. package/lib/tools/polygon/line.js +0 -15
  76. package/lib/tools/polygon/line.js.map +1 -1
  77. package/lib/tools/polygon/polygon.js +0 -18
  78. package/lib/tools/polygon/polygon.js.map +1 -1
  79. package/lib/tools/ray/component.js +0 -11
  80. package/lib/tools/ray/component.js.map +1 -1
  81. package/lib/tools/ray/index.js +0 -1
  82. package/lib/tools/ray/index.js.map +1 -1
  83. package/lib/tools/segment/component.js +0 -10
  84. package/lib/tools/segment/component.js.map +1 -1
  85. package/lib/tools/segment/index.js +0 -1
  86. package/lib/tools/segment/index.js.map +1 -1
  87. package/lib/tools/shared/arrow-head.js +0 -14
  88. package/lib/tools/shared/arrow-head.js.map +1 -1
  89. package/lib/tools/shared/icons/CorrectSVG.js +0 -1
  90. package/lib/tools/shared/icons/CorrectSVG.js.map +1 -1
  91. package/lib/tools/shared/icons/IncorrectSVG.js +0 -1
  92. package/lib/tools/shared/icons/IncorrectSVG.js.map +1 -1
  93. package/lib/tools/shared/icons/MissingSVG.js +0 -1
  94. package/lib/tools/shared/icons/MissingSVG.js.map +1 -1
  95. package/lib/tools/shared/line/index.js +7 -21
  96. package/lib/tools/shared/line/index.js.map +1 -1
  97. package/lib/tools/shared/line/line-path.js +0 -15
  98. package/lib/tools/shared/line/line-path.js.map +1 -1
  99. package/lib/tools/shared/line/with-root-edge.js +0 -11
  100. package/lib/tools/shared/line/with-root-edge.js.map +1 -1
  101. package/lib/tools/shared/point/arrow-point.js +0 -3
  102. package/lib/tools/shared/point/arrow-point.js.map +1 -1
  103. package/lib/tools/shared/point/arrow.js +0 -3
  104. package/lib/tools/shared/point/arrow.js.map +1 -1
  105. package/lib/tools/shared/point/base-point.js +0 -16
  106. package/lib/tools/shared/point/base-point.js.map +1 -1
  107. package/lib/tools/shared/point/index.js +0 -7
  108. package/lib/tools/shared/point/index.js.map +1 -1
  109. package/lib/tools/shared/styles.js +0 -1
  110. package/lib/tools/shared/styles.js.map +1 -1
  111. package/lib/tools/shared/types.js +0 -1
  112. package/lib/tools/shared/types.js.map +1 -1
  113. package/lib/tools/sine/component.js +0 -10
  114. package/lib/tools/sine/component.js.map +1 -1
  115. package/lib/tools/sine/index.js +0 -10
  116. package/lib/tools/sine/index.js.map +1 -1
  117. package/lib/tools/vector/component.js +0 -10
  118. package/lib/tools/vector/component.js.map +1 -1
  119. package/lib/tools/vector/index.js +0 -1
  120. package/lib/tools/vector/index.js.map +1 -1
  121. package/lib/undo-redo.js +0 -2
  122. package/lib/undo-redo.js.map +1 -1
  123. package/lib/use-debounce.js +0 -2
  124. package/lib/use-debounce.js.map +1 -1
  125. package/lib/utils.js +9 -26
  126. package/lib/utils.js.map +1 -1
  127. package/package.json +9 -9
  128. package/src/__tests__/bg.test.jsx +250 -0
  129. package/src/__tests__/coordinates-label.test.jsx +243 -0
  130. package/src/__tests__/graph-with-controls.test.jsx +9 -16
  131. package/src/__tests__/graph.test.jsx +560 -5
  132. package/src/__tests__/grid-setup.test.jsx +645 -0
  133. package/src/__tests__/grid.test.jsx +1 -1
  134. package/src/__tests__/key-legend.test.jsx +260 -0
  135. package/src/__tests__/label-svg-icon.test.jsx +278 -0
  136. package/src/__tests__/mark-label.test.jsx +1 -1
  137. package/src/__tests__/toggle-bar.test.jsx +0 -6
  138. package/src/__tests__/tool-menu.test.jsx +0 -4
  139. package/src/__tests__/use-debounce.test.js +1 -1
  140. package/src/__tests__/utils.test.js +15 -61
  141. package/src/axis/__tests__/axes.test.jsx +1 -1
  142. package/src/axis/axes.jsx +7 -21
  143. package/src/axis/index.js +1 -0
  144. package/src/bg.jsx +1 -1
  145. package/src/container/__tests__/actions.test.js +105 -0
  146. package/src/container/__tests__/index.test.jsx +319 -0
  147. package/src/container/__tests__/marks.test.js +172 -0
  148. package/src/container/__tests__/middleware.test.js +235 -0
  149. package/src/container/__tests__/reducer.test.js +324 -0
  150. package/src/container/index.jsx +2 -3
  151. package/src/coordinates-label.jsx +1 -7
  152. package/src/graph-with-controls.jsx +8 -6
  153. package/src/graph.jsx +2 -3
  154. package/src/grid-setup.jsx +1 -1
  155. package/src/key-legend.jsx +2 -1
  156. package/src/mark-label.jsx +7 -24
  157. package/src/toggle-bar.jsx +8 -1
  158. package/src/tools/absolute/__tests__/component.test.jsx +1 -2
  159. package/src/tools/absolute/component.jsx +2 -2
  160. package/src/tools/circle/__tests__/component.test.jsx +438 -0
  161. package/src/tools/circle/__tests__/index.test.js +480 -0
  162. package/src/tools/circle/bg-circle.jsx +2 -2
  163. package/src/tools/circle/component.jsx +10 -12
  164. package/src/tools/exponential/__tests__/component.test.jsx +0 -1
  165. package/src/tools/exponential/__tests__/index.test.js +729 -0
  166. package/src/tools/exponential/component.jsx +1 -1
  167. package/src/tools/line/__tests__/component.test.jsx +1 -0
  168. package/src/tools/line/component.jsx +4 -11
  169. package/src/tools/parabola/__tests__/component.test.jsx +0 -1
  170. package/src/tools/parabola/__tests__/index.test.js +470 -0
  171. package/src/tools/parabola/component.jsx +1 -1
  172. package/src/tools/point/__tests__/component.test.jsx +310 -2
  173. package/src/tools/point/__tests__/index.test.js +241 -0
  174. package/src/tools/point/component.jsx +1 -2
  175. package/src/tools/polygon/__tests__/component.test.jsx +391 -2
  176. package/src/tools/polygon/__tests__/index.test.js +237 -8
  177. package/src/tools/polygon/__tests__/line.test.jsx +13 -0
  178. package/src/tools/polygon/__tests__/polygon.test.jsx +19 -1
  179. package/src/tools/polygon/component.jsx +4 -14
  180. package/src/tools/polygon/line.jsx +1 -1
  181. package/src/tools/polygon/polygon.jsx +1 -1
  182. package/src/tools/ray/__tests__/component.test.jsx +1 -0
  183. package/src/tools/ray/component.jsx +3 -5
  184. package/src/tools/segment/__tests__/component.test.jsx +1 -0
  185. package/src/tools/segment/component.jsx +1 -1
  186. package/src/tools/shared/arrow-head.jsx +11 -6
  187. package/src/tools/shared/line/__tests__/index.test.jsx +1 -1
  188. package/src/tools/shared/line/__tests__/line-path.test.jsx +3 -3
  189. package/src/tools/shared/line/__tests__/with-root-edge.test.jsx +2 -2
  190. package/src/tools/shared/line/index.jsx +4 -6
  191. package/src/tools/shared/line/line-path.jsx +2 -8
  192. package/src/tools/shared/point/arrow-point.jsx +2 -5
  193. package/src/tools/sine/component.jsx +2 -2
  194. package/src/tools/vector/component.jsx +1 -1
  195. package/src/undo-redo.jsx +3 -9
  196. package/src/use-debounce.js +1 -1
  197. package/src/utils.js +1 -5
@@ -1,8 +1,6 @@
1
1
  import React from 'react';
2
2
  import { render } from '@pie-lib/test-utils';
3
-
4
- import { xy, graphProps } from './utils';
5
-
3
+ import { graphProps } from './utils';
6
4
  import Graph, { removeBuildingToolIfCurrentToolDiffers } from '../graph';
7
5
  import { toolsArr } from '../tools';
8
6
 
@@ -18,6 +16,7 @@ describe('removeBuildingToolIfCurrentToolDiffers', () => {
18
16
  {
19
17
  type: 'line',
20
18
  from: { x: 0, y: 0 },
19
+ to: { x: 1, y: 1 },
21
20
  label: 'Line',
22
21
  building: true,
23
22
  },
@@ -106,7 +105,7 @@ describe('Graph', () => {
106
105
  describe('props handling', () => {
107
106
  it('calls onChangeMarks when marks prop changes', () => {
108
107
  const { rerender } = render(<Graph {...defaultProps} />);
109
-
108
+
110
109
  const newMarks = [
111
110
  {
112
111
  type: 'point',
@@ -118,7 +117,7 @@ describe('Graph', () => {
118
117
  ];
119
118
 
120
119
  rerender(<Graph {...defaultProps} marks={newMarks} />);
121
-
120
+
122
121
  // Component should render with new marks
123
122
  // Note: onChangeMarks is called internally when marks are changed through user interaction,
124
123
  // not when props change, so we just verify the component renders correctly
@@ -163,4 +162,560 @@ describe('Graph', () => {
163
162
  expect(container.firstChild).toBeInTheDocument();
164
163
  });
165
164
  });
165
+
166
+ describe('updateMarks method', () => {
167
+ it('updates an existing mark', () => {
168
+ const wrapper = render(<Graph {...defaultProps} />);
169
+
170
+ const { rerender } = wrapper;
171
+
172
+ const updatedMarks = [
173
+ {
174
+ type: 'point',
175
+ x: 3,
176
+ y: 3,
177
+ label: 'Updated Point',
178
+ showLabel: true,
179
+ },
180
+ {
181
+ type: 'line',
182
+ from: { x: 0, y: 0 },
183
+ to: { x: 1, y: 1 },
184
+ label: 'Line',
185
+ },
186
+ ];
187
+
188
+ rerender(<Graph {...defaultProps} marks={updatedMarks} />);
189
+
190
+ // Component should render with updated marks
191
+ expect(wrapper.container.firstChild).toBeInTheDocument();
192
+ });
193
+
194
+ it('does not update if mark is duplicated', () => {
195
+ const duplicatedMarks = [
196
+ {
197
+ type: 'point',
198
+ x: 2,
199
+ y: 2,
200
+ label: 'Point',
201
+ showLabel: true,
202
+ },
203
+ {
204
+ type: 'point',
205
+ x: 2,
206
+ y: 2,
207
+ label: 'Point',
208
+ showLabel: true,
209
+ },
210
+ ];
211
+
212
+ const { container } = render(<Graph {...defaultProps} marks={duplicatedMarks} />);
213
+ expect(container.firstChild).toBeInTheDocument();
214
+ });
215
+
216
+ it('adds mark if addIfMissing is true and mark does not exist', () => {
217
+ const { container } = render(<Graph {...defaultProps} />);
218
+
219
+ // The component should handle adding new marks through user interaction
220
+ expect(container.firstChild).toBeInTheDocument();
221
+ });
222
+
223
+ it('does not update if mark has no building flag and is duplicated', () => {
224
+ const marks = [
225
+ {
226
+ type: 'point',
227
+ x: 2,
228
+ y: 2,
229
+ label: 'Point',
230
+ showLabel: true,
231
+ },
232
+ ];
233
+
234
+ const { container } = render(<Graph {...defaultProps} marks={marks} />);
235
+ expect(container.firstChild).toBeInTheDocument();
236
+ });
237
+
238
+ it('handles building marks correctly', () => {
239
+ const marksWithBuilding = [
240
+ {
241
+ type: 'line',
242
+ from: { x: 0, y: 0 },
243
+ building: true,
244
+ },
245
+ ];
246
+
247
+ const { container } = render(<Graph {...defaultProps} marks={marksWithBuilding} />);
248
+ expect(container.firstChild).toBeInTheDocument();
249
+ });
250
+
251
+ it('updates building mark to completed mark', () => {
252
+ const marksWithBuilding = [
253
+ {
254
+ type: 'line',
255
+ from: { x: 0, y: 0 },
256
+ building: true,
257
+ },
258
+ ];
259
+
260
+ const { rerender, container } = render(<Graph {...defaultProps} marks={marksWithBuilding} />);
261
+
262
+ const completedMarks = [
263
+ {
264
+ type: 'line',
265
+ from: { x: 0, y: 0 },
266
+ to: { x: 1, y: 1 },
267
+ building: false,
268
+ },
269
+ ];
270
+
271
+ rerender(<Graph {...defaultProps} marks={completedMarks} />);
272
+ expect(container.firstChild).toBeInTheDocument();
273
+ });
274
+
275
+ it('does not call onChangeMarks if update is undefined', () => {
276
+ const { container } = render(<Graph {...defaultProps} />);
277
+
278
+ // If update is undefined, onChangeMarks should not be called
279
+ // This is handled internally by the component
280
+ expect(container.firstChild).toBeInTheDocument();
281
+ });
282
+
283
+ it('does not add mark if existing is not found and addIfMissing is false', () => {
284
+ const { container } = render(<Graph {...defaultProps} />);
285
+
286
+ // By default, marks are not added if they don't exist and addIfMissing is false
287
+ expect(container.firstChild).toBeInTheDocument();
288
+ });
289
+
290
+ it('calls onChangeMarks with correct marks when updating', () => {
291
+ const marks = [
292
+ {
293
+ type: 'point',
294
+ x: 2,
295
+ y: 2,
296
+ label: 'Point',
297
+ showLabel: true,
298
+ },
299
+ ];
300
+
301
+ const { rerender } = render(<Graph {...defaultProps} marks={marks} />);
302
+
303
+ const updatedMarks = [
304
+ {
305
+ type: 'point',
306
+ x: 3,
307
+ y: 3,
308
+ label: 'Updated Point',
309
+ showLabel: true,
310
+ },
311
+ ];
312
+
313
+ rerender(<Graph {...defaultProps} marks={updatedMarks} />);
314
+
315
+ // The component handles mark updates internally
316
+ // onChangeMarks is called through user interactions
317
+ });
318
+
319
+ it('preserves other marks when updating a specific mark', () => {
320
+ const marks = [
321
+ {
322
+ type: 'point',
323
+ x: 2,
324
+ y: 2,
325
+ label: 'Point 1',
326
+ showLabel: true,
327
+ },
328
+ {
329
+ type: 'point',
330
+ x: 3,
331
+ y: 3,
332
+ label: 'Point 2',
333
+ showLabel: true,
334
+ },
335
+ ];
336
+
337
+ const { container } = render(<Graph {...defaultProps} marks={marks} />);
338
+ expect(container.firstChild).toBeInTheDocument();
339
+ });
340
+ });
341
+
342
+ describe('getComponent method', () => {
343
+ it('returns null if mark is null', () => {
344
+ const { container } = render(<Graph {...defaultProps} />);
345
+
346
+ // getComponent returns null for null marks
347
+ // This is tested by rendering with undefined marks
348
+ const propsWithoutMarks = { ...defaultProps, marks: [] };
349
+ const { container: emptyContainer } = render(<Graph {...propsWithoutMarks} />);
350
+ expect(emptyContainer.firstChild).toBeInTheDocument();
351
+ });
352
+
353
+ it('returns null if mark is undefined', () => {
354
+ const { container } = render(<Graph {...defaultProps} />);
355
+ expect(container.firstChild).toBeInTheDocument();
356
+ });
357
+
358
+ it('returns null if tools is undefined', () => {
359
+ const propsWithUndefinedTools = { ...defaultProps, tools: undefined };
360
+ const graphInstance = new Graph(propsWithUndefinedTools);
361
+ const mark = { type: 'point', x: 1, y: 1 };
362
+ const component = graphInstance.getComponent(mark);
363
+ expect(component).toBeNull();
364
+ });
365
+
366
+ it('returns null if tools is empty array', () => {
367
+ const propsWithEmptyTools = { ...defaultProps, tools: [] };
368
+ const graphInstance = new Graph(propsWithEmptyTools);
369
+ const mark = { type: 'point', x: 1, y: 1 };
370
+ const component = graphInstance.getComponent(mark);
371
+ expect(component).toBeNull();
372
+ });
373
+
374
+ it('returns null if mark type is not found in tools', () => {
375
+ const graphInstance = new Graph(defaultProps);
376
+ const mark = { type: 'nonexistent-type', x: 1, y: 1 };
377
+ const component = graphInstance.getComponent(mark);
378
+ expect(component).toBeNull();
379
+ });
380
+
381
+ it('returns component if tool has no Component property', () => {
382
+ const toolWithoutComponent = [{ type: 'custom', label: 'Custom' }];
383
+ const propsWithCustomTool = { ...defaultProps, tools: toolWithoutComponent };
384
+ const graphInstance = new Graph(propsWithCustomTool);
385
+ const mark = { type: 'custom', x: 1, y: 1 };
386
+ const component = graphInstance.getComponent(mark);
387
+ expect(component).toBeNull();
388
+ });
389
+
390
+ it('returns correct component for point mark', () => {
391
+ const graphInstance = new Graph(defaultProps);
392
+ const mark = { type: 'point', x: 1, y: 1 };
393
+ const component = graphInstance.getComponent(mark);
394
+ expect(component).toBeTruthy();
395
+ });
396
+
397
+ it('returns correct component for line mark', () => {
398
+ const graphInstance = new Graph(defaultProps);
399
+ const mark = { type: 'line', from: { x: 0, y: 0 }, to: { x: 1, y: 1 } };
400
+ const component = graphInstance.getComponent(mark);
401
+ expect(component).toBeTruthy();
402
+ });
403
+
404
+ it('returns correct component for different mark types', () => {
405
+ const graphInstance = new Graph(defaultProps);
406
+
407
+ const pointMark = { type: 'point', x: 1, y: 1 };
408
+ const lineMark = { type: 'line', from: { x: 0, y: 0 }, to: { x: 1, y: 1 } };
409
+ const circleMark = { type: 'circle', center: { x: 0, y: 0 }, edge: { x: 1, y: 1 } };
410
+
411
+ const pointComponent = graphInstance.getComponent(pointMark);
412
+ const lineComponent = graphInstance.getComponent(lineMark);
413
+ const circleComponent = graphInstance.getComponent(circleMark);
414
+
415
+ expect(pointComponent).toBeTruthy();
416
+ expect(lineComponent).toBeTruthy();
417
+ expect(circleComponent).toBeTruthy();
418
+ });
419
+ });
420
+
421
+ describe('getComponent method - unit tests', () => {
422
+ it('returns null if mark is null', () => {
423
+ const graphInstance = new Graph(defaultProps);
424
+ const component = graphInstance.getComponent(null);
425
+ expect(component).toBeNull();
426
+ });
427
+
428
+ it('returns null if mark is undefined', () => {
429
+ const graphInstance = new Graph(defaultProps);
430
+ const component = graphInstance.getComponent(undefined);
431
+ expect(component).toBeNull();
432
+ });
433
+
434
+ it('returns null if tools is undefined', () => {
435
+ const propsWithUndefinedTools = { ...defaultProps, tools: undefined };
436
+ const graphInstance = new Graph(propsWithUndefinedTools);
437
+ const mark = { type: 'point', x: 1, y: 1 };
438
+ const component = graphInstance.getComponent(mark);
439
+ expect(component).toBeNull();
440
+ });
441
+
442
+ it('returns null if tools is empty array', () => {
443
+ const propsWithEmptyTools = { ...defaultProps, tools: [] };
444
+ const graphInstance = new Graph(propsWithEmptyTools);
445
+ const mark = { type: 'point', x: 1, y: 1 };
446
+ const component = graphInstance.getComponent(mark);
447
+ expect(component).toBeNull();
448
+ });
449
+
450
+ it('returns null if mark type is not found in tools', () => {
451
+ const graphInstance = new Graph(defaultProps);
452
+ const mark = { type: 'nonexistent-type', x: 1, y: 1 };
453
+ const component = graphInstance.getComponent(mark);
454
+ expect(component).toBeNull();
455
+ });
456
+
457
+ it('returns component if tool has no Component property', () => {
458
+ const toolWithoutComponent = [{ type: 'custom', label: 'Custom' }];
459
+ const propsWithCustomTool = { ...defaultProps, tools: toolWithoutComponent };
460
+ const graphInstance = new Graph(propsWithCustomTool);
461
+ const mark = { type: 'custom', x: 1, y: 1 };
462
+ const component = graphInstance.getComponent(mark);
463
+ expect(component).toBeNull();
464
+ });
465
+
466
+ it('returns correct component for point mark', () => {
467
+ const graphInstance = new Graph(defaultProps);
468
+ const mark = { type: 'point', x: 1, y: 1 };
469
+ const component = graphInstance.getComponent(mark);
470
+ expect(component).toBeTruthy();
471
+ });
472
+
473
+ it('returns correct component for line mark', () => {
474
+ const graphInstance = new Graph(defaultProps);
475
+ const mark = { type: 'line', from: { x: 0, y: 0 }, to: { x: 1, y: 1 } };
476
+ const component = graphInstance.getComponent(mark);
477
+ expect(component).toBeTruthy();
478
+ });
479
+
480
+ it('returns correct component for different mark types', () => {
481
+ const graphInstance = new Graph(defaultProps);
482
+
483
+ const pointMark = { type: 'point', x: 1, y: 1 };
484
+ const lineMark = { type: 'line', from: { x: 0, y: 0 }, to: { x: 1, y: 1 } };
485
+ const circleMark = { type: 'circle', center: { x: 0, y: 0 }, edge: { x: 1, y: 1 } };
486
+
487
+ const pointComponent = graphInstance.getComponent(pointMark);
488
+ const lineComponent = graphInstance.getComponent(lineMark);
489
+ const circleComponent = graphInstance.getComponent(circleMark);
490
+
491
+ expect(pointComponent).toBeTruthy();
492
+ expect(lineComponent).toBeTruthy();
493
+ expect(circleComponent).toBeTruthy();
494
+ });
495
+ });
496
+
497
+ describe('updateMarks method - unit tests', () => {
498
+ it('does not update if update parameter is null', () => {
499
+ const onChangeMarks = jest.fn();
500
+ const marks = [{ type: 'point', x: 1, y: 1 }];
501
+ const props = { ...defaultProps, onChangeMarks, marks };
502
+ const graphInstance = new Graph(props);
503
+
504
+ const existing = marks[0];
505
+ graphInstance.updateMarks(existing, null);
506
+
507
+ expect(onChangeMarks).not.toHaveBeenCalled();
508
+ });
509
+
510
+ it('does not update if update parameter is undefined', () => {
511
+ const onChangeMarks = jest.fn();
512
+ const marks = [{ type: 'point', x: 1, y: 1 }];
513
+ const props = { ...defaultProps, onChangeMarks, marks };
514
+ const graphInstance = new Graph(props);
515
+
516
+ const existing = marks[0];
517
+ graphInstance.updateMarks(existing, undefined);
518
+
519
+ expect(onChangeMarks).not.toHaveBeenCalled();
520
+ });
521
+
522
+ it('updates existing mark when found', () => {
523
+ const onChangeMarks = jest.fn();
524
+ const marks = [
525
+ { type: 'point', x: 1, y: 1 },
526
+ { type: 'point', x: 2, y: 2 },
527
+ ];
528
+ const props = { ...defaultProps, onChangeMarks, marks };
529
+ const graphInstance = new Graph(props);
530
+
531
+ const existing = marks[0];
532
+ const updated = { type: 'point', x: 1, y: 3 };
533
+
534
+ graphInstance.updateMarks(existing, updated);
535
+
536
+ expect(onChangeMarks).toHaveBeenCalledTimes(1);
537
+ const updatedMarks = onChangeMarks.mock.calls[0][0];
538
+ expect(updatedMarks[0]).toEqual(updated);
539
+ expect(updatedMarks[1]).toEqual(marks[1]);
540
+ });
541
+
542
+ it('adds mark if addIfMissing is true and mark not found', () => {
543
+ const onChangeMarks = jest.fn();
544
+ const marks = [{ type: 'point', x: 1, y: 1 }];
545
+ const props = { ...defaultProps, onChangeMarks, marks };
546
+ const graphInstance = new Graph(props);
547
+
548
+ const existing = { type: 'point', x: 5, y: 5 };
549
+ const updated = { type: 'point', x: 2, y: 2 };
550
+
551
+ graphInstance.updateMarks(existing, updated, true);
552
+
553
+ expect(onChangeMarks).toHaveBeenCalledTimes(1);
554
+ const updatedMarks = onChangeMarks.mock.calls[0][0];
555
+ expect(updatedMarks.length).toBe(2);
556
+ expect(updatedMarks[1]).toEqual(updated);
557
+ });
558
+
559
+ it('does not add mark if addIfMissing is false and mark not found', () => {
560
+ const onChangeMarks = jest.fn();
561
+ const marks = [{ type: 'point', x: 1, y: 1 }];
562
+ const props = { ...defaultProps, onChangeMarks, marks };
563
+ const graphInstance = new Graph(props);
564
+
565
+ const existing = { type: 'point', x: 5, y: 5 };
566
+ const updated = { type: 'point', x: 2, y: 2 };
567
+
568
+ graphInstance.updateMarks(existing, updated, false);
569
+
570
+ expect(onChangeMarks).not.toHaveBeenCalled();
571
+ });
572
+
573
+ it('does not update if mark is still building', () => {
574
+ const onChangeMarks = jest.fn();
575
+ const marks = [{ type: 'point', x: 1, y: 1, building: true }];
576
+ const props = { ...defaultProps, onChangeMarks, marks };
577
+ const graphInstance = new Graph(props);
578
+
579
+ const existing = marks[0];
580
+ const updated = { type: 'point', x: 2, y: 2 };
581
+
582
+ graphInstance.updateMarks(existing, updated);
583
+
584
+ expect(onChangeMarks).toHaveBeenCalled();
585
+ });
586
+
587
+ it('preserves mark order when updating', () => {
588
+ const onChangeMarks = jest.fn();
589
+ const marks = [
590
+ { type: 'point', x: 1, y: 1 },
591
+ { type: 'point', x: 2, y: 2 },
592
+ { type: 'point', x: 3, y: 3 },
593
+ ];
594
+ const props = { ...defaultProps, onChangeMarks, marks };
595
+ const graphInstance = new Graph(props);
596
+
597
+ const existing = marks[1];
598
+ const updated = { type: 'point', x: 2, y: 5 };
599
+
600
+ graphInstance.updateMarks(existing, updated);
601
+
602
+ const updatedMarks = onChangeMarks.mock.calls[0][0];
603
+ expect(updatedMarks[0]).toEqual(marks[0]);
604
+ expect(updatedMarks[1]).toEqual(updated);
605
+ expect(updatedMarks[2]).toEqual(marks[2]);
606
+ });
607
+
608
+ it('handles empty marks array', () => {
609
+ const onChangeMarks = jest.fn();
610
+ const props = { ...defaultProps, onChangeMarks, marks: [] };
611
+ const graphInstance = new Graph(props);
612
+
613
+ const existing = { type: 'point', x: 1, y: 1 };
614
+ const updated = { type: 'point', x: 2, y: 2 };
615
+
616
+ graphInstance.updateMarks(existing, updated, true);
617
+
618
+ expect(onChangeMarks).toHaveBeenCalledTimes(1);
619
+ const updatedMarks = onChangeMarks.mock.calls[0][0];
620
+ expect(updatedMarks.length).toBe(1);
621
+ expect(updatedMarks[0]).toEqual(updated);
622
+ });
623
+ });
624
+
625
+ describe('updateMarks and getComponent integration', () => {
626
+ it('updates marks and renders with correct components', () => {
627
+ const initialMarks = [
628
+ {
629
+ type: 'point',
630
+ x: 2,
631
+ y: 2,
632
+ label: 'Point',
633
+ showLabel: true,
634
+ },
635
+ ];
636
+
637
+ const { rerender, container } = render(<Graph {...defaultProps} marks={initialMarks} />);
638
+
639
+ const updatedMarks = [
640
+ {
641
+ type: 'line',
642
+ from: { x: 0, y: 0 },
643
+ to: { x: 1, y: 1 },
644
+ label: 'Line',
645
+ },
646
+ ];
647
+
648
+ rerender(<Graph {...defaultProps} marks={updatedMarks} />);
649
+
650
+ expect(container.querySelector('#marks')).toBeInTheDocument();
651
+ });
652
+
653
+ it('handles adding new marks with different types', () => {
654
+ const initialMarks = [
655
+ {
656
+ type: 'point',
657
+ x: 2,
658
+ y: 2,
659
+ label: 'Point',
660
+ showLabel: true,
661
+ },
662
+ ];
663
+
664
+ const { rerender, container } = render(<Graph {...defaultProps} marks={initialMarks} />);
665
+
666
+ const updatedMarks = [
667
+ {
668
+ type: 'point',
669
+ x: 2,
670
+ y: 2,
671
+ label: 'Point',
672
+ showLabel: true,
673
+ },
674
+ {
675
+ type: 'line',
676
+ from: { x: 0, y: 0 },
677
+ to: { x: 1, y: 1 },
678
+ label: 'Line',
679
+ },
680
+ ];
681
+
682
+ rerender(<Graph {...defaultProps} marks={updatedMarks} />);
683
+
684
+ expect(container.querySelector('#marks')).toBeInTheDocument();
685
+ });
686
+
687
+ it('handles removing marks', () => {
688
+ const initialMarks = [
689
+ {
690
+ type: 'point',
691
+ x: 2,
692
+ y: 2,
693
+ label: 'Point',
694
+ showLabel: true,
695
+ },
696
+ {
697
+ type: 'line',
698
+ from: { x: 0, y: 0 },
699
+ to: { x: 1, y: 1 },
700
+ label: 'Line',
701
+ },
702
+ ];
703
+
704
+ const { rerender, container } = render(<Graph {...defaultProps} marks={initialMarks} />);
705
+
706
+ const updatedMarks = [
707
+ {
708
+ type: 'point',
709
+ x: 2,
710
+ y: 2,
711
+ label: 'Point',
712
+ showLabel: true,
713
+ },
714
+ ];
715
+
716
+ rerender(<Graph {...defaultProps} marks={updatedMarks} />);
717
+
718
+ expect(container.querySelector('#marks')).toBeInTheDocument();
719
+ });
720
+ });
166
721
  });