@capillarytech/creatives-library 8.0.345-alpha.13 → 8.0.345-alpha.15

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 (138) hide show
  1. package/constants/unified.js +29 -0
  2. package/package.json +1 -1
  3. package/services/api.js +0 -20
  4. package/services/tests/api.test.js +13 -59
  5. package/utils/commonUtils.js +19 -1
  6. package/utils/rcsPayloadUtils.js +92 -0
  7. package/utils/templateVarUtils.js +201 -0
  8. package/utils/tests/templateVarUtils.test.js +204 -0
  9. package/v2Components/CapActionButton/constants.js +7 -0
  10. package/v2Components/CapActionButton/index.js +167 -109
  11. package/v2Components/CapActionButton/index.scss +157 -6
  12. package/v2Components/CapActionButton/messages.js +19 -3
  13. package/v2Components/CapActionButton/tests/index.test.js +41 -17
  14. package/v2Components/CapCustomSkeleton/index.js +1 -1
  15. package/v2Components/CapCustomSkeleton/tests/__snapshots__/index.test.js.snap +12 -12
  16. package/v2Components/CapTagList/index.js +10 -0
  17. package/v2Components/CommonTestAndPreview/CustomValuesEditor.js +70 -49
  18. package/v2Components/CommonTestAndPreview/DeliverySettings/DeliverySettings.scss +8 -2
  19. package/v2Components/CommonTestAndPreview/DeliverySettings/ModifyDeliverySettings.js +207 -21
  20. package/v2Components/CommonTestAndPreview/DeliverySettings/constants.js +16 -0
  21. package/v2Components/CommonTestAndPreview/DeliverySettings/index.js +85 -10
  22. package/v2Components/CommonTestAndPreview/DeliverySettings/messages.js +30 -0
  23. package/v2Components/CommonTestAndPreview/DeliverySettings/utils/parseSenderDetailsResponse.js +79 -11
  24. package/v2Components/CommonTestAndPreview/SendTestMessage.js +10 -5
  25. package/v2Components/CommonTestAndPreview/UnifiedPreview/RcsPreviewContent.js +160 -15
  26. package/v2Components/CommonTestAndPreview/UnifiedPreview/RcsPreviewContent.js.rej +18 -0
  27. package/v2Components/CommonTestAndPreview/UnifiedPreview/_unifiedPreview.scss +341 -76
  28. package/v2Components/CommonTestAndPreview/UnifiedPreview/index.js +133 -4
  29. package/v2Components/CommonTestAndPreview/_commonTestAndPreview.scss +11 -0
  30. package/v2Components/CommonTestAndPreview/constants.js +38 -2
  31. package/v2Components/CommonTestAndPreview/index.js +676 -186
  32. package/v2Components/CommonTestAndPreview/messages.js +49 -3
  33. package/v2Components/CommonTestAndPreview/previewApiUtils.js +59 -0
  34. package/v2Components/CommonTestAndPreview/sagas.js +15 -6
  35. package/v2Components/CommonTestAndPreview/tests/CustomValuesEditor.test.js +308 -284
  36. package/v2Components/CommonTestAndPreview/tests/DeliverySettings/ModifyDeliverySettings.test.js +231 -65
  37. package/v2Components/CommonTestAndPreview/tests/DeliverySettings/index.test.js +118 -5
  38. package/v2Components/CommonTestAndPreview/tests/DeliverySettings/utils/parseSenderDetailsResponse.test.js +341 -0
  39. package/v2Components/CommonTestAndPreview/tests/PreviewSection.test.js +8 -1
  40. package/v2Components/CommonTestAndPreview/tests/SendTestMessage.test.js +34 -13
  41. package/v2Components/CommonTestAndPreview/tests/UnifiedPreview/RcsPreviewContent.test.js +281 -283
  42. package/v2Components/CommonTestAndPreview/tests/UnifiedPreview/index.test.js +199 -1
  43. package/v2Components/CommonTestAndPreview/tests/index.test.js +132 -4
  44. package/v2Components/CommonTestAndPreview/tests/previewApiUtils.test.js +67 -0
  45. package/v2Components/CommonTestAndPreview/tests/sagas.test.js +2 -2
  46. package/v2Components/FormBuilder/index.js +8 -10
  47. package/v2Components/SmsFallback/SmsFallbackLocalSelector.js +87 -0
  48. package/v2Components/SmsFallback/constants.js +73 -0
  49. package/v2Components/SmsFallback/index.js +955 -0
  50. package/v2Components/SmsFallback/index.scss +265 -0
  51. package/v2Components/SmsFallback/messages.js +78 -0
  52. package/v2Components/SmsFallback/smsFallbackUtils.js +118 -0
  53. package/v2Components/SmsFallback/tests/SmsFallbackLocalSelector.test.js +50 -0
  54. package/v2Components/SmsFallback/tests/rcsSmsFallback.acceptance.test.js +147 -0
  55. package/v2Components/SmsFallback/tests/smsFallbackHandlers.test.js +304 -0
  56. package/v2Components/SmsFallback/tests/smsFallbackUi.test.js +197 -0
  57. package/v2Components/SmsFallback/tests/smsFallbackUtils.test.js +277 -0
  58. package/v2Components/SmsFallback/tests/useLocalTemplateList.test.js +422 -0
  59. package/v2Components/SmsFallback/useLocalTemplateList.js +92 -0
  60. package/v2Components/TemplatePreview/_templatePreview.scss +33 -23
  61. package/v2Components/TemplatePreview/constants.js +2 -0
  62. package/v2Components/TemplatePreview/index.js +143 -28
  63. package/v2Components/TemplatePreview/tests/index.test.js +142 -0
  64. package/v2Components/TestAndPreviewSlidebox/index.js +13 -1
  65. package/v2Components/TestAndPreviewSlidebox/sagas.js +11 -4
  66. package/v2Components/TestAndPreviewSlidebox/tests/saga.test.js +3 -1
  67. package/v2Components/VarSegmentMessageEditor/constants.js +2 -0
  68. package/v2Components/VarSegmentMessageEditor/index.js +125 -0
  69. package/v2Components/VarSegmentMessageEditor/index.scss +46 -0
  70. package/v2Containers/CreativesContainer/CreativesSlideBoxWrapper.js +43 -0
  71. package/v2Containers/CreativesContainer/SlideBoxContent.js +36 -4
  72. package/v2Containers/CreativesContainer/SlideBoxFooter.js +11 -4
  73. package/v2Containers/CreativesContainer/SlideBoxHeader.js +29 -4
  74. package/v2Containers/CreativesContainer/constants.js +9 -0
  75. package/v2Containers/CreativesContainer/embeddedSlideboxUtils.js +67 -0
  76. package/v2Containers/CreativesContainer/index.js +300 -108
  77. package/v2Containers/CreativesContainer/index.scss +51 -1
  78. package/v2Containers/CreativesContainer/messages.js +0 -4
  79. package/v2Containers/CreativesContainer/tests/SlideBoxContent.localTemplates.test.js +90 -0
  80. package/v2Containers/CreativesContainer/tests/SlideBoxFooter.test.js +78 -34
  81. package/v2Containers/CreativesContainer/tests/SlideBoxHeader.test.js +79 -16
  82. package/v2Containers/CreativesContainer/tests/__snapshots__/SlideBoxContent.test.js.snap +8 -0
  83. package/v2Containers/CreativesContainer/tests/__snapshots__/SlideBoxHeader.test.js.snap +357 -98
  84. package/v2Containers/CreativesContainer/tests/__snapshots__/index.test.js.snap +20 -18
  85. package/v2Containers/CreativesContainer/tests/embeddedSlideboxUtils.test.js +258 -0
  86. package/v2Containers/CreativesContainer/tests/index.test.js +71 -9
  87. package/v2Containers/CreativesContainer/tests/useLocalTemplatesProp.test.js +125 -0
  88. package/v2Containers/Rcs/constants.js +119 -8
  89. package/v2Containers/Rcs/index.js +2379 -807
  90. package/v2Containers/Rcs/index.js.rej +1336 -0
  91. package/v2Containers/Rcs/index.scss +276 -6
  92. package/v2Containers/Rcs/index.scss.rej +74 -0
  93. package/v2Containers/Rcs/messages.js +38 -3
  94. package/v2Containers/Rcs/rcsLibraryHydrationUtils.js +225 -0
  95. package/v2Containers/Rcs/tests/__snapshots__/index.test.js.snap +98018 -70073
  96. package/v2Containers/Rcs/tests/__snapshots__/utils.test.js.snap +0 -5
  97. package/v2Containers/Rcs/tests/__snapshots__/utils.test.js.snap.rej +128 -0
  98. package/v2Containers/Rcs/tests/index.test.js +152 -121
  99. package/v2Containers/Rcs/tests/mockData.js +38 -0
  100. package/v2Containers/Rcs/tests/rcsLibraryHydrationUtils.test.js +318 -0
  101. package/v2Containers/Rcs/tests/utils.test.js +646 -30
  102. package/v2Containers/Rcs/utils.js +478 -11
  103. package/v2Containers/Sms/Create/index.js +100 -40
  104. package/v2Containers/Sms/smsFormDataHelpers.js +67 -0
  105. package/v2Containers/Sms/tests/smsFormDataHelpers.test.js +253 -0
  106. package/v2Containers/SmsTrai/Create/index.js +9 -4
  107. package/v2Containers/SmsTrai/Edit/constants.js +2 -0
  108. package/v2Containers/SmsTrai/Edit/index.js +636 -130
  109. package/v2Containers/SmsTrai/Edit/index.scss +121 -0
  110. package/v2Containers/SmsTrai/Edit/messages.js +14 -4
  111. package/v2Containers/SmsTrai/Edit/tests/__snapshots__/index.test.js.snap +4328 -2375
  112. package/v2Containers/SmsWrapper/index.js +37 -8
  113. package/v2Containers/TagList/index.js +6 -0
  114. package/v2Containers/Templates/ChannelTypeIllustration.js +6 -23
  115. package/v2Containers/Templates/TemplatesActionBar.js +101 -0
  116. package/v2Containers/Templates/_templates.scss +181 -126
  117. package/v2Containers/Templates/actions.js +11 -36
  118. package/v2Containers/Templates/constants.js +2 -23
  119. package/v2Containers/Templates/index.js +142 -333
  120. package/v2Containers/Templates/messages.js +0 -68
  121. package/v2Containers/Templates/reducer.js +0 -68
  122. package/v2Containers/Templates/sagas.js +55 -98
  123. package/v2Containers/Templates/selectors.js +0 -12
  124. package/v2Containers/Templates/tests/ChannelTypeIllustration.test.js +0 -12
  125. package/v2Containers/Templates/tests/TemplatesActionBar.test.js +120 -0
  126. package/v2Containers/Templates/tests/__snapshots__/index.test.js.snap +1042 -1256
  127. package/v2Containers/Templates/tests/index.test.js +0 -6
  128. package/v2Containers/Templates/tests/reducer.test.js +0 -178
  129. package/v2Containers/Templates/tests/sagas.test.js +200 -436
  130. package/v2Containers/Templates/tests/selector.test.js +0 -32
  131. package/v2Containers/Templates/tests/smsTemplatesListApi.test.js +180 -0
  132. package/v2Containers/Templates/utils/smsTemplatesListApi.js +79 -0
  133. package/v2Containers/TemplatesV2/TemplatesV2.style.js +72 -1
  134. package/v2Containers/TemplatesV2/index.js +86 -23
  135. package/v2Containers/TemplatesV2/tests/TemplatesV2.localTemplates.test.js +131 -0
  136. package/v2Containers/Whatsapp/index.js +3 -20
  137. package/v2Containers/Whatsapp/tests/__snapshots__/index.test.js.snap +578 -34
  138. package/v2Containers/Assets/images/archive_Empty_Illustration.svg +0 -9
@@ -55,6 +55,33 @@ describe('parseSenderDetailsResponse', () => {
55
55
  expect(result.domains[0].cdmaSenders[0].value).toBe('CDMA1');
56
56
  });
57
57
 
58
+ it('should populate SMS senders when channel argument is lowercase (normalized for branches)', () => {
59
+ const response = {
60
+ entity: {
61
+ SMS: [
62
+ {
63
+ id: 10,
64
+ priority: 1,
65
+ domainProperties: {
66
+ domainName: 'SMS Domain 1',
67
+ id: 100,
68
+ contactInfo: [
69
+ { type: 'gsm_sender_id', valid: true, value: 'GSM1', default: true },
70
+ { type: 'cdma_sender_id', valid: true, value: 'CDMA1' },
71
+ ],
72
+ connectionProperties: {},
73
+ },
74
+ },
75
+ ],
76
+ },
77
+ };
78
+ const result = parseSenderDetailsResponse('sms', response);
79
+ expect(result.domains).toHaveLength(1);
80
+ expect(result.domains[0].gsmSenders).toHaveLength(1);
81
+ expect(result.domains[0].gsmSenders[0].value).toBe('GSM1');
82
+ expect(result.domains[0].cdmaSenders).toHaveLength(1);
83
+ });
84
+
58
85
  it('should parse EMAIL channel entity array', () => {
59
86
  const response = {
60
87
  entity: {
@@ -231,5 +258,319 @@ describe('parseSenderDetailsResponse', () => {
231
258
  const result = parseSenderDetailsResponse('WHATSAPP', response);
232
259
  expect(result.domains[0].sourceAccountIdentifier).toBe('waba-456');
233
260
  });
261
+
262
+ it('should map hostName to domainName when domainName missing (RCS)', () => {
263
+ const response = {
264
+ entity: {
265
+ RCS: [
266
+ {
267
+ id: 1,
268
+ priority: 0,
269
+ domainProperties: {
270
+ hostName: 'rcs-host',
271
+ id: 42,
272
+ contactInfo: [
273
+ { type: 'gsm_sender_id', valid: true, value: 'S1', default: true },
274
+ ],
275
+ },
276
+ },
277
+ ],
278
+ },
279
+ };
280
+ const result = parseSenderDetailsResponse('RCS', response);
281
+ expect(result.domains[0].domainName).toBe('rcs-host');
282
+ expect(result.domains[0].domainId).toBe(42);
283
+ expect(result.domains[0].gsmSenders[0].value).toBe('S1');
284
+ });
285
+
286
+ it('should unwrap result.entity when top-level entity is absent', () => {
287
+ const response = {
288
+ result: {
289
+ entity: {
290
+ SMS: [
291
+ {
292
+ id: 1,
293
+ priority: 0,
294
+ domainProperties: {
295
+ domainName: 'ViaResult',
296
+ id: 9,
297
+ contactInfo: [{ type: 'gsm_sender_id', valid: true, value: 'G' }],
298
+ },
299
+ },
300
+ ],
301
+ },
302
+ },
303
+ };
304
+ const result = parseSenderDetailsResponse('SMS', response);
305
+ expect(result.domains[0].domainName).toBe('ViaResult');
306
+ });
307
+
308
+ it('should merge rcs_sender_id into gsmSenders for RCS', () => {
309
+ const response = {
310
+ entity: {
311
+ RCS: [
312
+ {
313
+ id: 1,
314
+ priority: 0,
315
+ domainProperties: {
316
+ domainName: 'RCS Domain',
317
+ id: 1,
318
+ contactInfo: [
319
+ { type: 'rcs_sender_id', valid: true, value: 'RCS123', default: true },
320
+ ],
321
+ },
322
+ },
323
+ ],
324
+ },
325
+ };
326
+ const result = parseSenderDetailsResponse('RCS', response);
327
+ expect(result.domains[0].gsmSenders).toHaveLength(1);
328
+ expect(result.domains[0].gsmSenders[0].value).toBe('RCS123');
329
+ });
330
+
331
+ it('should unwrap data.entity when top-level entity is absent', () => {
332
+ const response = {
333
+ data: {
334
+ entity: {
335
+ SMS: [
336
+ {
337
+ id: 1,
338
+ priority: 0,
339
+ domainProperties: {
340
+ domainName: 'ViaData',
341
+ id: 9,
342
+ contactInfo: [{ type: 'gsm_sender_id', valid: true, value: 'G' }],
343
+ },
344
+ },
345
+ ],
346
+ },
347
+ },
348
+ };
349
+ const result = parseSenderDetailsResponse('SMS', response);
350
+ expect(result.domains[0].domainName).toBe('ViaData');
351
+ });
352
+
353
+ it('should return empty domains when entity is explicitly null', () => {
354
+ expect(parseSenderDetailsResponse('SMS', { entity: null })).toEqual({ domains: [] });
355
+ });
356
+
357
+ it('should return empty when channel data is a non-domain object', () => {
358
+ expect(parseSenderDetailsResponse('SMS', { entity: { notAnArray: true } })).toEqual({ domains: [] });
359
+ });
360
+
361
+ it('should use userid when sourceAccountIdentifier and wabaId are missing (WHATSAPP)', () => {
362
+ const response = {
363
+ entity: {
364
+ WHATSAPP: [
365
+ {
366
+ id: 1,
367
+ priority: 0,
368
+ domainProperties: {
369
+ domainName: 'W',
370
+ id: 1,
371
+ connectionProperties: { userid: 'user-99' },
372
+ },
373
+ },
374
+ ],
375
+ },
376
+ };
377
+ const result = parseSenderDetailsResponse('WHATSAPP', response);
378
+ expect(result.domains[0].sourceAccountIdentifier).toBe('user-99');
379
+ });
380
+
381
+ it('should dedupe duplicate GSM values for RCS (merged contact types)', () => {
382
+ const response = {
383
+ entity: {
384
+ RCS: [
385
+ {
386
+ id: 1,
387
+ priority: 0,
388
+ domainProperties: {
389
+ domainName: 'R',
390
+ id: 1,
391
+ contactInfo: [
392
+ { type: 'gsm_sender_id', valid: true, value: 'SAME', default: true },
393
+ { type: 'rcs_sender_id', valid: true, value: 'SAME' },
394
+ ],
395
+ },
396
+ },
397
+ ],
398
+ },
399
+ };
400
+ const result = parseSenderDetailsResponse('RCS', response);
401
+ expect(result.domains[0].gsmSenders.filter((r) => r.value === 'SAME')).toHaveLength(1);
402
+ });
403
+
404
+ it('should return empty domains when response is not an object', () => {
405
+ expect(parseSenderDetailsResponse('SMS', 42)).toEqual({ domains: [] });
406
+ expect(parseSenderDetailsResponse('SMS', 'x')).toEqual({ domains: [] });
407
+ });
408
+
409
+ it('should return empty domains when entity is empty string', () => {
410
+ expect(parseSenderDetailsResponse('SMS', { entity: '' })).toEqual({ domains: [] });
411
+ });
412
+
413
+ it('should normalize hyphenated contactInfo.type to match gsm_sender_id', () => {
414
+ const response = {
415
+ entity: {
416
+ SMS: [
417
+ {
418
+ id: 1,
419
+ domainProperties: {
420
+ domainName: 'D',
421
+ id: 1,
422
+ contactInfo: [
423
+ { type: 'gsm-sender-id', valid: true, value: 'HYPH' },
424
+ ],
425
+ },
426
+ },
427
+ ],
428
+ },
429
+ };
430
+ const result = parseSenderDetailsResponse('SMS', response);
431
+ expect(result.domains[0].gsmSenders[0].value).toBe('HYPH');
432
+ });
433
+
434
+ it('should skip whitespace-only GSM values when merging RCS contact types', () => {
435
+ const response = {
436
+ entity: {
437
+ RCS: [
438
+ {
439
+ id: 1,
440
+ domainProperties: {
441
+ domainName: 'R',
442
+ id: 1,
443
+ contactInfo: [
444
+ { type: 'gsm_sender_id', valid: true, value: ' ' },
445
+ { type: 'gsm_sender_id', valid: true, value: 'OK' },
446
+ ],
447
+ },
448
+ },
449
+ ],
450
+ },
451
+ };
452
+ const values = parseSenderDetailsResponse('RCS', response).domains[0].gsmSenders.map((g) => g.value);
453
+ expect(values).toEqual(['OK']);
454
+ });
455
+
456
+ it('should use default priority 0 when priority is null', () => {
457
+ const response = {
458
+ entity: {
459
+ SMS: [
460
+ {
461
+ id: 1,
462
+ priority: null,
463
+ domainProperties: {
464
+ domainName: 'P',
465
+ id: 9,
466
+ contactInfo: [],
467
+ },
468
+ },
469
+ ],
470
+ },
471
+ };
472
+ expect(parseSenderDetailsResponse('SMS', response).domains[0].priority).toBe(0);
473
+ });
474
+
475
+ it('should unwrap entity from raw response object when entity key is absent', () => {
476
+ const response = { misc: 1 };
477
+ expect(parseSenderDetailsResponse('SMS', response)).toEqual({ domains: [] });
478
+ });
479
+ });
480
+
481
+ describe('unwrapEntity edge cases', () => {
482
+ it('should return empty domains when entity key is explicitly null', () => {
483
+ // unwrapEntity: response.entity === null → returns null → parseSenderDetailsResponse returns { domains: [] }
484
+ expect(parseSenderDetailsResponse('SMS', { entity: null })).toEqual({ domains: [] });
485
+ });
486
+ });
487
+
488
+ describe('null or empty channel', () => {
489
+ it('should return empty domains when channel is null', () => {
490
+ const response = { entity: { SMS: [{ id: 1, domainProperties: { domainName: 'D', id: 10, contactInfo: [] } }] } };
491
+ expect(parseSenderDetailsResponse(null, response)).toEqual({ domains: [] });
492
+ });
493
+
494
+ it('should return empty domains when channel is empty string', () => {
495
+ const response = { entity: { SMS: [{ id: 1, domainProperties: { domainName: 'D', id: 10, contactInfo: [] } }] } };
496
+ expect(parseSenderDetailsResponse('', response)).toEqual({ domains: [] });
497
+ });
498
+ });
499
+
500
+ describe('single non-array domain object', () => {
501
+ it('should wrap single domain with domainProperties in an array and return one domain', () => {
502
+ const response = {
503
+ entity: {
504
+ SMS: {
505
+ id: 5,
506
+ domainProperties: {
507
+ domainName: 'SingleDomain',
508
+ id: 99,
509
+ contactInfo: [{ type: 'gsm_sender_id', valid: true, value: 'GSMSINGLE' }],
510
+ },
511
+ },
512
+ },
513
+ };
514
+ const result = parseSenderDetailsResponse('SMS', response);
515
+ expect(result.domains).toHaveLength(1);
516
+ expect(result.domains[0].domainName).toBe('SingleDomain');
517
+ expect(result.domains[0].gsmSenders[0].value).toBe('GSMSINGLE');
518
+ });
519
+
520
+ it('should return empty domains for single non-array domain without domainProperties', () => {
521
+ const response = {
522
+ entity: { SMS: { id: 5, someProp: 'value' } },
523
+ };
524
+ expect(parseSenderDetailsResponse('SMS', response)).toEqual({ domains: [] });
525
+ });
526
+ });
527
+
528
+ describe('deduplication by domainId when domainName is null', () => {
529
+ it('should deduplicate two entries sharing the same domainId when domainName is null', () => {
530
+ const response = {
531
+ entity: {
532
+ SMS: [
533
+ {
534
+ id: 10,
535
+ priority: 1,
536
+ domainProperties: { domainName: null, id: 42, contactInfo: [] },
537
+ },
538
+ {
539
+ id: 11,
540
+ priority: 2,
541
+ domainProperties: { domainName: null, id: 42, contactInfo: [] },
542
+ },
543
+ ],
544
+ },
545
+ };
546
+ const result = parseSenderDetailsResponse('SMS', response);
547
+ expect(result.domains).toHaveLength(1);
548
+ expect(result.domains[0].domainId).toBe(42);
549
+ });
550
+ });
551
+
552
+ describe('typeMatches null-type filtering', () => {
553
+ it('should exclude contactInfo items with null type from gsmSenders', () => {
554
+ const response = {
555
+ entity: {
556
+ SMS: [
557
+ {
558
+ id: 1,
559
+ domainProperties: {
560
+ domainName: 'D',
561
+ id: 1,
562
+ contactInfo: [
563
+ { type: null, valid: true, value: 'EXCLUDED' },
564
+ { type: 'gsm_sender_id', valid: true, value: 'INCLUDED' },
565
+ ],
566
+ },
567
+ },
568
+ ],
569
+ },
570
+ };
571
+ const result = parseSenderDetailsResponse('SMS', response);
572
+ expect(result.domains[0].gsmSenders).toHaveLength(1);
573
+ expect(result.domains[0].gsmSenders[0].value).toBe('INCLUDED');
574
+ });
234
575
  });
235
576
  });
@@ -8,7 +8,7 @@ import React from 'react';
8
8
  import { render, screen } from '@testing-library/react';
9
9
  import PreviewSection from '../PreviewSection';
10
10
 
11
- // Mock UnifiedPreview component
11
+ // Mock UnifiedPreview component (lightweight DOM for this suite only).
12
12
  const mockUnifiedPreview = jest.fn();
13
13
  jest.mock('../UnifiedPreview', () => {
14
14
  // eslint-disable-next-line global-require, import/no-extraneous-dependencies
@@ -232,3 +232,10 @@ describe('PreviewSection', () => {
232
232
  });
233
233
  });
234
234
  });
235
+
236
+ // This suite mocks ../UnifiedPreview. resetModules() alone is not enough — Jest still applies the
237
+ // manual mock on the next load. Unmock + reset so later test files in this worker get the real module.
238
+ afterAll(() => {
239
+ jest.unmock('../UnifiedPreview');
240
+ jest.resetModules();
241
+ });
@@ -24,17 +24,19 @@ import SendTestMessage from '../SendTestMessage';
24
24
 
25
25
  // Mock DeliverySettings to assert props
26
26
  jest.mock('../DeliverySettings', () => function MockDeliverySettings(props) {
27
- return (
27
+ const smsRows = props.senderDetailsByChannel?.SMS || [];
28
+ return (
28
29
  <div data-testid="delivery-settings" data-props={JSON.stringify({
29
30
  channel: props.channel,
30
31
  hasDeliverySettings: !!props.deliverySettings,
31
- senderDetailsLength: (props.senderDetailsOptions || []).length,
32
+ senderDetailsLength: smsRows.length,
32
33
  wecrmAccountsLength: (props.wecrmAccounts || []).length,
33
34
  hasOnSave: typeof props.onSaveDeliverySettings === 'function',
34
35
  isLoadingSenderDetails: props.isLoadingSenderDetails,
35
36
  smsTraiDltEnabled: props.smsTraiDltEnabled,
36
37
  registeredSenderIds: props.registeredSenderIds,
37
38
  whatsappAccountFromForm: props.whatsappAccountFromForm,
39
+ isChannelSmsFallbackPreviewEnabled: props.isChannelSmsFallbackPreviewEnabled,
38
40
  })}
39
41
  />
40
42
  );
@@ -129,7 +131,7 @@ describe('SendTestMessage', () => {
129
131
  formatMessage: jest.fn((msg) => msg.defaultMessage || msg.id),
130
132
  channel: 'EMAIL',
131
133
  deliverySettings: {},
132
- senderDetailsOptions: [],
134
+ senderDetailsByChannel: {},
133
135
  wecrmAccounts: [],
134
136
  onSaveDeliverySettings: jest.fn(),
135
137
  isLoadingSenderDetails: false,
@@ -233,6 +235,22 @@ describe('SendTestMessage', () => {
233
235
  expect(screen.getByTestId('delivery-settings')).toBeTruthy();
234
236
  });
235
237
 
238
+ it('should render DeliverySettings when channel is RCS and pass SMS fallback preview flag', () => {
239
+ render(
240
+ <TestWrapper>
241
+ <SendTestMessage
242
+ {...defaultProps}
243
+ channel="RCS"
244
+ isChannelSmsFallbackPreviewEnabled
245
+ />
246
+ </TestWrapper>
247
+ );
248
+ const el = screen.getByTestId('delivery-settings');
249
+ const data = JSON.parse(el.getAttribute('data-props'));
250
+ expect(data.channel).toBe('RCS');
251
+ expect(data.isChannelSmsFallbackPreviewEnabled).toBe(true);
252
+ });
253
+
236
254
  it('should not render DeliverySettings when channel is INAPP', () => {
237
255
  render(
238
256
  <TestWrapper>
@@ -280,7 +298,9 @@ describe('SendTestMessage', () => {
280
298
  {...defaultProps}
281
299
  channel="SMS"
282
300
  deliverySettings={{ domainId: 1 }}
283
- senderDetailsOptions={[{ domainId: 1, domainName: 'SMS Dom' }]}
301
+ senderDetailsByChannel={{
302
+ SMS: [{ domainId: 1, domainName: 'SMS Dom' }],
303
+ }}
284
304
  wecrmAccounts={[]}
285
305
  onSaveDeliverySettings={onSave}
286
306
  isLoadingSenderDetails={true}
@@ -668,21 +688,22 @@ describe('SendTestMessage', () => {
668
688
  ...defaultProps,
669
689
  renderAddTestCustomerButton,
670
690
  };
671
- render(
691
+ const { container } = render(
672
692
  <TestWrapper>
673
693
  <SendTestMessage {...props} />
674
694
  </TestWrapper>
675
695
  );
676
- // Assert component wiring: renderAddTestCustomerButton must have been called during render.
677
- expect(renderAddTestCustomerButton).toHaveBeenCalled();
678
- // Assert the returned element has the expected shape.
679
- const notFoundContent = renderAddTestCustomerButton.mock.results[0].value;
696
+ // notFoundContent is passed to TreeSelect and shown when dropdown is open with no match.
697
+ // We assert the render function returns a button that can be clicked.
698
+ const notFoundContent = renderAddTestCustomerButton.mock.results[0]?.value;
680
699
  expect(notFoundContent).toBeTruthy();
681
- expect(notFoundContent.type).toBe('button');
682
700
  expect(notFoundContent.props['data-testid']).toBe('add-test-customer-btn');
683
- // Assert clicking the button triggers the handler.
684
- notFoundContent.props.onClick();
685
- expect(onAddClick).toHaveBeenCalled();
701
+ expect(notFoundContent.type).toBe('button');
702
+ // Simulate click on the returned element's onClick
703
+ if (notFoundContent.props.onClick) {
704
+ notFoundContent.props.onClick();
705
+ expect(onAddClick).toHaveBeenCalled();
706
+ }
686
707
  });
687
708
  });
688
709
  });