@principal-ai/principal-view-react 0.6.16 → 0.6.17

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.
@@ -0,0 +1,433 @@
1
+ import { describe, expect, test, beforeEach, beforeAll, mock } from 'bun:test';
2
+ import React, { createRef } from 'react';
3
+ import { render, fireEvent, act } from '@testing-library/react';
4
+ import { GraphRenderer, type GraphRendererHandle, type PendingChanges } from './GraphRenderer';
5
+ import type { ExtendedCanvas } from '@principal-ai/principal-view-core';
6
+ import { ThemeProvider, defaultEditorTheme } from '@principal-ade/industry-theme';
7
+ import { Window } from 'happy-dom';
8
+
9
+ // Setup DOM environment
10
+ const window = new Window();
11
+ const document = window.document;
12
+
13
+ // Set up global DOM objects
14
+ globalThis.document = document as any;
15
+ globalThis.window = window as any;
16
+ globalThis.navigator = window.navigator as any;
17
+ globalThis.HTMLElement = window.HTMLElement as any;
18
+ globalThis.Element = window.Element as any;
19
+ globalThis.Node = window.Node as any;
20
+ globalThis.Text = window.Text as any;
21
+ globalThis.DocumentFragment = window.DocumentFragment as any;
22
+ globalThis.Event = window.Event as any;
23
+ globalThis.CustomEvent = window.CustomEvent as any;
24
+ globalThis.MouseEvent = window.MouseEvent as any;
25
+ globalThis.KeyboardEvent = window.KeyboardEvent as any;
26
+ globalThis.getComputedStyle = window.getComputedStyle.bind(window) as any;
27
+ globalThis.localStorage = window.localStorage as any;
28
+ globalThis.sessionStorage = window.sessionStorage as any;
29
+
30
+ // Mock ResizeObserver for ReactFlow
31
+ globalThis.ResizeObserver = class ResizeObserver {
32
+ observe() {}
33
+ unobserve() {}
34
+ disconnect() {}
35
+ };
36
+
37
+ // Mock matchMedia
38
+ globalThis.matchMedia = mock(() => ({
39
+ matches: false,
40
+ addListener: () => {},
41
+ removeListener: () => {},
42
+ addEventListener: () => {},
43
+ removeEventListener: () => {},
44
+ dispatchEvent: () => false,
45
+ })) as any;
46
+
47
+ // Helper to wrap component with ThemeProvider
48
+ const renderWithTheme = (ui: React.ReactElement) => {
49
+ return render(
50
+ <ThemeProvider theme={defaultEditorTheme}>
51
+ {ui}
52
+ </ThemeProvider>
53
+ );
54
+ };
55
+
56
+ describe('PendingChanges', () => {
57
+ // Sample canvas for testing
58
+ const createTestCanvas = (): ExtendedCanvas => ({
59
+ nodes: [
60
+ {
61
+ id: 'node-1',
62
+ type: 'text',
63
+ x: 100,
64
+ y: 100,
65
+ width: 150,
66
+ height: 80,
67
+ text: 'Node 1',
68
+ pv: {
69
+ nodeType: 'service',
70
+ },
71
+ },
72
+ {
73
+ id: 'node-2',
74
+ type: 'text',
75
+ x: 300,
76
+ y: 100,
77
+ width: 150,
78
+ height: 80,
79
+ text: 'Node 2',
80
+ pv: {
81
+ nodeType: 'database',
82
+ },
83
+ },
84
+ {
85
+ id: 'node-3',
86
+ type: 'text',
87
+ x: 200,
88
+ y: 300,
89
+ width: 150,
90
+ height: 80,
91
+ text: 'Node 3',
92
+ pv: {
93
+ nodeType: 'service',
94
+ },
95
+ },
96
+ ],
97
+ edges: [
98
+ {
99
+ id: 'edge-1',
100
+ fromNode: 'node-1',
101
+ toNode: 'node-2',
102
+ pv: {
103
+ edgeType: 'connection',
104
+ },
105
+ },
106
+ ],
107
+ pv: {
108
+ name: 'Test Canvas',
109
+ version: '1.0.0',
110
+ nodeTypes: {
111
+ service: {
112
+ shape: 'rectangle',
113
+ color: '#4CAF50',
114
+ },
115
+ database: {
116
+ shape: 'circle',
117
+ color: '#2196F3',
118
+ },
119
+ },
120
+ edgeTypes: {
121
+ connection: {
122
+ style: 'solid',
123
+ directed: true,
124
+ },
125
+ },
126
+ },
127
+ });
128
+
129
+ describe('PendingChanges Interface', () => {
130
+ test('should have correct structure with all required fields', () => {
131
+ const ref = createRef<GraphRendererHandle>();
132
+
133
+ renderWithTheme(
134
+ <GraphRenderer
135
+ ref={ref}
136
+ canvas={createTestCanvas()}
137
+ editable={true}
138
+ />
139
+ );
140
+
141
+ const pendingChanges = ref.current?.getPendingChanges();
142
+
143
+ expect(pendingChanges).toBeDefined();
144
+ expect(pendingChanges).toHaveProperty('positionChanges');
145
+ expect(pendingChanges).toHaveProperty('dimensionChanges');
146
+ expect(pendingChanges).toHaveProperty('nodeUpdates');
147
+ expect(pendingChanges).toHaveProperty('deletedNodeIds');
148
+ expect(pendingChanges).toHaveProperty('createdEdges');
149
+ expect(pendingChanges).toHaveProperty('deletedEdges');
150
+ expect(pendingChanges).toHaveProperty('hasChanges');
151
+ });
152
+
153
+ test('should return empty arrays/false when no changes made', () => {
154
+ const ref = createRef<GraphRendererHandle>();
155
+
156
+ renderWithTheme(
157
+ <GraphRenderer
158
+ ref={ref}
159
+ canvas={createTestCanvas()}
160
+ editable={true}
161
+ />
162
+ );
163
+
164
+ const pendingChanges = ref.current?.getPendingChanges();
165
+
166
+ expect(pendingChanges?.positionChanges).toEqual([]);
167
+ expect(pendingChanges?.dimensionChanges).toEqual([]);
168
+ expect(pendingChanges?.nodeUpdates).toEqual([]);
169
+ expect(pendingChanges?.deletedNodeIds).toEqual([]);
170
+ expect(pendingChanges?.createdEdges).toEqual([]);
171
+ expect(pendingChanges?.deletedEdges).toEqual([]);
172
+ expect(pendingChanges?.hasChanges).toBe(false);
173
+ });
174
+
175
+ test('hasUnsavedChanges should return false initially', () => {
176
+ const ref = createRef<GraphRendererHandle>();
177
+
178
+ renderWithTheme(
179
+ <GraphRenderer
180
+ ref={ref}
181
+ canvas={createTestCanvas()}
182
+ editable={true}
183
+ />
184
+ );
185
+
186
+ expect(ref.current?.hasUnsavedChanges()).toBe(false);
187
+ });
188
+ });
189
+
190
+ describe('Position Changes', () => {
191
+ test('positionChanges should have correct structure', () => {
192
+ const ref = createRef<GraphRendererHandle>();
193
+
194
+ renderWithTheme(
195
+ <GraphRenderer
196
+ ref={ref}
197
+ canvas={createTestCanvas()}
198
+ editable={true}
199
+ />
200
+ );
201
+
202
+ const pendingChanges = ref.current?.getPendingChanges();
203
+
204
+ // positionChanges should be an array
205
+ expect(Array.isArray(pendingChanges?.positionChanges)).toBe(true);
206
+
207
+ // Each position change should have nodeId and position
208
+ // (empty array initially, but structure is correct)
209
+ expect(pendingChanges?.positionChanges).toEqual([]);
210
+ });
211
+ });
212
+
213
+ describe('Dimension Changes', () => {
214
+ test('dimensionChanges should have correct structure', () => {
215
+ const ref = createRef<GraphRendererHandle>();
216
+
217
+ renderWithTheme(
218
+ <GraphRenderer
219
+ ref={ref}
220
+ canvas={createTestCanvas()}
221
+ editable={true}
222
+ />
223
+ );
224
+
225
+ const pendingChanges = ref.current?.getPendingChanges();
226
+
227
+ // dimensionChanges should be an array
228
+ expect(Array.isArray(pendingChanges?.dimensionChanges)).toBe(true);
229
+
230
+ // Each dimension change should have nodeId and dimensions
231
+ // (empty array initially, but structure is correct)
232
+ expect(pendingChanges?.dimensionChanges).toEqual([]);
233
+ });
234
+
235
+ test('dimensionChanges array items should have nodeId and dimensions fields', () => {
236
+ // This test validates the type structure
237
+ const exampleDimensionChange = {
238
+ nodeId: 'node-1',
239
+ dimensions: { width: 200, height: 100 },
240
+ };
241
+
242
+ expect(exampleDimensionChange).toHaveProperty('nodeId');
243
+ expect(exampleDimensionChange).toHaveProperty('dimensions');
244
+ expect(exampleDimensionChange.dimensions).toHaveProperty('width');
245
+ expect(exampleDimensionChange.dimensions).toHaveProperty('height');
246
+ });
247
+ });
248
+
249
+ describe('Reset Edit State', () => {
250
+ test('resetEditState should clear all pending changes', () => {
251
+ const ref = createRef<GraphRendererHandle>();
252
+
253
+ renderWithTheme(
254
+ <GraphRenderer
255
+ ref={ref}
256
+ canvas={createTestCanvas()}
257
+ editable={true}
258
+ />
259
+ );
260
+
261
+ // Reset should work even when there are no changes
262
+ ref.current?.resetEditState();
263
+
264
+ const pendingChanges = ref.current?.getPendingChanges();
265
+ expect(pendingChanges?.hasChanges).toBe(false);
266
+ expect(ref.current?.hasUnsavedChanges()).toBe(false);
267
+ });
268
+ });
269
+
270
+ describe('onPendingChangesChange Callback', () => {
271
+ test('should call onPendingChangesChange with false initially', () => {
272
+ const ref = createRef<GraphRendererHandle>();
273
+ const onPendingChangesChange = mock(() => {});
274
+
275
+ renderWithTheme(
276
+ <GraphRenderer
277
+ ref={ref}
278
+ canvas={createTestCanvas()}
279
+ editable={true}
280
+ onPendingChangesChange={onPendingChangesChange}
281
+ />
282
+ );
283
+
284
+ // Initially should not have changes
285
+ expect(ref.current?.hasUnsavedChanges()).toBe(false);
286
+ });
287
+ });
288
+
289
+ describe('Non-Editable Mode', () => {
290
+ test('should still provide getPendingChanges in non-editable mode', () => {
291
+ const ref = createRef<GraphRendererHandle>();
292
+
293
+ renderWithTheme(
294
+ <GraphRenderer
295
+ ref={ref}
296
+ canvas={createTestCanvas()}
297
+ editable={false}
298
+ />
299
+ );
300
+
301
+ const pendingChanges = ref.current?.getPendingChanges();
302
+
303
+ expect(pendingChanges).toBeDefined();
304
+ expect(pendingChanges?.hasChanges).toBe(false);
305
+ });
306
+ });
307
+
308
+ describe('Node Updates', () => {
309
+ test('nodeUpdates should have correct structure', () => {
310
+ const ref = createRef<GraphRendererHandle>();
311
+
312
+ renderWithTheme(
313
+ <GraphRenderer
314
+ ref={ref}
315
+ canvas={createTestCanvas()}
316
+ editable={true}
317
+ />
318
+ );
319
+
320
+ const pendingChanges = ref.current?.getPendingChanges();
321
+
322
+ // nodeUpdates should be an array
323
+ expect(Array.isArray(pendingChanges?.nodeUpdates)).toBe(true);
324
+ });
325
+ });
326
+
327
+ describe('Edge Operations', () => {
328
+ test('createdEdges should have correct structure', () => {
329
+ const ref = createRef<GraphRendererHandle>();
330
+
331
+ renderWithTheme(
332
+ <GraphRenderer
333
+ ref={ref}
334
+ canvas={createTestCanvas()}
335
+ editable={true}
336
+ />
337
+ );
338
+
339
+ const pendingChanges = ref.current?.getPendingChanges();
340
+
341
+ // createdEdges should be an array
342
+ expect(Array.isArray(pendingChanges?.createdEdges)).toBe(true);
343
+ });
344
+
345
+ test('deletedEdges should have correct structure', () => {
346
+ const ref = createRef<GraphRendererHandle>();
347
+
348
+ renderWithTheme(
349
+ <GraphRenderer
350
+ ref={ref}
351
+ canvas={createTestCanvas()}
352
+ editable={true}
353
+ />
354
+ );
355
+
356
+ const pendingChanges = ref.current?.getPendingChanges();
357
+
358
+ // deletedEdges should be an array
359
+ expect(Array.isArray(pendingChanges?.deletedEdges)).toBe(true);
360
+ });
361
+ });
362
+
363
+ describe('Deleted Nodes', () => {
364
+ test('deletedNodeIds should have correct structure', () => {
365
+ const ref = createRef<GraphRendererHandle>();
366
+
367
+ renderWithTheme(
368
+ <GraphRenderer
369
+ ref={ref}
370
+ canvas={createTestCanvas()}
371
+ editable={true}
372
+ />
373
+ );
374
+
375
+ const pendingChanges = ref.current?.getPendingChanges();
376
+
377
+ // deletedNodeIds should be an array
378
+ expect(Array.isArray(pendingChanges?.deletedNodeIds)).toBe(true);
379
+ });
380
+ });
381
+ });
382
+
383
+ describe('PendingChanges Type Validation', () => {
384
+ test('NodePositionChange should have correct shape', () => {
385
+ const positionChange = {
386
+ nodeId: 'test-node',
387
+ position: { x: 100, y: 200 },
388
+ };
389
+
390
+ expect(positionChange.nodeId).toBe('test-node');
391
+ expect(positionChange.position.x).toBe(100);
392
+ expect(positionChange.position.y).toBe(200);
393
+ });
394
+
395
+ test('NodeDimensionChange should have correct shape', () => {
396
+ const dimensionChange = {
397
+ nodeId: 'test-node',
398
+ dimensions: { width: 150, height: 100 },
399
+ };
400
+
401
+ expect(dimensionChange.nodeId).toBe('test-node');
402
+ expect(dimensionChange.dimensions.width).toBe(150);
403
+ expect(dimensionChange.dimensions.height).toBe(100);
404
+ });
405
+
406
+ test('CreatedEdge should have correct shape', () => {
407
+ const createdEdge = {
408
+ from: 'node-1',
409
+ to: 'node-2',
410
+ type: 'connection',
411
+ sourceHandle: 'right-out',
412
+ targetHandle: 'left',
413
+ };
414
+
415
+ expect(createdEdge.from).toBe('node-1');
416
+ expect(createdEdge.to).toBe('node-2');
417
+ expect(createdEdge.type).toBe('connection');
418
+ expect(createdEdge.sourceHandle).toBe('right-out');
419
+ expect(createdEdge.targetHandle).toBe('left');
420
+ });
421
+
422
+ test('DeletedEdge should have correct shape', () => {
423
+ const deletedEdge = {
424
+ from: 'node-1',
425
+ to: 'node-2',
426
+ type: 'connection',
427
+ };
428
+
429
+ expect(deletedEdge.from).toBe('node-1');
430
+ expect(deletedEdge.to).toBe('node-2');
431
+ expect(deletedEdge.type).toBe('connection');
432
+ });
433
+ });