@nyaruka/temba-components 0.127.0 → 0.129.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +25 -0
- package/demo/chart/horizontal-demo.html +81 -0
- package/demo/components/datepicker/example.html +63 -0
- package/demo/components/datepicker/range-picker-demo.html +161 -0
- package/demo/data/flows/sample-flow.json +127 -100
- package/demo/index.html +8 -0
- package/demo/static/css/prism.css +2 -0
- package/demo/static/js/prism-loader.js +12 -0
- package/demo/sticky-note-demo.html +152 -0
- package/demo/styles.css +71 -1
- package/dist/locales/es.js +5 -5
- package/dist/locales/es.js.map +1 -1
- package/dist/locales/fr.js +5 -5
- package/dist/locales/fr.js.map +1 -1
- package/dist/locales/locale-codes.js +11 -2
- package/dist/locales/locale-codes.js.map +1 -1
- package/dist/locales/pt.js +5 -5
- package/dist/locales/pt.js.map +1 -1
- package/dist/temba-components.js +509 -87
- package/dist/temba-components.js.map +1 -1
- package/out-tsc/src/chart/TembaChart.js +136 -62
- package/out-tsc/src/chart/TembaChart.js.map +1 -1
- package/out-tsc/src/datepicker/DatePicker.js +11 -1
- package/out-tsc/src/datepicker/DatePicker.js.map +1 -1
- package/out-tsc/src/datepicker/RangePicker.js +595 -0
- package/out-tsc/src/datepicker/RangePicker.js.map +1 -0
- package/out-tsc/src/flow/Editor.js +210 -1
- package/out-tsc/src/flow/Editor.js.map +1 -1
- package/out-tsc/src/flow/EditorNode.js +98 -139
- package/out-tsc/src/flow/EditorNode.js.map +1 -1
- package/out-tsc/src/flow/StickyNote.js +272 -0
- package/out-tsc/src/flow/StickyNote.js.map +1 -0
- package/out-tsc/src/interfaces.js +1 -0
- package/out-tsc/src/interfaces.js.map +1 -1
- package/out-tsc/src/list/RunList.js +2 -1
- package/out-tsc/src/list/RunList.js.map +1 -1
- package/out-tsc/src/list/SortableList.js +9 -0
- package/out-tsc/src/list/SortableList.js.map +1 -1
- package/out-tsc/src/locales/es.js +5 -5
- package/out-tsc/src/locales/es.js.map +1 -1
- package/out-tsc/src/locales/fr.js +5 -5
- package/out-tsc/src/locales/fr.js.map +1 -1
- package/out-tsc/src/locales/locale-codes.js +11 -2
- package/out-tsc/src/locales/locale-codes.js.map +1 -1
- package/out-tsc/src/locales/pt.js +5 -5
- package/out-tsc/src/locales/pt.js.map +1 -1
- package/out-tsc/src/store/AppState.js +33 -0
- package/out-tsc/src/store/AppState.js.map +1 -1
- package/out-tsc/src/vectoricon/index.js +2 -1
- package/out-tsc/src/vectoricon/index.js.map +1 -1
- package/out-tsc/temba-modules.js +5 -1
- package/out-tsc/temba-modules.js.map +1 -1
- package/out-tsc/test/temba-chart.test.js +36 -0
- package/out-tsc/test/temba-chart.test.js.map +1 -1
- package/out-tsc/test/temba-datepicker.test.js +1 -1
- package/out-tsc/test/temba-datepicker.test.js.map +1 -1
- package/out-tsc/test/temba-flow-editor-node.test.js +249 -5
- package/out-tsc/test/temba-flow-editor-node.test.js.map +1 -1
- package/out-tsc/test/temba-range-picker.test.js +123 -0
- package/out-tsc/test/temba-range-picker.test.js.map +1 -0
- package/out-tsc/test/temba-select.test.js +10 -16
- package/out-tsc/test/temba-select.test.js.map +1 -1
- package/out-tsc/test/temba-webchat.test.js +4 -0
- package/out-tsc/test/temba-webchat.test.js.map +1 -1
- package/out-tsc/test/utils.test.js +62 -0
- package/out-tsc/test/utils.test.js.map +1 -1
- package/package.json +1 -1
- package/screenshots/truth/datepicker/range-picker-all.png +0 -0
- package/screenshots/truth/datepicker/range-picker-button-states.png +0 -0
- package/screenshots/truth/datepicker/range-picker-default.png +0 -0
- package/screenshots/truth/datepicker/range-picker-editing-start.png +0 -0
- package/screenshots/truth/datepicker/range-picker-initial-values.png +0 -0
- package/screenshots/truth/datepicker/range-picker-min-max.png +0 -0
- package/screenshots/truth/datepicker/range-picker-week.png +0 -0
- package/screenshots/truth/datepicker/range-picker-year.png +0 -0
- package/screenshots/truth/sticky-note/blue.png +0 -0
- package/screenshots/truth/sticky-note/gray.png +0 -0
- package/screenshots/truth/sticky-note/green.png +0 -0
- package/screenshots/truth/sticky-note/pink.png +0 -0
- package/screenshots/truth/sticky-note/yellow.png +0 -0
- package/screenshots/truth/webchat/connected-state.png +0 -0
- package/src/chart/TembaChart.ts +144 -66
- package/src/datepicker/DatePicker.ts +9 -1
- package/src/datepicker/RangePicker.ts +602 -0
- package/src/flow/Editor.ts +252 -2
- package/src/flow/EditorNode.ts +98 -156
- package/src/flow/StickyNote.ts +284 -0
- package/src/interfaces.ts +2 -1
- package/src/list/RunList.ts +2 -1
- package/src/list/SortableList.ts +11 -0
- package/src/locales/es.ts +18 -13
- package/src/locales/fr.ts +18 -13
- package/src/locales/locale-codes.ts +11 -2
- package/src/locales/pt.ts +18 -13
- package/src/store/AppState.ts +51 -1
- package/src/store/flow-definition.d.ts +8 -0
- package/src/vectoricon/index.ts +2 -1
- package/static/svg/index.pdf +137 -0
- package/temba-modules.ts +5 -1
- package/test/temba-chart.test.ts +47 -0
- package/test/temba-datepicker.test.ts +1 -1
- package/test/temba-flow-editor-node.test.ts +322 -6
- package/test/temba-range-picker.test.ts +193 -0
- package/test/temba-select.test.ts +11 -19
- package/test/temba-webchat.test.ts +7 -0
- package/test/utils.test.ts +98 -0
- package/web-dev-server.config.mjs +30 -22
- package/web-test-runner.config.mjs +2 -0
- package/demo/datepicker/example.html +0 -69
|
@@ -1,8 +1,8 @@
|
|
|
1
|
+
import '../temba-modules';
|
|
1
2
|
import { html, fixture, expect } from '@open-wc/testing';
|
|
2
3
|
import { EditorNode } from '../src/flow/EditorNode';
|
|
3
4
|
import { stub, restore } from 'sinon';
|
|
4
|
-
|
|
5
|
-
customElements.define('temba-editor-node', EditorNode);
|
|
5
|
+
import { CustomEventType } from '../src/interfaces';
|
|
6
6
|
describe('EditorNode', () => {
|
|
7
7
|
let editorNode;
|
|
8
8
|
let mockPlumber;
|
|
@@ -39,7 +39,7 @@ describe('EditorNode', () => {
|
|
|
39
39
|
text: 'Test message',
|
|
40
40
|
quick_replies: []
|
|
41
41
|
};
|
|
42
|
-
const result = editorNode.renderAction(mockNode, action);
|
|
42
|
+
const result = editorNode.renderAction(mockNode, action, 0);
|
|
43
43
|
expect(result).to.exist;
|
|
44
44
|
});
|
|
45
45
|
it('renders action with unknown config', () => {
|
|
@@ -52,7 +52,7 @@ describe('EditorNode', () => {
|
|
|
52
52
|
type: 'unknown_action',
|
|
53
53
|
uuid: 'action-1'
|
|
54
54
|
};
|
|
55
|
-
const result = editorNode.renderAction(mockNode, action);
|
|
55
|
+
const result = editorNode.renderAction(mockNode, action, 1);
|
|
56
56
|
expect(result).to.exist;
|
|
57
57
|
});
|
|
58
58
|
});
|
|
@@ -258,7 +258,7 @@ describe('EditorNode', () => {
|
|
|
258
258
|
// Test individual render methods work
|
|
259
259
|
editorNode = new EditorNode();
|
|
260
260
|
// Test renderAction
|
|
261
|
-
const actionResult = editorNode.renderAction(mockNode, mockNode.actions[0]);
|
|
261
|
+
const actionResult = editorNode.renderAction(mockNode, mockNode.actions[0], 0);
|
|
262
262
|
expect(actionResult).to.exist;
|
|
263
263
|
// Test renderExit
|
|
264
264
|
const exitResult = editorNode.renderExit(mockNode.exits[0]);
|
|
@@ -269,5 +269,249 @@ describe('EditorNode', () => {
|
|
|
269
269
|
expect(mockNode.exits).to.have.length(1);
|
|
270
270
|
});
|
|
271
271
|
});
|
|
272
|
+
describe('drag and drop functionality', () => {
|
|
273
|
+
let editorNode;
|
|
274
|
+
beforeEach(() => {
|
|
275
|
+
editorNode = new EditorNode();
|
|
276
|
+
});
|
|
277
|
+
it('renders actions with sortable class and proper IDs', async () => {
|
|
278
|
+
const mockNode = {
|
|
279
|
+
uuid: 'sortable-test-node',
|
|
280
|
+
actions: [
|
|
281
|
+
{
|
|
282
|
+
type: 'send_msg',
|
|
283
|
+
uuid: 'action-1',
|
|
284
|
+
text: 'Hello',
|
|
285
|
+
quick_replies: []
|
|
286
|
+
},
|
|
287
|
+
{
|
|
288
|
+
type: 'send_msg',
|
|
289
|
+
uuid: 'action-2',
|
|
290
|
+
text: 'World',
|
|
291
|
+
quick_replies: []
|
|
292
|
+
}
|
|
293
|
+
],
|
|
294
|
+
exits: []
|
|
295
|
+
};
|
|
296
|
+
// Test that renderAction includes sortable class and proper ID
|
|
297
|
+
const result1 = editorNode.renderAction(mockNode, mockNode.actions[0], 0);
|
|
298
|
+
const result2 = editorNode.renderAction(mockNode, mockNode.actions[1], 1);
|
|
299
|
+
expect(result1).to.exist;
|
|
300
|
+
expect(result2).to.exist;
|
|
301
|
+
// Render the template to check the actual DOM
|
|
302
|
+
const container1 = await fixture(html `<div>${result1}</div>`);
|
|
303
|
+
const container2 = await fixture(html `<div>${result2}</div>`);
|
|
304
|
+
const actionElement1 = container1.querySelector('.action');
|
|
305
|
+
const actionElement2 = container2.querySelector('.action');
|
|
306
|
+
expect(actionElement1).to.exist;
|
|
307
|
+
expect(actionElement1 === null || actionElement1 === void 0 ? void 0 : actionElement1.classList.contains('sortable')).to.be.true;
|
|
308
|
+
expect(actionElement1 === null || actionElement1 === void 0 ? void 0 : actionElement1.id).to.equal('action-0');
|
|
309
|
+
expect(actionElement2).to.exist;
|
|
310
|
+
expect(actionElement2 === null || actionElement2 === void 0 ? void 0 : actionElement2.classList.contains('sortable')).to.be.true;
|
|
311
|
+
expect(actionElement2 === null || actionElement2 === void 0 ? void 0 : actionElement2.id).to.equal('action-1');
|
|
312
|
+
});
|
|
313
|
+
it('includes drag handle in rendered actions', async () => {
|
|
314
|
+
const mockNode = {
|
|
315
|
+
uuid: 'drag-handle-test',
|
|
316
|
+
actions: [
|
|
317
|
+
{
|
|
318
|
+
type: 'send_msg',
|
|
319
|
+
uuid: 'action-1',
|
|
320
|
+
text: 'Hello',
|
|
321
|
+
quick_replies: []
|
|
322
|
+
}
|
|
323
|
+
],
|
|
324
|
+
exits: []
|
|
325
|
+
};
|
|
326
|
+
let editorNode = await fixture(html `<temba-flow-node
|
|
327
|
+
.node=${mockNode}
|
|
328
|
+
.ui=${{ position: { left: 0, top: 0 } }}
|
|
329
|
+
></temba-flow-node>`);
|
|
330
|
+
// No drag handle should be present if only one action
|
|
331
|
+
let dragHandle = editorNode.querySelector('.drag-handle');
|
|
332
|
+
expect(dragHandle).to.not.exist;
|
|
333
|
+
// Now add a second action to verify drag handle appears
|
|
334
|
+
mockNode.actions.push({
|
|
335
|
+
type: 'send_msg',
|
|
336
|
+
uuid: 'action-2',
|
|
337
|
+
text: 'World',
|
|
338
|
+
quick_replies: []
|
|
339
|
+
});
|
|
340
|
+
editorNode = await fixture(html `<temba-flow-node
|
|
341
|
+
.node=${mockNode}
|
|
342
|
+
.ui=${{ position: { left: 0, top: 0 } }}
|
|
343
|
+
></temba-flow-node>`);
|
|
344
|
+
dragHandle = editorNode.querySelector('.drag-handle');
|
|
345
|
+
expect(dragHandle).to.exist;
|
|
346
|
+
});
|
|
347
|
+
it('renders SortableList when actions are present', async () => {
|
|
348
|
+
const mockNode = {
|
|
349
|
+
uuid: 'sortable-list-test',
|
|
350
|
+
actions: [
|
|
351
|
+
{
|
|
352
|
+
type: 'send_msg',
|
|
353
|
+
uuid: 'action-1',
|
|
354
|
+
text: 'Hello',
|
|
355
|
+
quick_replies: []
|
|
356
|
+
},
|
|
357
|
+
{
|
|
358
|
+
type: 'send_msg',
|
|
359
|
+
uuid: 'action-2',
|
|
360
|
+
text: 'World',
|
|
361
|
+
quick_replies: []
|
|
362
|
+
}
|
|
363
|
+
],
|
|
364
|
+
exits: []
|
|
365
|
+
};
|
|
366
|
+
const mockUI = {
|
|
367
|
+
position: { left: 100, top: 200 },
|
|
368
|
+
type: 'execute_actions'
|
|
369
|
+
};
|
|
370
|
+
// Set properties on the component
|
|
371
|
+
editorNode.node = mockNode;
|
|
372
|
+
editorNode.ui = mockUI;
|
|
373
|
+
const renderResult = editorNode.render();
|
|
374
|
+
// Render the template to check the actual DOM
|
|
375
|
+
const container = await fixture(html `<div>${renderResult}</div>`);
|
|
376
|
+
const sortableList = container.querySelector('temba-sortable-list');
|
|
377
|
+
expect(sortableList).to.exist;
|
|
378
|
+
});
|
|
379
|
+
it('does not render SortableList when no actions', async () => {
|
|
380
|
+
const mockNode = {
|
|
381
|
+
uuid: 'no-actions-test',
|
|
382
|
+
actions: [],
|
|
383
|
+
exits: [{ uuid: 'exit-1' }]
|
|
384
|
+
};
|
|
385
|
+
const mockUI = {
|
|
386
|
+
position: { left: 100, top: 200 },
|
|
387
|
+
type: 'execute_actions'
|
|
388
|
+
};
|
|
389
|
+
// Set properties on the component
|
|
390
|
+
editorNode.node = mockNode;
|
|
391
|
+
editorNode.ui = mockUI;
|
|
392
|
+
const renderResult = editorNode.render();
|
|
393
|
+
// Check that template does not include temba-sortable-list
|
|
394
|
+
expect(renderResult.strings.join('')).to.not.contain('temba-sortable-list');
|
|
395
|
+
});
|
|
396
|
+
it('handles order changed events correctly', async () => {
|
|
397
|
+
const mockNode = {
|
|
398
|
+
uuid: 'order-test',
|
|
399
|
+
actions: [
|
|
400
|
+
{
|
|
401
|
+
type: 'send_msg',
|
|
402
|
+
uuid: 'action-1',
|
|
403
|
+
text: 'First',
|
|
404
|
+
quick_replies: []
|
|
405
|
+
},
|
|
406
|
+
{
|
|
407
|
+
type: 'send_msg',
|
|
408
|
+
uuid: 'action-2',
|
|
409
|
+
text: 'Second',
|
|
410
|
+
quick_replies: []
|
|
411
|
+
},
|
|
412
|
+
{
|
|
413
|
+
type: 'send_msg',
|
|
414
|
+
uuid: 'action-3',
|
|
415
|
+
text: 'Third',
|
|
416
|
+
quick_replies: []
|
|
417
|
+
}
|
|
418
|
+
],
|
|
419
|
+
exits: []
|
|
420
|
+
};
|
|
421
|
+
editorNode.node = mockNode;
|
|
422
|
+
// Create a mock order changed event (swap first and last actions)
|
|
423
|
+
const orderChangedEvent = new CustomEvent(CustomEventType.OrderChanged, {
|
|
424
|
+
detail: { swap: [0, 2] }
|
|
425
|
+
});
|
|
426
|
+
// Call the handler directly
|
|
427
|
+
editorNode.handleActionOrderChanged(orderChangedEvent);
|
|
428
|
+
// Verify the actions were reordered correctly
|
|
429
|
+
expect(editorNode.node.actions).to.have.length(3);
|
|
430
|
+
expect(editorNode.node.actions[0].text).to.equal('Second');
|
|
431
|
+
expect(editorNode.node.actions[1].text).to.equal('Third');
|
|
432
|
+
expect(editorNode.node.actions[2].text).to.equal('First');
|
|
433
|
+
});
|
|
434
|
+
it('preserves action data during reordering', () => {
|
|
435
|
+
const mockNode = {
|
|
436
|
+
uuid: 'preserve-test',
|
|
437
|
+
actions: [
|
|
438
|
+
{
|
|
439
|
+
type: 'send_msg',
|
|
440
|
+
uuid: 'action-1',
|
|
441
|
+
text: 'Message 1',
|
|
442
|
+
quick_replies: ['Yes', 'No']
|
|
443
|
+
},
|
|
444
|
+
{
|
|
445
|
+
type: 'send_msg',
|
|
446
|
+
uuid: 'action-2',
|
|
447
|
+
text: 'Message 2',
|
|
448
|
+
quick_replies: []
|
|
449
|
+
}
|
|
450
|
+
],
|
|
451
|
+
exits: []
|
|
452
|
+
};
|
|
453
|
+
editorNode.node = mockNode;
|
|
454
|
+
// Swap the two actions
|
|
455
|
+
const orderChangedEvent = new CustomEvent(CustomEventType.OrderChanged, {
|
|
456
|
+
detail: { swap: [0, 1] }
|
|
457
|
+
});
|
|
458
|
+
editorNode.handleActionOrderChanged(orderChangedEvent);
|
|
459
|
+
// Verify all action data is preserved
|
|
460
|
+
expect(editorNode.node.actions).to.have.length(2);
|
|
461
|
+
expect(editorNode.node.actions[0].text).to.equal('Message 2');
|
|
462
|
+
expect(editorNode.node.actions[0].quick_replies).to.deep.equal([]);
|
|
463
|
+
expect(editorNode.node.actions[1].text).to.equal('Message 1');
|
|
464
|
+
expect(editorNode.node.actions[1].quick_replies).to.deep.equal(['Yes', 'No']);
|
|
465
|
+
});
|
|
466
|
+
it('integrates with SortableList for full drag functionality', async () => {
|
|
467
|
+
const mockNode = {
|
|
468
|
+
uuid: 'integration-drag-test',
|
|
469
|
+
actions: [
|
|
470
|
+
{
|
|
471
|
+
type: 'send_msg',
|
|
472
|
+
uuid: 'action-1',
|
|
473
|
+
text: 'First Action',
|
|
474
|
+
quick_replies: []
|
|
475
|
+
},
|
|
476
|
+
{
|
|
477
|
+
type: 'send_msg',
|
|
478
|
+
uuid: 'action-2',
|
|
479
|
+
text: 'Second Action',
|
|
480
|
+
quick_replies: []
|
|
481
|
+
},
|
|
482
|
+
{
|
|
483
|
+
type: 'send_msg',
|
|
484
|
+
uuid: 'action-3',
|
|
485
|
+
text: 'Third Action',
|
|
486
|
+
quick_replies: []
|
|
487
|
+
}
|
|
488
|
+
],
|
|
489
|
+
exits: []
|
|
490
|
+
};
|
|
491
|
+
const mockUI = {
|
|
492
|
+
position: { left: 100, top: 200 },
|
|
493
|
+
type: 'execute_actions'
|
|
494
|
+
};
|
|
495
|
+
// Set properties on the component
|
|
496
|
+
editorNode.node = mockNode;
|
|
497
|
+
editorNode.ui = mockUI;
|
|
498
|
+
// Render the full component
|
|
499
|
+
const renderResult = editorNode.render();
|
|
500
|
+
const container = await fixture(html `<div>${renderResult}</div>`);
|
|
501
|
+
// Find the sortable list
|
|
502
|
+
const sortableList = container.querySelector('temba-sortable-list');
|
|
503
|
+
expect(sortableList).to.exist;
|
|
504
|
+
// Verify all actions are rendered as sortable items
|
|
505
|
+
const sortableItems = container.querySelectorAll('.sortable');
|
|
506
|
+
expect(sortableItems).to.have.length(3);
|
|
507
|
+
// Verify each action has correct ID and structure
|
|
508
|
+
expect(sortableItems[0].id).to.equal('action-0');
|
|
509
|
+
expect(sortableItems[1].id).to.equal('action-1');
|
|
510
|
+
expect(sortableItems[2].id).to.equal('action-2');
|
|
511
|
+
// Verify drag handles are present
|
|
512
|
+
const dragHandles = container.querySelectorAll('.drag-handle');
|
|
513
|
+
expect(dragHandles).to.have.length(3);
|
|
514
|
+
});
|
|
515
|
+
});
|
|
272
516
|
});
|
|
273
517
|
//# sourceMappingURL=temba-flow-editor-node.test.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"temba-flow-editor-node.test.js","sourceRoot":"","sources":["../../test/temba-flow-editor-node.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { html, fixture, expect } from '@open-wc/testing';\nimport { EditorNode } from '../src/flow/EditorNode';\nimport {\n Node,\n NodeUI,\n Action,\n Exit,\n Router\n} from '../src/store/flow-definition.d';\nimport { stub, restore } from 'sinon';\n\n// Register the component\ncustomElements.define('temba-editor-node', EditorNode);\n\ndescribe('EditorNode', () => {\n let editorNode: EditorNode;\n let mockPlumber: any;\n\n beforeEach(async () => {\n // Mock plumber\n mockPlumber = {\n makeTarget: stub(),\n makeSource: stub(),\n connectIds: stub()\n };\n });\n\n afterEach(() => {\n restore();\n });\n\n describe('basic functionality', () => {\n it('creates render root as element itself', () => {\n const editorNode = new EditorNode();\n expect(editorNode.createRenderRoot()).to.equal(editorNode);\n });\n });\n\n describe('renderAction', () => {\n beforeEach(() => {\n editorNode = new EditorNode();\n });\n\n it('renders action with known config', () => {\n const mockNode: Node = {\n uuid: 'test-node-3',\n actions: [],\n exits: []\n };\n\n const action: any = {\n type: 'send_msg',\n uuid: 'action-1',\n text: 'Test message',\n quick_replies: []\n };\n\n const result = (editorNode as any).renderAction(mockNode, action);\n expect(result).to.exist;\n });\n\n it('renders action with unknown config', () => {\n const mockNode: Node = {\n uuid: 'test-node-4',\n actions: [],\n exits: []\n };\n\n const action: Action = {\n type: 'unknown_action' as any,\n uuid: 'action-1'\n };\n\n const result = (editorNode as any).renderAction(mockNode, action);\n expect(result).to.exist;\n });\n });\n\n describe('renderRouter', () => {\n beforeEach(() => {\n editorNode = new EditorNode();\n });\n\n it('renders router with result name', () => {\n const mockRouter: Router = {\n type: 'switch',\n result_name: 'test_result',\n categories: []\n };\n\n const mockUI: NodeUI = {\n position: { left: 50, top: 100 },\n type: 'wait_for_response'\n };\n\n const result = (editorNode as any).renderRouter(mockRouter, mockUI);\n expect(result).to.exist;\n });\n\n it('renders router without result name', () => {\n const mockRouter: Router = {\n type: 'switch',\n categories: []\n };\n\n const mockUI: NodeUI = {\n position: { left: 50, top: 100 },\n type: 'wait_for_response'\n };\n\n const result = (editorNode as any).renderRouter(mockRouter, mockUI);\n expect(result).to.exist;\n });\n\n it('returns undefined for router with unknown UI type', () => {\n const mockRouter: Router = {\n type: 'switch',\n categories: []\n };\n\n const mockUI: NodeUI = {\n position: { left: 50, top: 100 },\n type: 'unknown_type' as any\n };\n\n const result = (editorNode as any).renderRouter(mockRouter, mockUI);\n expect(result).to.be.undefined;\n });\n });\n\n describe('renderCategories', () => {\n beforeEach(() => {\n editorNode = new EditorNode();\n });\n\n it('returns null when no router', () => {\n const mockNode: Node = {\n uuid: 'test-node-7',\n actions: [],\n exits: []\n };\n\n const result = (editorNode as any).renderCategories(mockNode);\n expect(result).to.be.null;\n });\n\n it('returns null when no categories', () => {\n const mockNode: Node = {\n uuid: 'test-node-8',\n actions: [],\n exits: [],\n router: {\n type: 'switch',\n categories: undefined as any\n }\n };\n\n const result = (editorNode as any).renderCategories(mockNode);\n expect(result).to.be.null;\n });\n\n it('renders categories with exits', () => {\n const mockNode: Node = {\n uuid: 'test-node-9',\n actions: [],\n exits: [{ uuid: 'exit-1' }, { uuid: 'exit-2' }],\n router: {\n type: 'switch',\n categories: [\n { uuid: 'cat-1', name: 'Category 1', exit_uuid: 'exit-1' },\n { uuid: 'cat-2', name: 'Category 2', exit_uuid: 'exit-2' }\n ]\n }\n };\n\n const result = (editorNode as any).renderCategories(mockNode);\n expect(result).to.exist;\n });\n });\n\n describe('renderExit', () => {\n beforeEach(() => {\n editorNode = new EditorNode();\n });\n\n it('renders exit with connected class when destination exists', async () => {\n const exit: Exit = {\n uuid: 'exit-connected',\n destination_uuid: 'destination-node'\n };\n\n const result = (editorNode as any).renderExit(exit);\n const container = await fixture(html`<div>${result}</div>`);\n\n const exitElement = container.querySelector('.exit');\n expect(exitElement).to.exist;\n expect(exitElement?.classList.contains('connected')).to.be.true;\n expect(exitElement?.getAttribute('id')).to.equal('exit-connected');\n });\n\n it('renders exit without connected class when no destination', async () => {\n const exit: Exit = {\n uuid: 'exit-unconnected'\n };\n\n const result = (editorNode as any).renderExit(exit);\n const container = await fixture(html`<div>${result}</div>`);\n\n const exitElement = container.querySelector('.exit');\n expect(exitElement).to.exist;\n expect(exitElement?.classList.contains('connected')).to.be.false;\n expect(exitElement?.getAttribute('id')).to.equal('exit-unconnected');\n });\n });\n\n describe('renderTitle', () => {\n beforeEach(() => {\n editorNode = new EditorNode();\n });\n\n it('renders title with config color and name', async () => {\n const config = {\n name: 'Test Action',\n color: '#ff0000'\n };\n\n const result = (editorNode as any).renderTitle(config);\n const container = await fixture(html`<div>${result}</div>`);\n\n const title = container.querySelector('.title');\n expect(title).to.exist;\n expect(title?.textContent?.trim()).to.equal('Test Action');\n expect(title?.getAttribute('style')).to.contain('background:#ff0000');\n });\n });\n\n describe('updated lifecycle', () => {\n it('handles updated without node changes', () => {\n editorNode = new EditorNode();\n (editorNode as any).plumber = mockPlumber;\n\n const changes = new Map();\n changes.set('other', true);\n\n // Should not throw and not call plumber methods\n expect(() => {\n (editorNode as any).updated(changes);\n }).to.not.throw();\n\n expect(mockPlumber.makeTarget).to.not.have.been.called;\n });\n\n it('verifies updated method exists', () => {\n editorNode = new EditorNode();\n expect(typeof (editorNode as any).updated).to.equal('function');\n });\n\n it('processes node changes and calls plumber methods', () => {\n editorNode = new EditorNode();\n (editorNode as any).plumber = mockPlumber;\n\n const mockNode: Node = {\n uuid: 'test-node-10',\n actions: [],\n exits: [\n { uuid: 'exit-1', destination_uuid: 'node-2' },\n { uuid: 'exit-2' } // This should call makeSource\n ]\n };\n\n // Mock querySelector to return a mock element with getBoundingClientRect\n const mockElement = {\n getBoundingClientRect: stub().returns({ width: 200, height: 100 })\n };\n stub(editorNode, 'querySelector').returns(mockElement as any);\n\n // Simulate the updated lifecycle\n (editorNode as any).node = mockNode;\n\n const changes = new Map();\n changes.set('node', true);\n\n // Test just the plumber method calls without store dependency\n // by directly calling the logic that would be in updated\n if ((editorNode as any).plumber && mockNode) {\n (editorNode as any).plumber.makeTarget(mockNode.uuid);\n\n for (const exit of mockNode.exits) {\n if (!exit.destination_uuid) {\n (editorNode as any).plumber.makeSource(exit.uuid);\n } else {\n (editorNode as any).plumber.connectIds(\n exit.uuid,\n exit.destination_uuid\n );\n }\n }\n }\n\n expect(mockPlumber.makeTarget).to.have.been.calledWith('test-node-10');\n expect(mockPlumber.makeSource).to.have.been.calledWith('exit-2');\n expect(mockPlumber.connectIds).to.have.been.calledWith(\n 'exit-1',\n 'node-2'\n );\n });\n });\n\n describe('basic integration', () => {\n it('can create and verify structure without full rendering', () => {\n const mockNode: Node = {\n uuid: 'integration-test-node',\n actions: [\n {\n type: 'send_msg',\n uuid: 'action-1',\n text: 'Hello',\n quick_replies: []\n } as any\n ],\n exits: [{ uuid: 'exit-1', destination_uuid: 'next-node' }]\n };\n\n // Test individual render methods work\n editorNode = new EditorNode();\n\n // Test renderAction\n const actionResult = (editorNode as any).renderAction(\n mockNode,\n mockNode.actions[0]\n );\n expect(actionResult).to.exist;\n\n // Test renderExit\n const exitResult = (editorNode as any).renderExit(mockNode.exits[0]);\n expect(exitResult).to.exist;\n\n // Verify the node structure is as expected\n expect(mockNode.uuid).to.equal('integration-test-node');\n expect(mockNode.actions).to.have.length(1);\n expect(mockNode.exits).to.have.length(1);\n });\n });\n});\n"]}
|
|
1
|
+
{"version":3,"file":"temba-flow-editor-node.test.js","sourceRoot":"","sources":["../../test/temba-flow-editor-node.test.ts"],"names":[],"mappings":"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","sourcesContent":["import '../temba-modules';\nimport { html, fixture, expect } from '@open-wc/testing';\nimport { EditorNode } from '../src/flow/EditorNode';\nimport {\n Node,\n NodeUI,\n Action,\n Exit,\n Router\n} from '../src/store/flow-definition.d';\nimport { stub, restore } from 'sinon';\nimport { CustomEventType } from '../src/interfaces';\n\ndescribe('EditorNode', () => {\n let editorNode: EditorNode;\n let mockPlumber: any;\n\n beforeEach(async () => {\n // Mock plumber\n mockPlumber = {\n makeTarget: stub(),\n makeSource: stub(),\n connectIds: stub()\n };\n });\n\n afterEach(() => {\n restore();\n });\n\n describe('basic functionality', () => {\n it('creates render root as element itself', () => {\n const editorNode = new EditorNode();\n expect(editorNode.createRenderRoot()).to.equal(editorNode);\n });\n });\n\n describe('renderAction', () => {\n beforeEach(() => {\n editorNode = new EditorNode();\n });\n\n it('renders action with known config', () => {\n const mockNode: Node = {\n uuid: 'test-node-3',\n actions: [],\n exits: []\n };\n\n const action: any = {\n type: 'send_msg',\n uuid: 'action-1',\n text: 'Test message',\n quick_replies: []\n };\n\n const result = (editorNode as any).renderAction(mockNode, action, 0);\n expect(result).to.exist;\n });\n\n it('renders action with unknown config', () => {\n const mockNode: Node = {\n uuid: 'test-node-4',\n actions: [],\n exits: []\n };\n\n const action: Action = {\n type: 'unknown_action' as any,\n uuid: 'action-1'\n };\n\n const result = (editorNode as any).renderAction(mockNode, action, 1);\n expect(result).to.exist;\n });\n });\n\n describe('renderRouter', () => {\n beforeEach(() => {\n editorNode = new EditorNode();\n });\n\n it('renders router with result name', () => {\n const mockRouter: Router = {\n type: 'switch',\n result_name: 'test_result',\n categories: []\n };\n\n const mockUI: NodeUI = {\n position: { left: 50, top: 100 },\n type: 'wait_for_response'\n };\n\n const result = (editorNode as any).renderRouter(mockRouter, mockUI);\n expect(result).to.exist;\n });\n\n it('renders router without result name', () => {\n const mockRouter: Router = {\n type: 'switch',\n categories: []\n };\n\n const mockUI: NodeUI = {\n position: { left: 50, top: 100 },\n type: 'wait_for_response'\n };\n\n const result = (editorNode as any).renderRouter(mockRouter, mockUI);\n expect(result).to.exist;\n });\n\n it('returns undefined for router with unknown UI type', () => {\n const mockRouter: Router = {\n type: 'switch',\n categories: []\n };\n\n const mockUI: NodeUI = {\n position: { left: 50, top: 100 },\n type: 'unknown_type' as any\n };\n\n const result = (editorNode as any).renderRouter(mockRouter, mockUI);\n expect(result).to.be.undefined;\n });\n });\n\n describe('renderCategories', () => {\n beforeEach(() => {\n editorNode = new EditorNode();\n });\n\n it('returns null when no router', () => {\n const mockNode: Node = {\n uuid: 'test-node-7',\n actions: [],\n exits: []\n };\n\n const result = (editorNode as any).renderCategories(mockNode);\n expect(result).to.be.null;\n });\n\n it('returns null when no categories', () => {\n const mockNode: Node = {\n uuid: 'test-node-8',\n actions: [],\n exits: [],\n router: {\n type: 'switch',\n categories: undefined as any\n }\n };\n\n const result = (editorNode as any).renderCategories(mockNode);\n expect(result).to.be.null;\n });\n\n it('renders categories with exits', () => {\n const mockNode: Node = {\n uuid: 'test-node-9',\n actions: [],\n exits: [{ uuid: 'exit-1' }, { uuid: 'exit-2' }],\n router: {\n type: 'switch',\n categories: [\n { uuid: 'cat-1', name: 'Category 1', exit_uuid: 'exit-1' },\n { uuid: 'cat-2', name: 'Category 2', exit_uuid: 'exit-2' }\n ]\n }\n };\n\n const result = (editorNode as any).renderCategories(mockNode);\n expect(result).to.exist;\n });\n });\n\n describe('renderExit', () => {\n beforeEach(() => {\n editorNode = new EditorNode();\n });\n\n it('renders exit with connected class when destination exists', async () => {\n const exit: Exit = {\n uuid: 'exit-connected',\n destination_uuid: 'destination-node'\n };\n\n const result = (editorNode as any).renderExit(exit);\n const container = await fixture(html`<div>${result}</div>`);\n\n const exitElement = container.querySelector('.exit');\n expect(exitElement).to.exist;\n expect(exitElement?.classList.contains('connected')).to.be.true;\n expect(exitElement?.getAttribute('id')).to.equal('exit-connected');\n });\n\n it('renders exit without connected class when no destination', async () => {\n const exit: Exit = {\n uuid: 'exit-unconnected'\n };\n\n const result = (editorNode as any).renderExit(exit);\n const container = await fixture(html`<div>${result}</div>`);\n\n const exitElement = container.querySelector('.exit');\n expect(exitElement).to.exist;\n expect(exitElement?.classList.contains('connected')).to.be.false;\n expect(exitElement?.getAttribute('id')).to.equal('exit-unconnected');\n });\n });\n\n describe('renderTitle', () => {\n beforeEach(() => {\n editorNode = new EditorNode();\n });\n\n it('renders title with config color and name', async () => {\n const config = {\n name: 'Test Action',\n color: '#ff0000'\n };\n\n const result = (editorNode as any).renderTitle(config);\n const container = await fixture(html`<div>${result}</div>`);\n\n const title = container.querySelector('.title');\n expect(title).to.exist;\n expect(title?.textContent?.trim()).to.equal('Test Action');\n expect(title?.getAttribute('style')).to.contain('background:#ff0000');\n });\n });\n\n describe('updated lifecycle', () => {\n it('handles updated without node changes', () => {\n editorNode = new EditorNode();\n (editorNode as any).plumber = mockPlumber;\n\n const changes = new Map();\n changes.set('other', true);\n\n // Should not throw and not call plumber methods\n expect(() => {\n (editorNode as any).updated(changes);\n }).to.not.throw();\n\n expect(mockPlumber.makeTarget).to.not.have.been.called;\n });\n\n it('verifies updated method exists', () => {\n editorNode = new EditorNode();\n expect(typeof (editorNode as any).updated).to.equal('function');\n });\n\n it('processes node changes and calls plumber methods', () => {\n editorNode = new EditorNode();\n (editorNode as any).plumber = mockPlumber;\n\n const mockNode: Node = {\n uuid: 'test-node-10',\n actions: [],\n exits: [\n { uuid: 'exit-1', destination_uuid: 'node-2' },\n { uuid: 'exit-2' } // This should call makeSource\n ]\n };\n\n // Mock querySelector to return a mock element with getBoundingClientRect\n const mockElement = {\n getBoundingClientRect: stub().returns({ width: 200, height: 100 })\n };\n stub(editorNode, 'querySelector').returns(mockElement as any);\n\n // Simulate the updated lifecycle\n (editorNode as any).node = mockNode;\n\n const changes = new Map();\n changes.set('node', true);\n\n // Test just the plumber method calls without store dependency\n // by directly calling the logic that would be in updated\n if ((editorNode as any).plumber && mockNode) {\n (editorNode as any).plumber.makeTarget(mockNode.uuid);\n\n for (const exit of mockNode.exits) {\n if (!exit.destination_uuid) {\n (editorNode as any).plumber.makeSource(exit.uuid);\n } else {\n (editorNode as any).plumber.connectIds(\n exit.uuid,\n exit.destination_uuid\n );\n }\n }\n }\n\n expect(mockPlumber.makeTarget).to.have.been.calledWith('test-node-10');\n expect(mockPlumber.makeSource).to.have.been.calledWith('exit-2');\n expect(mockPlumber.connectIds).to.have.been.calledWith(\n 'exit-1',\n 'node-2'\n );\n });\n });\n\n describe('basic integration', () => {\n it('can create and verify structure without full rendering', () => {\n const mockNode: Node = {\n uuid: 'integration-test-node',\n actions: [\n {\n type: 'send_msg',\n uuid: 'action-1',\n text: 'Hello',\n quick_replies: []\n } as any\n ],\n exits: [{ uuid: 'exit-1', destination_uuid: 'next-node' }]\n };\n\n // Test individual render methods work\n editorNode = new EditorNode();\n\n // Test renderAction\n const actionResult = (editorNode as any).renderAction(\n mockNode,\n mockNode.actions[0],\n 0\n );\n expect(actionResult).to.exist;\n\n // Test renderExit\n const exitResult = (editorNode as any).renderExit(mockNode.exits[0]);\n expect(exitResult).to.exist;\n\n // Verify the node structure is as expected\n expect(mockNode.uuid).to.equal('integration-test-node');\n expect(mockNode.actions).to.have.length(1);\n expect(mockNode.exits).to.have.length(1);\n });\n });\n\n describe('drag and drop functionality', () => {\n let editorNode: EditorNode;\n\n beforeEach(() => {\n editorNode = new EditorNode();\n });\n\n it('renders actions with sortable class and proper IDs', async () => {\n const mockNode: Node = {\n uuid: 'sortable-test-node',\n actions: [\n {\n type: 'send_msg',\n uuid: 'action-1',\n text: 'Hello',\n quick_replies: []\n } as any,\n {\n type: 'send_msg',\n uuid: 'action-2',\n text: 'World',\n quick_replies: []\n } as any\n ],\n exits: []\n };\n\n // Test that renderAction includes sortable class and proper ID\n const result1 = (editorNode as any).renderAction(\n mockNode,\n mockNode.actions[0],\n 0\n );\n const result2 = (editorNode as any).renderAction(\n mockNode,\n mockNode.actions[1],\n 1\n );\n\n expect(result1).to.exist;\n expect(result2).to.exist;\n\n // Render the template to check the actual DOM\n const container1 = await fixture(html`<div>${result1}</div>`);\n const container2 = await fixture(html`<div>${result2}</div>`);\n\n const actionElement1 = container1.querySelector('.action');\n const actionElement2 = container2.querySelector('.action');\n\n expect(actionElement1).to.exist;\n expect(actionElement1?.classList.contains('sortable')).to.be.true;\n expect(actionElement1?.id).to.equal('action-0');\n\n expect(actionElement2).to.exist;\n expect(actionElement2?.classList.contains('sortable')).to.be.true;\n expect(actionElement2?.id).to.equal('action-1');\n });\n\n it('includes drag handle in rendered actions', async () => {\n const mockNode: Node = {\n uuid: 'drag-handle-test',\n actions: [\n {\n type: 'send_msg',\n uuid: 'action-1',\n text: 'Hello',\n quick_replies: []\n } as any\n ],\n exits: []\n };\n\n let editorNode: EditorNode = await fixture(\n html`<temba-flow-node\n .node=${mockNode}\n .ui=${{ position: { left: 0, top: 0 } }}\n ></temba-flow-node>`\n );\n\n // No drag handle should be present if only one action\n let dragHandle = editorNode.querySelector('.drag-handle');\n expect(dragHandle).to.not.exist;\n\n // Now add a second action to verify drag handle appears\n mockNode.actions.push({\n type: 'send_msg',\n uuid: 'action-2',\n text: 'World',\n quick_replies: []\n } as any);\n\n editorNode = await fixture(\n html`<temba-flow-node\n .node=${mockNode}\n .ui=${{ position: { left: 0, top: 0 } }}\n ></temba-flow-node>`\n );\n\n dragHandle = editorNode.querySelector('.drag-handle');\n expect(dragHandle).to.exist;\n });\n\n it('renders SortableList when actions are present', async () => {\n const mockNode: Node = {\n uuid: 'sortable-list-test',\n actions: [\n {\n type: 'send_msg',\n uuid: 'action-1',\n text: 'Hello',\n quick_replies: []\n } as any,\n {\n type: 'send_msg',\n uuid: 'action-2',\n text: 'World',\n quick_replies: []\n } as any\n ],\n exits: []\n };\n\n const mockUI: NodeUI = {\n position: { left: 100, top: 200 },\n type: 'execute_actions'\n };\n\n // Set properties on the component\n (editorNode as any).node = mockNode;\n (editorNode as any).ui = mockUI;\n\n const renderResult = editorNode.render();\n\n // Render the template to check the actual DOM\n const container = await fixture(html`<div>${renderResult}</div>`);\n\n const sortableList = container.querySelector('temba-sortable-list');\n expect(sortableList).to.exist;\n });\n\n it('does not render SortableList when no actions', async () => {\n const mockNode: Node = {\n uuid: 'no-actions-test',\n actions: [],\n exits: [{ uuid: 'exit-1' }]\n };\n\n const mockUI: NodeUI = {\n position: { left: 100, top: 200 },\n type: 'execute_actions'\n };\n\n // Set properties on the component\n (editorNode as any).node = mockNode;\n (editorNode as any).ui = mockUI;\n\n const renderResult = editorNode.render();\n\n // Check that template does not include temba-sortable-list\n expect(renderResult.strings.join('')).to.not.contain(\n 'temba-sortable-list'\n );\n });\n\n it('handles order changed events correctly', async () => {\n const mockNode: Node = {\n uuid: 'order-test',\n actions: [\n {\n type: 'send_msg',\n uuid: 'action-1',\n text: 'First',\n quick_replies: []\n } as any,\n {\n type: 'send_msg',\n uuid: 'action-2',\n text: 'Second',\n quick_replies: []\n } as any,\n {\n type: 'send_msg',\n uuid: 'action-3',\n text: 'Third',\n quick_replies: []\n } as any\n ],\n exits: []\n };\n\n (editorNode as any).node = mockNode;\n\n // Create a mock order changed event (swap first and last actions)\n const orderChangedEvent = new CustomEvent(CustomEventType.OrderChanged, {\n detail: { swap: [0, 2] }\n });\n\n // Call the handler directly\n (editorNode as any).handleActionOrderChanged(orderChangedEvent);\n\n // Verify the actions were reordered correctly\n expect((editorNode as any).node.actions).to.have.length(3);\n expect(((editorNode as any).node.actions[0] as any).text).to.equal(\n 'Second'\n );\n expect(((editorNode as any).node.actions[1] as any).text).to.equal(\n 'Third'\n );\n expect(((editorNode as any).node.actions[2] as any).text).to.equal(\n 'First'\n );\n });\n\n it('preserves action data during reordering', () => {\n const mockNode: Node = {\n uuid: 'preserve-test',\n actions: [\n {\n type: 'send_msg',\n uuid: 'action-1',\n text: 'Message 1',\n quick_replies: ['Yes', 'No']\n } as any,\n {\n type: 'send_msg',\n uuid: 'action-2',\n text: 'Message 2',\n quick_replies: []\n } as any\n ],\n exits: []\n };\n\n (editorNode as any).node = mockNode;\n\n // Swap the two actions\n const orderChangedEvent = new CustomEvent(CustomEventType.OrderChanged, {\n detail: { swap: [0, 1] }\n });\n\n (editorNode as any).handleActionOrderChanged(orderChangedEvent);\n\n // Verify all action data is preserved\n expect((editorNode as any).node.actions).to.have.length(2);\n expect(((editorNode as any).node.actions[0] as any).text).to.equal(\n 'Message 2'\n );\n expect(\n ((editorNode as any).node.actions[0] as any).quick_replies\n ).to.deep.equal([]);\n expect(((editorNode as any).node.actions[1] as any).text).to.equal(\n 'Message 1'\n );\n expect(\n ((editorNode as any).node.actions[1] as any).quick_replies\n ).to.deep.equal(['Yes', 'No']);\n });\n\n it('integrates with SortableList for full drag functionality', async () => {\n const mockNode: Node = {\n uuid: 'integration-drag-test',\n actions: [\n {\n type: 'send_msg',\n uuid: 'action-1',\n text: 'First Action',\n quick_replies: []\n } as any,\n {\n type: 'send_msg',\n uuid: 'action-2',\n text: 'Second Action',\n quick_replies: []\n } as any,\n {\n type: 'send_msg',\n uuid: 'action-3',\n text: 'Third Action',\n quick_replies: []\n } as any\n ],\n exits: []\n };\n\n const mockUI: NodeUI = {\n position: { left: 100, top: 200 },\n type: 'execute_actions'\n };\n\n // Set properties on the component\n (editorNode as any).node = mockNode;\n (editorNode as any).ui = mockUI;\n\n // Render the full component\n const renderResult = editorNode.render();\n const container = await fixture(html`<div>${renderResult}</div>`);\n\n // Find the sortable list\n const sortableList = container.querySelector('temba-sortable-list');\n expect(sortableList).to.exist;\n\n // Verify all actions are rendered as sortable items\n const sortableItems = container.querySelectorAll('.sortable');\n expect(sortableItems).to.have.length(3);\n\n // Verify each action has correct ID and structure\n expect(sortableItems[0].id).to.equal('action-0');\n expect(sortableItems[1].id).to.equal('action-1');\n expect(sortableItems[2].id).to.equal('action-2');\n\n // Verify drag handles are present\n const dragHandles = container.querySelectorAll('.drag-handle');\n expect(dragHandles).to.have.length(3);\n });\n });\n});\n"]}
|
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
import { fixture, expect, assert } from '@open-wc/testing';
|
|
2
|
+
import { RangePicker } from '../src/datepicker/RangePicker';
|
|
3
|
+
import { assertScreenshot, getAttributes, getClip } from './utils.test';
|
|
4
|
+
import { DateTime } from 'luxon';
|
|
5
|
+
export const getRangePickerHTML = (attrs = {}) => {
|
|
6
|
+
return `<temba-range-picker ${getAttributes(attrs)}></temba-range-picker>`;
|
|
7
|
+
};
|
|
8
|
+
export const createRangePicker = async (def) => {
|
|
9
|
+
const parentNode = document.createElement('div');
|
|
10
|
+
parentNode.setAttribute('style', 'width: 600px;');
|
|
11
|
+
parentNode.id = 'parent';
|
|
12
|
+
const picker = await fixture(def, { parentNode });
|
|
13
|
+
return picker;
|
|
14
|
+
};
|
|
15
|
+
describe('temba-range-picker', () => {
|
|
16
|
+
it('can create a range picker', async () => {
|
|
17
|
+
const picker = await createRangePicker(getRangePickerHTML());
|
|
18
|
+
assert.instanceOf(picker, RangePicker);
|
|
19
|
+
// Should have default range (last month)
|
|
20
|
+
expect(picker.selectedRange).to.equal('M');
|
|
21
|
+
expect(picker.startDate).to.not.be.empty;
|
|
22
|
+
expect(picker.endDate).to.not.be.empty;
|
|
23
|
+
await assertScreenshot('datepicker/range-picker-default', getClip(picker));
|
|
24
|
+
});
|
|
25
|
+
it('can be initialized with start and end dates', async () => {
|
|
26
|
+
const picker = await createRangePicker(getRangePickerHTML({ start: '2024-01-01', end: '2024-01-31' }));
|
|
27
|
+
expect(picker.startDate).to.equal('2024-01-01');
|
|
28
|
+
expect(picker.endDate).to.equal('2024-01-31');
|
|
29
|
+
expect(picker.selectedRange).to.equal('');
|
|
30
|
+
await assertScreenshot('datepicker/range-picker-initial-values', getClip(picker));
|
|
31
|
+
});
|
|
32
|
+
it('can set min and max dates', async () => {
|
|
33
|
+
const picker = await createRangePicker(getRangePickerHTML({
|
|
34
|
+
start: '2024-06-01',
|
|
35
|
+
end: '2024-06-30',
|
|
36
|
+
min: '2024-01-01',
|
|
37
|
+
max: '2024-12-31'
|
|
38
|
+
}));
|
|
39
|
+
expect(picker.minDate).to.equal('2024-01-01');
|
|
40
|
+
expect(picker.maxDate).to.equal('2024-12-31');
|
|
41
|
+
await assertScreenshot('datepicker/range-picker-min-max', getClip(picker));
|
|
42
|
+
});
|
|
43
|
+
it('can set range using buttons', async () => {
|
|
44
|
+
var _a;
|
|
45
|
+
const picker = await createRangePicker(getRangePickerHTML());
|
|
46
|
+
// Click Week button
|
|
47
|
+
const weekBtn = (_a = picker.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('.range-btn');
|
|
48
|
+
weekBtn.click();
|
|
49
|
+
await picker.updateComplete;
|
|
50
|
+
expect(picker.selectedRange).to.equal('W');
|
|
51
|
+
expect(picker.startDate).to.equal(DateTime.now().minus({ days: 6 }).toISODate());
|
|
52
|
+
expect(picker.endDate).to.equal(DateTime.now().toISODate());
|
|
53
|
+
await assertScreenshot('datepicker/range-picker-week', getClip(picker));
|
|
54
|
+
});
|
|
55
|
+
it('can set year range using button', async () => {
|
|
56
|
+
var _a;
|
|
57
|
+
const picker = await createRangePicker(getRangePickerHTML());
|
|
58
|
+
// Click Year button (3rd button)
|
|
59
|
+
const yearBtn = (_a = picker.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelectorAll('.range-btn')[2];
|
|
60
|
+
yearBtn.click();
|
|
61
|
+
await picker.updateComplete;
|
|
62
|
+
expect(picker.selectedRange).to.equal('Y');
|
|
63
|
+
expect(picker.startDate).to.equal(DateTime.now().minus({ years: 1 }).plus({ days: 1 }).toISODate());
|
|
64
|
+
expect(picker.endDate).to.equal(DateTime.now().toISODate());
|
|
65
|
+
await assertScreenshot('datepicker/range-picker-year', getClip(picker));
|
|
66
|
+
});
|
|
67
|
+
it('can set all range using button', async () => {
|
|
68
|
+
var _a;
|
|
69
|
+
const picker = await createRangePicker(getRangePickerHTML());
|
|
70
|
+
// Click All button (4th button)
|
|
71
|
+
const allBtn = (_a = picker.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelectorAll('.range-btn')[3];
|
|
72
|
+
allBtn.click();
|
|
73
|
+
await picker.updateComplete;
|
|
74
|
+
expect(picker.selectedRange).to.equal('ALL');
|
|
75
|
+
expect(picker.startDate).to.equal('2012-01-01');
|
|
76
|
+
expect(picker.endDate).to.equal(DateTime.now().toISODate());
|
|
77
|
+
await assertScreenshot('datepicker/range-picker-all', getClip(picker));
|
|
78
|
+
});
|
|
79
|
+
it('enforces valid date ranges', async () => {
|
|
80
|
+
const picker = await createRangePicker(getRangePickerHTML({ start: '2024-06-01', end: '2024-06-30' }));
|
|
81
|
+
// Verify initial state is valid
|
|
82
|
+
expect(DateTime.fromISO(picker.endDate) >= DateTime.fromISO(picker.startDate)).to.be.true;
|
|
83
|
+
// The validation logic is internal and triggered through user interaction
|
|
84
|
+
// We can verify the component has the correct min/max constraints
|
|
85
|
+
expect(picker.startDate).to.equal('2024-06-01');
|
|
86
|
+
expect(picker.endDate).to.equal('2024-06-30');
|
|
87
|
+
});
|
|
88
|
+
it('enforces min/max date constraints', async () => {
|
|
89
|
+
const picker = await createRangePicker(getRangePickerHTML({ min: '2024-01-01', max: '2024-12-31' }));
|
|
90
|
+
expect(picker.minDate).to.equal('2024-01-01');
|
|
91
|
+
expect(picker.maxDate).to.equal('2024-12-31');
|
|
92
|
+
// Min/max are enforced through the temba-datepicker components
|
|
93
|
+
// when user interacts with the date inputs
|
|
94
|
+
});
|
|
95
|
+
it('shows correct button selection states', async () => {
|
|
96
|
+
var _a, _b;
|
|
97
|
+
const picker = await createRangePicker(getRangePickerHTML());
|
|
98
|
+
// Initially should have M selected
|
|
99
|
+
const monthBtn = (_a = picker.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelectorAll('.range-btn')[1];
|
|
100
|
+
expect(monthBtn.classList.contains('selected')).to.be.true;
|
|
101
|
+
// Click week button
|
|
102
|
+
const weekBtn = (_b = picker.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('.range-btn');
|
|
103
|
+
weekBtn.click();
|
|
104
|
+
await picker.updateComplete;
|
|
105
|
+
expect(weekBtn.classList.contains('selected')).to.be.true;
|
|
106
|
+
expect(monthBtn.classList.contains('selected')).to.be.false;
|
|
107
|
+
await assertScreenshot('datepicker/range-picker-button-states', getClip(picker));
|
|
108
|
+
});
|
|
109
|
+
it('can click to edit dates', async () => {
|
|
110
|
+
var _a, _b;
|
|
111
|
+
const picker = await createRangePicker(getRangePickerHTML());
|
|
112
|
+
// Click on start date display
|
|
113
|
+
const startDisplay = (_a = picker.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('.date-display');
|
|
114
|
+
startDisplay.click();
|
|
115
|
+
await picker.updateComplete;
|
|
116
|
+
expect(picker.editingStart).to.be.true;
|
|
117
|
+
// Should show temba-datepicker for start
|
|
118
|
+
const startPicker = (_b = picker.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('temba-datepicker.start-picker');
|
|
119
|
+
expect(startPicker).to.not.be.null;
|
|
120
|
+
await assertScreenshot('datepicker/range-picker-editing-start', getClip(picker));
|
|
121
|
+
});
|
|
122
|
+
});
|
|
123
|
+
//# sourceMappingURL=temba-range-picker.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"temba-range-picker.test.js","sourceRoot":"","sources":["../../test/temba-range-picker.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { fixture, expect, assert } from '@open-wc/testing';\nimport { RangePicker } from '../src/datepicker/RangePicker';\nimport { assertScreenshot, getAttributes, getClip } from './utils.test';\nimport { DateTime } from 'luxon';\n\nexport const getRangePickerHTML = (attrs: any = {}) => {\n return `<temba-range-picker ${getAttributes(attrs)}></temba-range-picker>`;\n};\n\nexport const createRangePicker = async (def: string) => {\n const parentNode = document.createElement('div');\n parentNode.setAttribute('style', 'width: 600px;');\n parentNode.id = 'parent';\n const picker: RangePicker = await fixture(def, { parentNode });\n return picker;\n};\n\ndescribe('temba-range-picker', () => {\n it('can create a range picker', async () => {\n const picker: RangePicker = await createRangePicker(getRangePickerHTML());\n assert.instanceOf(picker, RangePicker);\n\n // Should have default range (last month)\n expect(picker.selectedRange).to.equal('M');\n expect(picker.startDate).to.not.be.empty;\n expect(picker.endDate).to.not.be.empty;\n\n await assertScreenshot('datepicker/range-picker-default', getClip(picker));\n });\n\n it('can be initialized with start and end dates', async () => {\n const picker: RangePicker = await createRangePicker(\n getRangePickerHTML({ start: '2024-01-01', end: '2024-01-31' })\n );\n\n expect(picker.startDate).to.equal('2024-01-01');\n expect(picker.endDate).to.equal('2024-01-31');\n expect(picker.selectedRange).to.equal('');\n\n await assertScreenshot(\n 'datepicker/range-picker-initial-values',\n getClip(picker)\n );\n });\n\n it('can set min and max dates', async () => {\n const picker: RangePicker = await createRangePicker(\n getRangePickerHTML({\n start: '2024-06-01',\n end: '2024-06-30',\n min: '2024-01-01',\n max: '2024-12-31'\n })\n );\n\n expect(picker.minDate).to.equal('2024-01-01');\n expect(picker.maxDate).to.equal('2024-12-31');\n\n await assertScreenshot('datepicker/range-picker-min-max', getClip(picker));\n });\n\n it('can set range using buttons', async () => {\n const picker: RangePicker = await createRangePicker(getRangePickerHTML());\n\n // Click Week button\n const weekBtn = picker.shadowRoot?.querySelector(\n '.range-btn'\n ) as HTMLButtonElement;\n weekBtn.click();\n await picker.updateComplete;\n\n expect(picker.selectedRange).to.equal('W');\n expect(picker.startDate).to.equal(\n DateTime.now().minus({ days: 6 }).toISODate()\n );\n expect(picker.endDate).to.equal(DateTime.now().toISODate());\n\n await assertScreenshot('datepicker/range-picker-week', getClip(picker));\n });\n\n it('can set year range using button', async () => {\n const picker: RangePicker = await createRangePicker(getRangePickerHTML());\n\n // Click Year button (3rd button)\n const yearBtn = picker.shadowRoot?.querySelectorAll(\n '.range-btn'\n )[2] as HTMLButtonElement;\n yearBtn.click();\n await picker.updateComplete;\n\n expect(picker.selectedRange).to.equal('Y');\n expect(picker.startDate).to.equal(\n DateTime.now().minus({ years: 1 }).plus({ days: 1 }).toISODate()\n );\n expect(picker.endDate).to.equal(DateTime.now().toISODate());\n\n await assertScreenshot('datepicker/range-picker-year', getClip(picker));\n });\n\n it('can set all range using button', async () => {\n const picker: RangePicker = await createRangePicker(getRangePickerHTML());\n\n // Click All button (4th button)\n const allBtn = picker.shadowRoot?.querySelectorAll(\n '.range-btn'\n )[3] as HTMLButtonElement;\n allBtn.click();\n await picker.updateComplete;\n\n expect(picker.selectedRange).to.equal('ALL');\n expect(picker.startDate).to.equal('2012-01-01');\n expect(picker.endDate).to.equal(DateTime.now().toISODate());\n\n await assertScreenshot('datepicker/range-picker-all', getClip(picker));\n });\n\n it('enforces valid date ranges', async () => {\n const picker: RangePicker = await createRangePicker(\n getRangePickerHTML({ start: '2024-06-01', end: '2024-06-30' })\n );\n\n // Verify initial state is valid\n expect(\n DateTime.fromISO(picker.endDate) >= DateTime.fromISO(picker.startDate)\n ).to.be.true;\n\n // The validation logic is internal and triggered through user interaction\n // We can verify the component has the correct min/max constraints\n expect(picker.startDate).to.equal('2024-06-01');\n expect(picker.endDate).to.equal('2024-06-30');\n });\n\n it('enforces min/max date constraints', async () => {\n const picker: RangePicker = await createRangePicker(\n getRangePickerHTML({ min: '2024-01-01', max: '2024-12-31' })\n );\n\n expect(picker.minDate).to.equal('2024-01-01');\n expect(picker.maxDate).to.equal('2024-12-31');\n\n // Min/max are enforced through the temba-datepicker components\n // when user interacts with the date inputs\n });\n\n it('shows correct button selection states', async () => {\n const picker: RangePicker = await createRangePicker(getRangePickerHTML());\n\n // Initially should have M selected\n const monthBtn = picker.shadowRoot?.querySelectorAll(\n '.range-btn'\n )[1] as HTMLButtonElement;\n expect(monthBtn.classList.contains('selected')).to.be.true;\n\n // Click week button\n const weekBtn = picker.shadowRoot?.querySelector(\n '.range-btn'\n ) as HTMLButtonElement;\n weekBtn.click();\n await picker.updateComplete;\n\n expect(weekBtn.classList.contains('selected')).to.be.true;\n expect(monthBtn.classList.contains('selected')).to.be.false;\n\n await assertScreenshot(\n 'datepicker/range-picker-button-states',\n getClip(picker)\n );\n });\n\n it('can click to edit dates', async () => {\n const picker: RangePicker = await createRangePicker(getRangePickerHTML());\n\n // Click on start date display\n const startDisplay = picker.shadowRoot?.querySelector(\n '.date-display'\n ) as HTMLElement;\n startDisplay.click();\n await picker.updateComplete;\n\n expect(picker.editingStart).to.be.true;\n\n // Should show temba-datepicker for start\n const startPicker = picker.shadowRoot?.querySelector(\n 'temba-datepicker.start-picker'\n );\n expect(startPicker).to.not.be.null;\n\n await assertScreenshot(\n 'datepicker/range-picker-editing-start',\n getClip(picker)\n );\n });\n});\n"]}
|
|
@@ -3,7 +3,7 @@ import { fixture, expect, assert } from '@open-wc/testing';
|
|
|
3
3
|
import { useFakeTimers } from 'sinon';
|
|
4
4
|
import { Options } from '../src/options/Options';
|
|
5
5
|
import { Select } from '../src/select/Select';
|
|
6
|
-
import { assertScreenshot,
|
|
6
|
+
import { assertScreenshot, getClip, getOptions, loadStore, openAndClick, openSelect, waitForSelectPagination } from './utils.test';
|
|
7
7
|
const colors = [
|
|
8
8
|
{ name: 'Red', value: '0' },
|
|
9
9
|
{ name: 'Green', value: '1' },
|
|
@@ -438,7 +438,6 @@ describe('temba-select', () => {
|
|
|
438
438
|
// await open(select);
|
|
439
439
|
// assert.equal(select.visibleOptions.length, 2);
|
|
440
440
|
});
|
|
441
|
-
// this test is flaky on CI, skip it for now
|
|
442
441
|
xit('pages through cursor results', async () => {
|
|
443
442
|
const select = await createSelect(clock, getSelectHTML([], {
|
|
444
443
|
placeholder: 'Select a group',
|
|
@@ -446,23 +445,13 @@ describe('temba-select', () => {
|
|
|
446
445
|
valueKey: 'uuid'
|
|
447
446
|
}));
|
|
448
447
|
await openSelect(clock, select);
|
|
449
|
-
// Wait for pagination to complete
|
|
450
|
-
//
|
|
451
|
-
|
|
452
|
-
const maxAttempts = 10;
|
|
453
|
-
while (select.fetching || select.visibleOptions.length < 15) {
|
|
454
|
-
if (attempts >= maxAttempts) {
|
|
455
|
-
throw new Error(`Pagination did not complete after ${maxAttempts} attempts. fetching: ${select.fetching}, visibleOptions: ${select.visibleOptions.length}`);
|
|
456
|
-
}
|
|
457
|
-
await select.updateComplete;
|
|
458
|
-
clock.runAll();
|
|
459
|
-
attempts++;
|
|
460
|
-
await delay(100);
|
|
461
|
-
}
|
|
448
|
+
// Wait for pagination to complete using our improved helper
|
|
449
|
+
// Use more attempts for this test since pagination can be slow in CI
|
|
450
|
+
await waitForSelectPagination(select, clock, 15, 50);
|
|
462
451
|
// should have all three pages visible right away
|
|
463
452
|
assert.equal(select.visibleOptions.length, 15);
|
|
464
453
|
});
|
|
465
|
-
|
|
454
|
+
it('shows cached results', async () => {
|
|
466
455
|
const select = await createSelect(clock, getSelectHTML([], {
|
|
467
456
|
placeholder: 'Select a group',
|
|
468
457
|
endpoint: '/test-assets/select/groups.json',
|
|
@@ -471,11 +460,16 @@ describe('temba-select', () => {
|
|
|
471
460
|
}));
|
|
472
461
|
// wait for updates from fetching three pages
|
|
473
462
|
await openSelect(clock, select);
|
|
463
|
+
await waitForSelectPagination(select, clock, 15, 50);
|
|
474
464
|
assert.equal(select.visibleOptions.length, 15);
|
|
475
465
|
// close and reopen
|
|
476
466
|
select.blur();
|
|
477
467
|
await clock.tick(250);
|
|
468
|
+
// Ensure the select is properly closed before reopening
|
|
469
|
+
await select.updateComplete;
|
|
478
470
|
await openSelect(clock, select);
|
|
471
|
+
// Cached results should be available immediately, but give some time for rendering
|
|
472
|
+
await waitForSelectPagination(select, clock, 15, 30);
|
|
479
473
|
assert.equal(select.visibleOptions.length, 15);
|
|
480
474
|
// close and reopen once more (previous bug failed on third opening)
|
|
481
475
|
// select.blur();
|