cap-creatives-ui 8.0.280 → 8.0.321

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 (247) hide show
  1. package/.github/workflows/pr-title-check.yml +88 -0
  2. package/app/constants/unified.js +21 -1
  3. package/app/containers/App/constants.js +0 -1
  4. package/app/containers/Login/test/index.test.js +123 -0
  5. package/app/containers/Login/test/selectors.test.js +165 -0
  6. package/app/initialState.js +0 -2
  7. package/app/services/api.js +6 -0
  8. package/app/services/tests/api.test.js +7 -0
  9. package/app/services/tests/getSchema.test.js +95 -0
  10. package/app/utils/common.js +23 -9
  11. package/app/utils/commonUtils.js +64 -93
  12. package/app/utils/tagValidations.js +83 -219
  13. package/app/utils/templateVarUtils.js +172 -0
  14. package/app/utils/tests/common.test.js +265 -323
  15. package/app/utils/tests/commonUtil.test.js +461 -118
  16. package/app/utils/tests/commonUtils.test.js +581 -0
  17. package/app/utils/tests/messageUtils.test.js +95 -0
  18. package/app/utils/tests/smsCharCount.test.js +304 -0
  19. package/app/utils/tests/smsCharCountV2.test.js +213 -10
  20. package/app/utils/tests/tagValidations.test.js +474 -357
  21. package/app/utils/tests/templateVarUtils.test.js +160 -0
  22. package/app/v2Components/CapDeviceContent/index.js +10 -7
  23. package/app/v2Components/CapTagList/index.js +32 -24
  24. package/app/v2Components/CapTagList/style.scss +48 -0
  25. package/app/v2Components/CapTagListWithInput/__tests__/CapTagListWithInput.test.js +63 -0
  26. package/app/v2Components/CapTagListWithInput/index.js +8 -0
  27. package/app/v2Components/CapWhatsappCTA/index.js +2 -0
  28. package/app/v2Components/CapWhatsappCarouselButton/index.js +32 -14
  29. package/app/v2Components/CapWhatsappCarouselButton/tests/index.test.js +120 -2
  30. package/app/v2Components/CommonTestAndPreview/CustomValuesEditor.js +70 -49
  31. package/app/v2Components/CommonTestAndPreview/DeliverySettings/DeliverySettings.scss +39 -0
  32. package/app/v2Components/CommonTestAndPreview/DeliverySettings/ModifyDeliverySettings.js +606 -0
  33. package/app/v2Components/CommonTestAndPreview/DeliverySettings/ModifyDeliverySettings.scss +36 -0
  34. package/app/v2Components/CommonTestAndPreview/DeliverySettings/constants.js +79 -0
  35. package/app/v2Components/CommonTestAndPreview/DeliverySettings/index.js +314 -0
  36. package/app/v2Components/CommonTestAndPreview/DeliverySettings/messages.js +141 -0
  37. package/app/v2Components/CommonTestAndPreview/DeliverySettings/utils/parseSenderDetailsResponse.js +156 -0
  38. package/app/v2Components/CommonTestAndPreview/SendTestMessage.js +57 -1
  39. package/app/v2Components/CommonTestAndPreview/UnifiedPreview/_unifiedPreview.scss +20 -1
  40. package/app/v2Components/CommonTestAndPreview/UnifiedPreview/index.js +133 -4
  41. package/app/v2Components/CommonTestAndPreview/_commonTestAndPreview.scss +210 -4
  42. package/app/v2Components/CommonTestAndPreview/actions.js +20 -0
  43. package/app/v2Components/CommonTestAndPreview/constants.js +57 -1
  44. package/app/v2Components/CommonTestAndPreview/index.js +878 -156
  45. package/app/v2Components/CommonTestAndPreview/messages.js +41 -3
  46. package/app/v2Components/CommonTestAndPreview/previewApiUtils.js +59 -0
  47. package/app/v2Components/CommonTestAndPreview/reducer.js +47 -0
  48. package/app/v2Components/CommonTestAndPreview/sagas.js +75 -5
  49. package/app/v2Components/CommonTestAndPreview/selectors.js +51 -0
  50. package/app/v2Components/CommonTestAndPreview/tests/CustomValuesEditor.test.js +352 -0
  51. package/app/v2Components/CommonTestAndPreview/tests/DeliverySettings/ModifyDeliverySettings.test.js +1156 -0
  52. package/app/v2Components/CommonTestAndPreview/tests/DeliverySettings/index.test.js +334 -0
  53. package/app/v2Components/CommonTestAndPreview/tests/DeliverySettings/utils/parseSenderDetailsResponse.test.js +576 -0
  54. package/app/v2Components/CommonTestAndPreview/tests/SendTestMessage.test.js +156 -0
  55. package/app/v2Components/CommonTestAndPreview/tests/UnifiedPreview/index.test.js +199 -1
  56. package/app/v2Components/CommonTestAndPreview/tests/actions.test.js +50 -0
  57. package/app/v2Components/CommonTestAndPreview/tests/constants.test.js +18 -7
  58. package/app/v2Components/CommonTestAndPreview/tests/index.test.js +914 -5
  59. package/app/v2Components/CommonTestAndPreview/tests/previewApiUtils.test.js +67 -0
  60. package/app/v2Components/CommonTestAndPreview/tests/reducer.test.js +118 -0
  61. package/app/v2Components/CommonTestAndPreview/tests/sagas.test.js +146 -378
  62. package/app/v2Components/CommonTestAndPreview/tests/selectors.test.js +146 -0
  63. package/app/v2Components/ErrorInfoNote/index.js +24 -26
  64. package/app/v2Components/FormBuilder/index.js +182 -204
  65. package/app/v2Components/FormBuilder/messages.js +4 -8
  66. package/app/v2Components/HtmlEditor/HTMLEditor.js +7 -6
  67. package/app/v2Components/HtmlEditor/__tests__/HTMLEditor.apiErrors.test.js +1 -1
  68. package/app/v2Components/HtmlEditor/__tests__/HTMLEditor.test.js +928 -17
  69. package/app/v2Components/HtmlEditor/components/CodeEditorPane/index.js +4 -2
  70. package/app/v2Components/HtmlEditor/hooks/__tests__/useValidation.test.js +452 -3
  71. package/app/v2Components/HtmlEditor/hooks/useValidation.js +12 -9
  72. package/app/v2Components/HtmlEditor/utils/__tests__/htmlValidator.enhanced.test.js +132 -0
  73. package/app/v2Components/HtmlEditor/utils/htmlValidator.js +4 -2
  74. package/app/v2Components/SmsFallback/SmsFallbackLocalSelector.js +87 -0
  75. package/app/v2Components/SmsFallback/constants.js +73 -0
  76. package/app/v2Components/SmsFallback/index.js +956 -0
  77. package/app/v2Components/SmsFallback/index.scss +265 -0
  78. package/app/v2Components/SmsFallback/messages.js +78 -0
  79. package/app/v2Components/SmsFallback/smsFallbackUtils.js +107 -0
  80. package/app/v2Components/SmsFallback/tests/SmsFallbackLocalSelector.test.js +50 -0
  81. package/app/v2Components/SmsFallback/tests/rcsSmsFallback.acceptance.test.js +147 -0
  82. package/app/v2Components/SmsFallback/tests/smsFallbackHandlers.test.js +304 -0
  83. package/app/v2Components/SmsFallback/tests/smsFallbackUi.test.js +197 -0
  84. package/app/v2Components/SmsFallback/tests/smsFallbackUtils.test.js +261 -0
  85. package/app/v2Components/SmsFallback/tests/useLocalTemplateList.test.js +422 -0
  86. package/app/v2Components/SmsFallback/useLocalTemplateList.js +92 -0
  87. package/app/v2Components/TestAndPreviewSlidebox/index.js +22 -1
  88. package/app/v2Components/TestAndPreviewSlidebox/sagas.js +11 -4
  89. package/app/v2Components/TestAndPreviewSlidebox/tests/saga.test.js +3 -1
  90. package/app/v2Components/VarSegmentMessageEditor/constants.js +2 -0
  91. package/app/v2Components/VarSegmentMessageEditor/index.js +125 -0
  92. package/app/v2Components/VarSegmentMessageEditor/index.scss +46 -0
  93. package/app/v2Containers/BeeEditor/index.js +3 -0
  94. package/app/v2Containers/BeePopupEditor/index.js +9 -2
  95. package/app/v2Containers/Cap/mockData.js +0 -14
  96. package/app/v2Containers/Cap/reducer.js +3 -55
  97. package/app/v2Containers/Cap/tests/reducer.test.js +0 -102
  98. package/app/v2Containers/CommunicationFlow/CommunicationFlow.js +291 -0
  99. package/app/v2Containers/CommunicationFlow/CommunicationFlow.scss +25 -0
  100. package/app/v2Containers/CommunicationFlow/Tests/CommunicationFlow.test.js +255 -0
  101. package/app/v2Containers/CommunicationFlow/constants.js +200 -0
  102. package/app/v2Containers/CommunicationFlow/index.js +102 -0
  103. package/app/v2Containers/CommunicationFlow/messages.js +346 -0
  104. package/app/v2Containers/CommunicationFlow/steps/ChannelSelectionStep/ChannelSelectionStep.js +522 -0
  105. package/app/v2Containers/CommunicationFlow/steps/ChannelSelectionStep/ChannelSelectionStep.scss +170 -0
  106. package/app/v2Containers/CommunicationFlow/steps/ChannelSelectionStep/Tests/ChannelSelectionStep.test.js +796 -0
  107. package/app/v2Containers/CommunicationFlow/steps/ChannelSelectionStep/index.js +5 -0
  108. package/app/v2Containers/CommunicationFlow/steps/CommunicationStrategyStep/CommunicationStrategyStep.js +95 -0
  109. package/app/v2Containers/CommunicationFlow/steps/CommunicationStrategyStep/Tests/CommunicationStrategyStep.test.js +133 -0
  110. package/app/v2Containers/CommunicationFlow/steps/CommunicationStrategyStep/index.js +5 -0
  111. package/app/v2Containers/CommunicationFlow/steps/DeliverySettingsStep/DeliverySettingsSection.js +289 -0
  112. package/app/v2Containers/CommunicationFlow/steps/DeliverySettingsStep/DeliverySettingsSection.scss +70 -0
  113. package/app/v2Containers/CommunicationFlow/steps/DeliverySettingsStep/SenderDetails.js +319 -0
  114. package/app/v2Containers/CommunicationFlow/steps/DeliverySettingsStep/SenderDetails.scss +69 -0
  115. package/app/v2Containers/CommunicationFlow/steps/DeliverySettingsStep/Tests/DeliverySettingsSection.test.js +616 -0
  116. package/app/v2Containers/CommunicationFlow/steps/DeliverySettingsStep/Tests/SenderDetails.test.js +577 -0
  117. package/app/v2Containers/CommunicationFlow/steps/DeliverySettingsStep/Tests/deliverySettingsConfig.test.js +1111 -0
  118. package/app/v2Containers/CommunicationFlow/steps/DeliverySettingsStep/deliverySettingsConfig.js +696 -0
  119. package/app/v2Containers/CommunicationFlow/steps/DeliverySettingsStep/index.js +7 -0
  120. package/app/v2Containers/CommunicationFlow/steps/DynamicControlsStep/DynamicControlsStep.js +102 -0
  121. package/app/v2Containers/CommunicationFlow/steps/DynamicControlsStep/DynamicControlsStep.scss +36 -0
  122. package/app/v2Containers/CommunicationFlow/steps/DynamicControlsStep/Tests/DynamicControlsStep.test.js +91 -0
  123. package/app/v2Containers/CommunicationFlow/steps/DynamicControlsStep/index.js +5 -0
  124. package/app/v2Containers/CommunicationFlow/steps/MessageTypeStep/MessageTypeStep.js +86 -0
  125. package/app/v2Containers/CommunicationFlow/steps/MessageTypeStep/Tests/MessageTypeStep.test.js +100 -0
  126. package/app/v2Containers/CommunicationFlow/steps/MessageTypeStep/index.js +5 -0
  127. package/app/v2Containers/CommunicationFlow/utils/getEnabledSteps.js +30 -0
  128. package/app/v2Containers/CreativesContainer/CreativesSlideBoxWrapper.js +43 -0
  129. package/app/v2Containers/CreativesContainer/SlideBoxContent.js +127 -11
  130. package/app/v2Containers/CreativesContainer/SlideBoxFooter.js +62 -9
  131. package/app/v2Containers/CreativesContainer/SlideBoxHeader.js +29 -4
  132. package/app/v2Containers/CreativesContainer/constants.js +24 -0
  133. package/app/v2Containers/CreativesContainer/embeddedSlideboxUtils.js +67 -0
  134. package/app/v2Containers/CreativesContainer/index.js +346 -71
  135. package/app/v2Containers/CreativesContainer/index.scss +51 -1
  136. package/app/v2Containers/CreativesContainer/messages.js +12 -0
  137. package/app/v2Containers/CreativesContainer/tests/SlideBoxContent.localTemplates.test.js +90 -0
  138. package/app/v2Containers/CreativesContainer/tests/SlideBoxContent.test.js +69 -1
  139. package/app/v2Containers/CreativesContainer/tests/SlideBoxFooter.test.js +443 -0
  140. package/app/v2Containers/CreativesContainer/tests/SlideBoxHeader.test.js +110 -0
  141. package/app/v2Containers/CreativesContainer/tests/__snapshots__/SlideBoxContent.test.js.snap +147 -4
  142. package/app/v2Containers/CreativesContainer/tests/__snapshots__/SlideBoxHeader.test.js.snap +363 -0
  143. package/app/v2Containers/CreativesContainer/tests/__snapshots__/index.test.js.snap +57 -10
  144. package/app/v2Containers/CreativesContainer/tests/embeddedSlideboxUtils.test.js +258 -0
  145. package/app/v2Containers/CreativesContainer/tests/index.test.js +71 -9
  146. package/app/v2Containers/CreativesContainer/tests/useLocalTemplatesProp.test.js +125 -0
  147. package/app/v2Containers/Email/index.js +2 -5
  148. package/app/v2Containers/EmailWrapper/components/EmailHTMLEditor.js +58 -77
  149. package/app/v2Containers/EmailWrapper/components/EmailWrapperView.js +3 -0
  150. package/app/v2Containers/EmailWrapper/components/__tests__/EmailHTMLEditor.test.js +158 -89
  151. package/app/v2Containers/EmailWrapper/components/__tests__/EmailWrapperView.test.js +16 -1
  152. package/app/v2Containers/EmailWrapper/hooks/useEmailWrapper.js +17 -12
  153. package/app/v2Containers/EmailWrapper/index.js +4 -0
  154. package/app/v2Containers/EmailWrapper/tests/useEmailWrapper.edgeCases.test.js +1 -0
  155. package/app/v2Containers/EmailWrapper/tests/useEmailWrapper.test.js +133 -0
  156. package/app/v2Containers/FTP/index.js +2 -51
  157. package/app/v2Containers/FTP/messages.js +0 -4
  158. package/app/v2Containers/InApp/__tests__/InAppHTMLEditor.test.js +110 -155
  159. package/app/v2Containers/InApp/index.js +297 -118
  160. package/app/v2Containers/InApp/tests/index.test.js +17 -6
  161. package/app/v2Containers/InApp/tests/mockData.js +1 -1
  162. package/app/v2Containers/InAppWrapper/hooks/__tests__/useInAppWrapper.test.js +19 -0
  163. package/app/v2Containers/InAppWrapper/hooks/useInAppWrapper.js +3 -0
  164. package/app/v2Containers/InAppWrapper/index.js +3 -0
  165. package/app/v2Containers/InappAdvance/index.js +5 -104
  166. package/app/v2Containers/InappAdvance/tests/index.test.js +2 -0
  167. package/app/v2Containers/Line/Container/ImageCarousel/tests/__snapshots__/content.test.js.snap +24 -3
  168. package/app/v2Containers/Line/Container/Text/index.js +0 -1
  169. package/app/v2Containers/MobilePush/Create/index.js +105 -28
  170. package/app/v2Containers/MobilePush/Create/messages.js +4 -0
  171. package/app/v2Containers/MobilePush/Edit/index.js +250 -68
  172. package/app/v2Containers/MobilePush/Edit/messages.js +4 -0
  173. package/app/v2Containers/MobilePushNew/components/PlatformContentFields.js +36 -12
  174. package/app/v2Containers/MobilePushNew/components/tests/PlatformContentFields.test.js +68 -27
  175. package/app/v2Containers/MobilePushNew/index.js +78 -35
  176. package/app/v2Containers/MobilePushNew/messages.js +8 -0
  177. package/app/v2Containers/MobilepushWrapper/index.js +11 -1
  178. package/app/v2Containers/Rcs/constants.js +32 -1
  179. package/app/v2Containers/Rcs/index.js +963 -916
  180. package/app/v2Containers/Rcs/index.scss +85 -6
  181. package/app/v2Containers/Rcs/messages.js +10 -1
  182. package/app/v2Containers/Rcs/rcsLibraryHydrationUtils.js +205 -0
  183. package/app/v2Containers/Rcs/tests/__snapshots__/index.test.js.snap +41136 -1566
  184. package/app/v2Containers/Rcs/tests/__snapshots__/utils.test.js.snap +0 -5
  185. package/app/v2Containers/Rcs/tests/index.test.js +41 -38
  186. package/app/v2Containers/Rcs/tests/mockData.js +38 -0
  187. package/app/v2Containers/Rcs/tests/rcsLibraryHydrationUtils.test.js +251 -0
  188. package/app/v2Containers/Rcs/tests/utils.test.js +379 -1
  189. package/app/v2Containers/Rcs/utils.js +358 -10
  190. package/app/v2Containers/Sms/Create/index.js +122 -39
  191. package/app/v2Containers/Sms/Create/messages.js +4 -0
  192. package/app/v2Containers/Sms/Edit/index.js +37 -3
  193. package/app/v2Containers/Sms/commonMethods.js +3 -6
  194. package/app/v2Containers/Sms/smsFormDataHelpers.js +67 -0
  195. package/app/v2Containers/Sms/tests/commonMethods.test.js +122 -0
  196. package/app/v2Containers/Sms/tests/smsFormDataHelpers.test.js +253 -0
  197. package/app/v2Containers/SmsTrai/Create/index.js +9 -4
  198. package/app/v2Containers/SmsTrai/Create/index.scss +1 -1
  199. package/app/v2Containers/SmsTrai/Edit/constants.js +2 -0
  200. package/app/v2Containers/SmsTrai/Edit/index.js +667 -160
  201. package/app/v2Containers/SmsTrai/Edit/index.scss +121 -0
  202. package/app/v2Containers/SmsTrai/Edit/messages.js +9 -4
  203. package/app/v2Containers/SmsTrai/Edit/tests/__snapshots__/index.test.js.snap +4590 -2436
  204. package/app/v2Containers/SmsWrapper/index.js +41 -8
  205. package/app/v2Containers/TagList/index.js +63 -2
  206. package/app/v2Containers/TagList/messages.js +8 -0
  207. package/app/v2Containers/TagList/tests/TagList.test.js +122 -20
  208. package/app/v2Containers/TagList/tests/mockdata.js +17 -0
  209. package/app/v2Containers/Templates/TemplatesActionBar.js +101 -0
  210. package/app/v2Containers/Templates/_templates.scss +61 -2
  211. package/app/v2Containers/Templates/actions.js +11 -0
  212. package/app/v2Containers/Templates/constants.js +2 -0
  213. package/app/v2Containers/Templates/index.js +90 -40
  214. package/app/v2Containers/Templates/reducer.js +3 -1
  215. package/app/v2Containers/Templates/sagas.js +57 -12
  216. package/app/v2Containers/Templates/tests/TemplatesActionBar.test.js +120 -0
  217. package/app/v2Containers/Templates/tests/__snapshots__/index.test.js.snap +1043 -1079
  218. package/app/v2Containers/Templates/tests/reducer.test.js +12 -0
  219. package/app/v2Containers/Templates/tests/sagas.test.js +193 -12
  220. package/app/v2Containers/Templates/tests/smsTemplatesListApi.test.js +180 -0
  221. package/app/v2Containers/Templates/utils/smsTemplatesListApi.js +79 -0
  222. package/app/v2Containers/TemplatesV2/TemplatesV2.style.js +72 -1
  223. package/app/v2Containers/TemplatesV2/index.js +147 -49
  224. package/app/v2Containers/TemplatesV2/tests/TemplatesV2.localTemplates.test.js +131 -0
  225. package/app/v2Containers/Viber/index.js +9 -10
  226. package/app/v2Containers/Viber/index.scss +1 -1
  227. package/app/v2Containers/WebPush/Create/components/BrandIconSection.test.js +264 -0
  228. package/app/v2Containers/WebPush/Create/components/MessageSection.js +78 -19
  229. package/app/v2Containers/WebPush/Create/components/MessageSection.test.js +82 -0
  230. package/app/v2Containers/WebPush/Create/components/__snapshots__/BrandIconSection.test.js.snap +187 -0
  231. package/app/v2Containers/WebPush/Create/components/__snapshots__/MessageSection.test.js.snap +25 -17
  232. package/app/v2Containers/WebPush/Create/hooks/useAiraTriggerPosition.js +80 -0
  233. package/app/v2Containers/WebPush/Create/hooks/useAiraTriggerPosition.test.js +210 -0
  234. package/app/v2Containers/WebPush/Create/hooks/useTagManagement.js +1 -5
  235. package/app/v2Containers/WebPush/Create/hooks/useTagManagement.test.js +0 -7
  236. package/app/v2Containers/WebPush/Create/index.js +36 -6
  237. package/app/v2Containers/WebPush/Create/index.scss +5 -0
  238. package/app/v2Containers/WebPush/Create/messages.js +8 -1
  239. package/app/v2Containers/WebPush/Create/preview/tests/NotificationContainer.test.js +269 -0
  240. package/app/v2Containers/WebPush/Create/utils/validation.js +31 -15
  241. package/app/v2Containers/WebPush/Create/utils/validation.test.js +72 -24
  242. package/app/v2Containers/Whatsapp/index.js +28 -53
  243. package/app/v2Containers/Whatsapp/tests/__snapshots__/index.test.js.snap +26939 -3982
  244. package/app/v2Containers/Whatsapp/tests/index.test.js +172 -0
  245. package/app/v2Containers/Zalo/index.js +5 -11
  246. package/package.json +2 -2
  247. package/version +9 -0
@@ -0,0 +1,304 @@
1
+ import React from 'react';
2
+ import { render, screen, fireEvent, waitFor } from '@testing-library/react';
3
+ import '@testing-library/jest-dom';
4
+ import { SmsFallback } from '../index';
5
+ import * as Api from '../../../services/api';
6
+ import { useLocalTemplateList } from '../useLocalTemplateList';
7
+
8
+ jest.mock('react-dom', () => ({
9
+ ...jest.requireActual('react-dom'),
10
+ createPortal: (node) => node,
11
+ }));
12
+
13
+ jest.mock('../../../services/api', () => ({
14
+ getTemplateDetails: jest.fn(),
15
+ }));
16
+
17
+ jest.mock('../useLocalTemplateList', () => ({
18
+ useLocalTemplateList: jest.fn(),
19
+ }));
20
+
21
+ jest.mock('../../../v2Containers/Templates/actions', () => ({
22
+ getLocalSmsTemplates: jest.fn(() => ({ type: 'LOCAL_SMS' })),
23
+ }));
24
+
25
+ const mockSmsFallbackLocalSelector = jest.fn((props) => (
26
+ <div data-testid="sms-fallback-selector-mock" data-hidden={String(props.hidden)} />
27
+ ));
28
+ jest.mock('../SmsFallbackLocalSelector', () => ({
29
+ __esModule: true,
30
+ default: (props) => mockSmsFallbackLocalSelector(props),
31
+ }));
32
+
33
+ const mockSmsTraiEdit = jest.fn(() => <div data-testid="sms-trai-edit-mock" />);
34
+ jest.mock('../../../v2Containers/SmsTrai/Edit', () => ({
35
+ __esModule: true,
36
+ default: (props) => mockSmsTraiEdit(props),
37
+ }));
38
+
39
+ const mockSmsWrapper = jest.fn(() => <div data-testid="sms-wrapper-mock" />);
40
+ jest.mock('../../../v2Containers/SmsWrapper', () => ({
41
+ __esModule: true,
42
+ default: (props) => mockSmsWrapper(props),
43
+ }));
44
+
45
+ jest.mock('../../../v2Containers/Templates/TemplatesActionBar', () => ({
46
+ __esModule: true,
47
+ default: ({ children }) => <div data-testid="templates-action-bar">{children}</div>,
48
+ }));
49
+
50
+ jest.mock('@capillarytech/cap-ui-library/CapRow', () => (props) => <div {...props} />);
51
+ jest.mock('@capillarytech/cap-ui-library/CapHeader', () => (props) => (
52
+ <div className={props.className}>
53
+ {props.prefix || null}
54
+ <div>{props.title}</div>
55
+ <div>{props.description}</div>
56
+ </div>
57
+ ));
58
+ jest.mock('@capillarytech/cap-ui-library/CapHeading', () => (props) => <div>{props.children}</div>);
59
+ jest.mock('@capillarytech/cap-ui-library/CapButton', () => (props) => (
60
+ <button type="button" {...props}>
61
+ {props.children}
62
+ </button>
63
+ ));
64
+ jest.mock('@capillarytech/cap-ui-library/CapIcon', () => (props) => <span {...props} />);
65
+ jest.mock('@capillarytech/cap-ui-library/CapDropdown', () => (props) => (
66
+ <div>
67
+ {props.children}
68
+ {props.overlay}
69
+ </div>
70
+ ));
71
+ jest.mock('@capillarytech/cap-ui-library/CapMenu', () => {
72
+ const Menu = ({ children }) => <div>{children}</div>;
73
+ Menu.Item = ({ children, onClick }) => (
74
+ <button type="button" onClick={onClick}>
75
+ {children}
76
+ </button>
77
+ );
78
+ return Menu;
79
+ });
80
+ jest.mock('@capillarytech/cap-ui-library/CapCustomCard', () => ({
81
+ CapCustomCardList: ({ cardList }) => (
82
+ <div data-testid="custom-card">
83
+ <div>{cardList?.[0]?.title}</div>
84
+ <div>{cardList?.[0]?.content}</div>
85
+ <div>{cardList?.[0]?.extra}</div>
86
+ </div>
87
+ ),
88
+ }));
89
+ jest.mock('@capillarytech/cap-ui-library/CapSlideBox', () => (props) => (
90
+ <div data-testid="cap-slidebox">
91
+ {props.header}
92
+ {props.content}
93
+ </div>
94
+ ));
95
+ jest.mock('@capillarytech/cap-ui-library', () => ({
96
+ CapLabel: ({ children }) => <span>{children}</span>,
97
+ CapIcons: {
98
+ backIcon: (props) => <span data-testid="cap-icons-back" {...props} />,
99
+ },
100
+ CapRadio: {
101
+ CapRadioGroup: ({ children }) => <div>{children}</div>,
102
+ Button: ({ children, value }) => <button type="button" data-value={value}>{children}</button>,
103
+ },
104
+ }));
105
+
106
+ const intl = { formatMessage: (m) => m.defaultMessage || m.id || '' };
107
+
108
+ /** Last jest mock invocation (Node <16 has no Array.prototype.at on mock.calls). */
109
+ const lastMockCall = (mockFn) => {
110
+ const { calls } = mockFn.mock;
111
+ return calls[calls.length - 1];
112
+ };
113
+
114
+ const buildList = () => ({
115
+ templates: [],
116
+ totalCount: 0,
117
+ loading: false,
118
+ page: 1,
119
+ search: '',
120
+ setSearch: jest.fn(),
121
+ loadMore: jest.fn(),
122
+ reset: jest.fn(),
123
+ canLoadMore: false,
124
+ });
125
+
126
+ const renderComp = (extraProps = {}) => {
127
+ const props = {
128
+ dispatch: jest.fn(),
129
+ value: null,
130
+ onChange: jest.fn(),
131
+ smsRegister: 'DLT',
132
+ selectedOfferDetails: {},
133
+ channelsToHide: ['email'],
134
+ sectionTitle: null,
135
+ intl,
136
+ showAsCard: true,
137
+ disableSelectTemplate: false,
138
+ eventContextTags: [],
139
+ isFullMode: false,
140
+ onRcsFallbackEditorStateChange: jest.fn(),
141
+ isRcsEditFlow: false,
142
+ ...extraProps,
143
+ };
144
+ return { ...render(<SmsFallback {...props} />), props };
145
+ };
146
+
147
+ describe('SmsFallback handlers', () => {
148
+ beforeEach(() => {
149
+ jest.clearAllMocks();
150
+ useLocalTemplateList.mockReturnValue(buildList());
151
+ Api.getTemplateDetails.mockResolvedValue({ response: {} });
152
+ });
153
+
154
+ it('opens selector and forwards embedded SMS save payload', async () => {
155
+ const { props } = renderComp();
156
+ fireEvent.click(screen.getByRole('button'));
157
+
158
+ const selectorProps = lastMockCall(mockSmsFallbackLocalSelector)[0];
159
+ selectorProps.getCreativesData({
160
+ channel: 'SMS',
161
+ messageBody: 'Body',
162
+ templateConfigs: { templateName: 'N', templateId: 9, template: 'Body' },
163
+ });
164
+
165
+ await waitFor(() => {
166
+ expect(props.onChange).toHaveBeenCalled();
167
+ });
168
+ });
169
+
170
+ it('ignores embedded non-SMS payload in getCreativesData', () => {
171
+ const { props } = renderComp();
172
+ fireEvent.click(screen.getByRole('button'));
173
+ const selectorProps = lastMockCall(mockSmsFallbackLocalSelector)[0];
174
+ selectorProps.getCreativesData({ channel: 'RCS' });
175
+ expect(props.onChange).not.toHaveBeenCalled();
176
+ });
177
+
178
+ it('selects template without id and opens edit with pending data', async () => {
179
+ renderComp();
180
+ fireEvent.click(screen.getByRole('button'));
181
+ const selectorProps = lastMockCall(mockSmsFallbackLocalSelector)[0];
182
+ selectorProps.onSelectTemplate({
183
+ name: 'No id template',
184
+ versions: { base: { 'sms-editor': 'raw-content' } },
185
+ });
186
+
187
+ await waitFor(() => {
188
+ expect(mockSmsTraiEdit).toHaveBeenCalled();
189
+ });
190
+ });
191
+
192
+ it('fetches template details for selected template id and opens edit', async () => {
193
+ renderComp();
194
+ fireEvent.click(screen.getByRole('button'));
195
+ const selectorProps = lastMockCall(mockSmsFallbackLocalSelector)[0];
196
+ await selectorProps.onSelectTemplate({
197
+ _id: 'temp-1',
198
+ name: 'Server template',
199
+ });
200
+
201
+ expect(Api.getTemplateDetails).toHaveBeenCalledWith({ id: 'temp-1', channel: 'Sms' });
202
+ await waitFor(() => {
203
+ expect(mockSmsTraiEdit).toHaveBeenCalled();
204
+ });
205
+ });
206
+
207
+ it('falls back to selected template when details fetch fails', async () => {
208
+ Api.getTemplateDetails.mockRejectedValueOnce(new Error('x'));
209
+ renderComp();
210
+ fireEvent.click(screen.getByRole('button'));
211
+ const selectorProps = lastMockCall(mockSmsFallbackLocalSelector)[0];
212
+ await selectorProps.onSelectTemplate({
213
+ _id: 'temp-2',
214
+ name: 'Fallback template',
215
+ versions: { base: { 'sms-editor': 'body' } },
216
+ });
217
+
218
+ await waitFor(() => {
219
+ expect(mockSmsTraiEdit).toHaveBeenCalled();
220
+ });
221
+ });
222
+
223
+ it('saves edit form from inline editor without closing when skipCloseOnSave true', async () => {
224
+ const onChange = jest.fn();
225
+ renderComp({
226
+ onChange,
227
+ value: {
228
+ templateName: 'Existing',
229
+ content: 'Old',
230
+ templateContent: 'Old',
231
+ unicodeValidity: true,
232
+ },
233
+ showAsCard: false,
234
+ });
235
+
236
+ const editProps = lastMockCall(mockSmsTraiEdit)[0];
237
+ editProps.getFormSubscriptionData({
238
+ versions: { base: { 'sms-editor': 'New body', 'unicode-validity': false } },
239
+ });
240
+
241
+ await waitFor(() => {
242
+ expect(onChange).toHaveBeenCalledWith(
243
+ expect.objectContaining({
244
+ content: 'New body',
245
+ templateContent: 'New body',
246
+ unicodeValidity: false,
247
+ }),
248
+ );
249
+ });
250
+ });
251
+
252
+ it('renders selector when internal view is set through open action', () => {
253
+ renderComp({ value: null });
254
+ fireEvent.click(screen.getByRole('button'));
255
+ const selectorProps = lastMockCall(mockSmsFallbackLocalSelector)[0];
256
+ expect(selectorProps.hidden).toBe(false);
257
+ expect(selectorProps.fetchDetailsLoading).toBe(false);
258
+ });
259
+
260
+ it('passes create view hook callbacks to SmsWrapper', async () => {
261
+ const onChange = jest.fn();
262
+ /* Non-DLT so SmsWrapper mounts SmsCreate, not SmsTraiCreate (DLT + library forces Trai flow). */
263
+ renderComp({ onChange, smsRegister: 'REGISTERED' });
264
+ fireEvent.click(screen.getByRole('button'));
265
+ const selectorProps = lastMockCall(mockSmsFallbackLocalSelector)[0];
266
+ selectorProps.filterContent.props.onCtaClick();
267
+
268
+ await waitFor(() => {
269
+ expect(screen.getByTestId('cap-slidebox')).toBeInTheDocument();
270
+ expect(mockSmsWrapper).toHaveBeenCalled();
271
+ });
272
+
273
+ const wrapperProps = lastMockCall(mockSmsWrapper)[0];
274
+ wrapperProps.getFormSubscriptionData({
275
+ validity: true,
276
+ value: { base: { 'sms-editor': 'created text', 'template-name': 'created', 'unicode-validity': true } },
277
+ });
278
+
279
+ await waitFor(() => {
280
+ expect(onChange).toHaveBeenCalledWith(
281
+ expect.objectContaining({
282
+ templateName: 'created',
283
+ content: 'created text',
284
+ templateContent: 'created text',
285
+ }),
286
+ );
287
+ });
288
+ });
289
+
290
+ it('does not save create payload when validity is false', async () => {
291
+ const onChange = jest.fn();
292
+ renderComp({ onChange, smsRegister: 'REGISTERED' });
293
+ fireEvent.click(screen.getByRole('button'));
294
+ const selectorProps = lastMockCall(mockSmsFallbackLocalSelector)[0];
295
+ selectorProps.filterContent.props.onCtaClick();
296
+ await waitFor(() => {
297
+ expect(mockSmsWrapper).toHaveBeenCalled();
298
+ });
299
+ const wrapperProps = lastMockCall(mockSmsWrapper)[0];
300
+ wrapperProps.getFormSubscriptionData({ validity: false, value: { base: { 'sms-editor': 'x' } } });
301
+ expect(onChange).not.toHaveBeenCalled();
302
+ });
303
+ });
304
+
@@ -0,0 +1,197 @@
1
+ /**
2
+ * Additional SmsFallback UI coverage (props, layout branches, interaction flows) for Sonar new-code metrics.
3
+ */
4
+ import React from 'react';
5
+ import { render, screen, fireEvent, waitFor } from '@testing-library/react';
6
+ import '@testing-library/jest-dom';
7
+ import { Provider } from 'react-redux';
8
+ import { createStore } from 'redux';
9
+ import { IntlProvider } from 'react-intl';
10
+ import SmsFallback from '../index';
11
+ import smsFallbackMessages from '../messages';
12
+
13
+ jest.mock('react-redux', () => ({
14
+ ...jest.requireActual('react-redux'),
15
+ useDispatch: () => jest.fn(),
16
+ }));
17
+
18
+ jest.mock('../../../services/api', () => ({
19
+ getTemplateDetails: jest.fn().mockResolvedValue({ response: {} }),
20
+ }));
21
+
22
+ jest.mock('../../../v2Containers/Templates/utils/smsTemplatesListApi', () => ({
23
+ fetchSmsTemplatesListPage: jest.fn().mockResolvedValue({ templates: [], totalCount: 0 }),
24
+ }));
25
+
26
+ jest.mock('react-dom', () => ({
27
+ ...jest.requireActual('react-dom'),
28
+ createPortal: (node) => node,
29
+ }));
30
+
31
+ // Render CapDropdown overlay inline — rc-trigger appends to the DOM directly and
32
+ // bypasses the createPortal mock, so the overlay would never appear in JSDOM otherwise.
33
+ jest.mock('@capillarytech/cap-ui-library/CapDropdown', () => ({
34
+ __esModule: true,
35
+ default: ({ overlay, children }) => (
36
+ <span>
37
+ {children}
38
+ {overlay}
39
+ </span>
40
+ ),
41
+ }));
42
+
43
+ // Render localTemplatesFilterContent (the TemplatesActionBar with "Create new") so that
44
+ // tests can interact with the CTA button. It arrives nested inside localTemplatesConfig,
45
+ // not as a top-level prop.
46
+ jest.mock('../../../v2Containers/CreativesContainer', () => ({
47
+ __esModule: true,
48
+ default: ({ localTemplatesConfig }) => (
49
+ <div data-testid="sms-fallback-creatives-container">
50
+ {localTemplatesConfig?.localTemplatesFilterContent}
51
+ </div>
52
+ ),
53
+ }));
54
+
55
+ jest.mock('../../../v2Containers/SmsTrai/Edit', () => ({
56
+ __esModule: true,
57
+ default: () => <div data-testid="sms-trai-edit-mock">SmsTraiEdit</div>,
58
+ }));
59
+
60
+ jest.mock('../../../v2Containers/SmsWrapper', () => ({
61
+ __esModule: true,
62
+ default: () => <div data-testid="sms-wrapper-mock">SmsWrapper</div>,
63
+ }));
64
+
65
+ const intlMessages = Object.keys(smsFallbackMessages).reduce((acc, key) => {
66
+ const m = smsFallbackMessages[key];
67
+ acc[m.id] = m.defaultMessage;
68
+ return acc;
69
+ }, {});
70
+
71
+ const renderSmsFallback = (props = {}) => {
72
+ const store = createStore((state = {}) => state);
73
+ return render(
74
+ <Provider store={store}>
75
+ <IntlProvider locale="en" messages={intlMessages}>
76
+ <SmsFallback
77
+ onChange={jest.fn()}
78
+ smsRegister={props.smsRegister ?? 'DLT'}
79
+ isFullMode={props.isFullMode ?? true}
80
+ {...props}
81
+ />
82
+ </IntlProvider>
83
+ </Provider>
84
+ );
85
+ };
86
+
87
+ describe('SmsFallback UI branches', () => {
88
+ it('renders inline editor when value exists and showAsCard is false', () => {
89
+ const { container } = renderSmsFallback({
90
+ value: {
91
+ templateName: 'T',
92
+ content: 'C',
93
+ templateContent: 'C',
94
+ },
95
+ showAsCard: false,
96
+ });
97
+ expect(container.querySelector('.sms-fallback-inline-editor')).toBeTruthy();
98
+ expect(screen.getByTestId('sms-trai-edit-mock')).toBeInTheDocument();
99
+ });
100
+
101
+ it('disables Select template when disableSelectTemplate is true', () => {
102
+ renderSmsFallback({ value: null, disableSelectTemplate: true });
103
+ expect(
104
+ screen.getByRole('button', { name: smsFallbackMessages.selectTemplate.defaultMessage })
105
+ ).toBeDisabled();
106
+ });
107
+
108
+ it('uses custom sectionTitle when provided', () => {
109
+ renderSmsFallback({ value: null, sectionTitle: 'Custom SMS section' });
110
+ expect(screen.getByText('Custom SMS section')).toBeInTheDocument();
111
+ });
112
+
113
+ it('uses sectionTitleWithoutOptional when value is set and sectionTitle omitted', () => {
114
+ renderSmsFallback({
115
+ value: { templateName: 'N', content: 'x' },
116
+ showAsCard: true,
117
+ });
118
+ expect(
119
+ screen.getByText(smsFallbackMessages.sectionTitleWithoutOptional.defaultMessage)
120
+ ).toBeInTheDocument();
121
+ });
122
+ });
123
+
124
+ describe('SmsFallback interaction flows', () => {
125
+ it('open → selector visible → close → selector gone', async () => {
126
+ renderSmsFallback({ value: null });
127
+
128
+ // Portal not rendered before opening
129
+ expect(screen.queryByTestId('sms-fallback-creatives-container')).not.toBeInTheDocument();
130
+
131
+ fireEvent.click(
132
+ screen.getByRole('button', { name: smsFallbackMessages.selectTemplate.defaultMessage }),
133
+ );
134
+
135
+ await waitFor(() => {
136
+ expect(screen.getByTestId('sms-fallback-creatives-container')).toBeInTheDocument();
137
+ });
138
+ });
139
+
140
+ it('open → click create new → SmsWrapper rendered inside slidebox', async () => {
141
+ // Non-DLT so the CTA label is "Create new" (not "Upload new") and SmsWrapper is used
142
+ renderSmsFallback({ value: null, smsRegister: 'REGISTERED', isFullMode: false });
143
+
144
+ fireEvent.click(
145
+ screen.getByRole('button', { name: smsFallbackMessages.selectTemplate.defaultMessage }),
146
+ );
147
+
148
+ await waitFor(() => {
149
+ expect(screen.getByTestId('sms-fallback-creatives-container')).toBeInTheDocument();
150
+ });
151
+
152
+ fireEvent.click(
153
+ screen.getByRole('button', { name: smsFallbackMessages.createNew.defaultMessage }),
154
+ );
155
+
156
+ await waitFor(() => {
157
+ expect(screen.getByTestId('sms-wrapper-mock')).toBeInTheDocument();
158
+ });
159
+
160
+ // Selector should still be mounted (hidden behind slidebox) — not unmounted
161
+ expect(screen.getByTestId('sms-fallback-creatives-container')).toBeInTheDocument();
162
+ });
163
+
164
+ it('value present → card shown → edit option → SmsTraiEdit rendered', async () => {
165
+ const { container } = renderSmsFallback({
166
+ value: { templateName: 'My SMS', content: 'Hello', templateContent: 'Hello' },
167
+ showAsCard: true,
168
+ });
169
+
170
+ // Card should display the template name
171
+ expect(screen.getByText('My SMS')).toBeInTheDocument();
172
+
173
+ // The "more" trigger is a CapIcon rendered as an <i> (no accessible role) — query by class.
174
+ fireEvent.click(container.querySelector('.cap-icon-v2-more'));
175
+ // CapMenu.Item renders as <li> — find by visible text.
176
+ fireEvent.click(screen.getByText(smsFallbackMessages.editTemplate.defaultMessage));
177
+
178
+ await waitFor(() => {
179
+ expect(screen.getByTestId('sms-trai-edit-mock')).toBeInTheDocument();
180
+ });
181
+ });
182
+
183
+ it('value present → remove clears the value', async () => {
184
+ const onChange = jest.fn();
185
+ const { container } = renderSmsFallback({
186
+ onChange,
187
+ value: { templateName: 'To Remove', content: 'x', templateContent: 'x' },
188
+ showAsCard: true,
189
+ });
190
+
191
+ fireEvent.click(container.querySelector('.cap-icon-v2-more'));
192
+ // CapMenu.Item renders as <li> — find by visible text.
193
+ fireEvent.click(screen.getByText(smsFallbackMessages.removeTemplate.defaultMessage));
194
+
195
+ expect(onChange).toHaveBeenCalledWith(null);
196
+ });
197
+ });