@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.
- package/dist/components/GraphRenderer.d.ts +10 -0
- package/dist/components/GraphRenderer.d.ts.map +1 -1
- package/dist/components/GraphRenderer.js +120 -21
- package/dist/components/GraphRenderer.js.map +1 -1
- package/dist/components/SelectionSidebar.d.ts +18 -0
- package/dist/components/SelectionSidebar.d.ts.map +1 -0
- package/dist/components/SelectionSidebar.js +183 -0
- package/dist/components/SelectionSidebar.js.map +1 -0
- package/dist/nodes/CustomNode.d.ts.map +1 -1
- package/dist/nodes/CustomNode.js +28 -11
- package/dist/nodes/CustomNode.js.map +1 -1
- package/package.json +1 -1
- package/src/components/GraphRenderer.tsx +167 -23
- package/src/components/PendingChanges.test.tsx +433 -0
- package/src/components/SelectionSidebar.tsx +341 -0
- package/src/nodes/CustomNode.tsx +43 -11
|
@@ -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
|
+
});
|