@capillarytech/creatives-library 8.0.257 → 8.0.258

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 (278) hide show
  1. package/assets/Android.png +0 -0
  2. package/assets/iOS.png +0 -0
  3. package/constants/unified.js +2 -2
  4. package/initialReducer.js +0 -2
  5. package/package.json +1 -1
  6. package/services/api.js +5 -10
  7. package/services/tests/api.test.js +0 -34
  8. package/translations/en.json +4 -3
  9. package/utils/common.js +6 -5
  10. package/utils/commonUtils.js +5 -28
  11. package/utils/imageUrlUpload.js +141 -0
  12. package/utils/tests/commonUtil.test.js +0 -224
  13. package/utils/tests/transformerUtils.test.js +297 -0
  14. package/utils/transformTemplateConfig.js +10 -0
  15. package/utils/transformerUtils.js +40 -0
  16. package/v2Components/CapDeviceContent/index.js +56 -61
  17. package/v2Components/CapImageUpload/constants.js +2 -0
  18. package/v2Components/CapImageUpload/index.js +65 -16
  19. package/v2Components/CapImageUpload/index.scss +4 -1
  20. package/v2Components/CapImageUpload/messages.js +5 -1
  21. package/v2Components/CapImageUrlUpload/constants.js +26 -0
  22. package/v2Components/CapImageUrlUpload/index.js +365 -0
  23. package/v2Components/CapImageUrlUpload/index.scss +35 -0
  24. package/v2Components/CapImageUrlUpload/messages.js +47 -0
  25. package/v2Components/CapTagList/index.js +1 -6
  26. package/v2Components/CapTagListWithInput/index.js +1 -5
  27. package/v2Components/CapTagListWithInput/messages.js +1 -1
  28. package/v2Components/CapWhatsappCTA/tests/index.test.js +0 -5
  29. package/v2Components/ErrorInfoNote/index.js +72 -457
  30. package/v2Components/ErrorInfoNote/messages.js +6 -36
  31. package/v2Components/ErrorInfoNote/style.scss +6 -282
  32. package/v2Components/FormBuilder/tests/index.test.js +4 -13
  33. package/v2Components/HtmlEditor/HTMLEditor.js +94 -547
  34. package/v2Components/HtmlEditor/__tests__/HTMLEditor.test.js +133 -1441
  35. package/v2Components/HtmlEditor/__tests__/index.lazy.test.js +16 -27
  36. package/v2Components/HtmlEditor/_htmlEditor.scss +45 -108
  37. package/v2Components/HtmlEditor/_index.lazy.scss +1 -0
  38. package/v2Components/HtmlEditor/components/CodeEditorPane/_codeEditorPane.scss +102 -23
  39. package/v2Components/HtmlEditor/components/CodeEditorPane/index.js +140 -148
  40. package/v2Components/HtmlEditor/components/DeviceToggle/_deviceToggle.scss +1 -2
  41. package/v2Components/HtmlEditor/components/DeviceToggle/index.js +3 -3
  42. package/v2Components/HtmlEditor/components/EditorToolbar/_editorToolbar.scss +0 -9
  43. package/v2Components/HtmlEditor/components/EditorToolbar/index.js +4 -4
  44. package/v2Components/HtmlEditor/components/FullscreenModal/_fullscreenModal.scss +0 -22
  45. package/v2Components/HtmlEditor/components/InAppPreviewPane/DeviceFrame.js +7 -4
  46. package/v2Components/HtmlEditor/components/InAppPreviewPane/__tests__/DeviceFrame.test.js +45 -35
  47. package/v2Components/HtmlEditor/components/InAppPreviewPane/_inAppPreviewPane.scss +3 -1
  48. package/v2Components/HtmlEditor/components/InAppPreviewPane/constants.js +33 -33
  49. package/v2Components/HtmlEditor/components/InAppPreviewPane/index.js +6 -7
  50. package/v2Components/HtmlEditor/components/PreviewPane/_previewPane.scss +6 -3
  51. package/v2Components/HtmlEditor/components/PreviewPane/index.js +43 -22
  52. package/v2Components/HtmlEditor/components/SplitContainer/_splitContainer.scss +1 -1
  53. package/v2Components/HtmlEditor/components/ValidationErrorDisplay/__tests__/index.test.js +152 -0
  54. package/v2Components/HtmlEditor/components/ValidationErrorDisplay/_validationErrorDisplay.scss +0 -1
  55. package/v2Components/HtmlEditor/components/ValidationErrorDisplay/index.js +31 -49
  56. package/v2Components/HtmlEditor/components/ValidationPanel/_validationPanel.scss +34 -50
  57. package/v2Components/HtmlEditor/components/ValidationPanel/index.js +41 -70
  58. package/v2Components/HtmlEditor/constants.js +20 -42
  59. package/v2Components/HtmlEditor/hooks/__tests__/useInAppContent.test.js +16 -373
  60. package/v2Components/HtmlEditor/hooks/__tests__/useValidation.test.js +0 -103
  61. package/v2Components/HtmlEditor/hooks/useEditorContent.js +2 -5
  62. package/v2Components/HtmlEditor/hooks/useInAppContent.js +146 -88
  63. package/v2Components/HtmlEditor/hooks/useValidation.js +53 -189
  64. package/v2Components/HtmlEditor/index.js +1 -1
  65. package/v2Components/HtmlEditor/messages.js +94 -92
  66. package/v2Components/HtmlEditor/utils/__tests__/htmlValidator.enhanced.test.js +45 -94
  67. package/v2Components/HtmlEditor/utils/__tests__/validationAdapter.test.js +0 -134
  68. package/v2Components/HtmlEditor/utils/contentSanitizer.js +41 -40
  69. package/v2Components/HtmlEditor/utils/htmlValidator.js +72 -71
  70. package/v2Components/HtmlEditor/utils/liquidTemplateSupport.js +102 -134
  71. package/v2Components/HtmlEditor/utils/properSyntaxHighlighting.js +25 -23
  72. package/v2Components/HtmlEditor/utils/validationAdapter.js +41 -66
  73. package/v2Components/MobilePushPreviewV2/index.js +7 -32
  74. package/v2Components/TemplatePreview/_templatePreview.scss +24 -55
  75. package/v2Components/TemplatePreview/index.js +32 -47
  76. package/v2Components/TemplatePreview/messages.js +0 -4
  77. package/v2Components/TestAndPreviewSlidebox/_testAndPreviewSlidebox.scss +0 -1
  78. package/v2Containers/App/constants.js +5 -0
  79. package/v2Containers/BeeEditor/index.js +90 -172
  80. package/v2Containers/Cap/tests/__snapshots__/index.test.js.snap +4 -3
  81. package/v2Containers/CreativesContainer/SlideBoxContent.js +107 -129
  82. package/v2Containers/CreativesContainer/SlideBoxFooter.js +13 -163
  83. package/v2Containers/CreativesContainer/SlideBoxHeader.js +2 -2
  84. package/v2Containers/CreativesContainer/constants.js +3 -1
  85. package/v2Containers/CreativesContainer/index.js +214 -239
  86. package/v2Containers/CreativesContainer/messages.js +4 -8
  87. package/v2Containers/CreativesContainer/tests/SlideBoxContent.test.js +210 -0
  88. package/v2Containers/CreativesContainer/tests/SlideBoxFooter.test.js +2 -11
  89. package/v2Containers/CreativesContainer/tests/__snapshots__/SlideBoxContent.test.js.snap +354 -38
  90. package/v2Containers/CreativesContainer/tests/__snapshots__/index.test.js.snap +0 -106
  91. package/v2Containers/Email/actions.js +0 -7
  92. package/v2Containers/Email/constants.js +1 -5
  93. package/v2Containers/Email/index.js +29 -234
  94. package/v2Containers/Email/messages.js +0 -32
  95. package/v2Containers/Email/reducer.js +1 -12
  96. package/v2Containers/Email/sagas.js +7 -61
  97. package/v2Containers/Email/tests/__snapshots__/reducer.test.js.snap +0 -2
  98. package/v2Containers/Email/tests/reducer.test.js +0 -46
  99. package/v2Containers/Email/tests/sagas.test.js +29 -320
  100. package/v2Containers/EmailWrapper/components/EmailWrapperView.js +21 -211
  101. package/v2Containers/EmailWrapper/components/HTMLEditorTesting.js +74 -40
  102. package/v2Containers/EmailWrapper/components/__tests__/HTMLEditorTesting.test.js +67 -2
  103. package/v2Containers/EmailWrapper/constants.js +0 -2
  104. package/v2Containers/EmailWrapper/hooks/useEmailWrapper.js +77 -629
  105. package/v2Containers/EmailWrapper/index.js +23 -103
  106. package/v2Containers/EmailWrapper/messages.js +1 -65
  107. package/v2Containers/EmailWrapper/tests/EmailWrapperView.test.js +214 -0
  108. package/v2Containers/EmailWrapper/tests/useEmailWrapper.test.js +77 -594
  109. package/v2Containers/InApp/actions.js +0 -7
  110. package/v2Containers/InApp/constants.js +4 -20
  111. package/v2Containers/InApp/index.js +359 -802
  112. package/v2Containers/InApp/index.scss +3 -4
  113. package/v2Containers/InApp/messages.js +3 -7
  114. package/v2Containers/InApp/reducer.js +3 -21
  115. package/v2Containers/InApp/sagas.js +9 -29
  116. package/v2Containers/InApp/selectors.js +5 -25
  117. package/v2Containers/InApp/tests/index.test.js +50 -154
  118. package/v2Containers/InApp/tests/reducer.test.js +0 -34
  119. package/v2Containers/InApp/tests/sagas.test.js +9 -61
  120. package/v2Containers/Line/Container/ImageCarousel/tests/__snapshots__/content.test.js.snap +36 -15
  121. package/v2Containers/Line/Container/ImageCarousel/tests/__snapshots__/index.test.js.snap +8 -8
  122. package/v2Containers/Line/Container/Wrapper/tests/__snapshots__/index.test.js.snap +100 -77
  123. package/v2Containers/Line/Container/tests/__snapshots__/index.test.js.snap +72 -63
  124. package/v2Containers/Rcs/tests/__snapshots__/index.test.js.snap +250 -190
  125. package/v2Containers/SmsTrai/Create/tests/__snapshots__/index.test.js.snap +16 -12
  126. package/v2Containers/SmsTrai/Edit/tests/__snapshots__/index.test.js.snap +48 -40
  127. package/v2Containers/TagList/index.js +19 -62
  128. package/v2Containers/Templates/ChannelTypeIllustration.js +13 -1
  129. package/v2Containers/Templates/_templates.scss +202 -56
  130. package/v2Containers/Templates/actions.js +2 -1
  131. package/v2Containers/Templates/constants.js +1 -0
  132. package/v2Containers/Templates/index.js +278 -123
  133. package/v2Containers/Templates/messages.js +24 -4
  134. package/v2Containers/Templates/reducer.js +2 -0
  135. package/v2Containers/Templates/tests/index.test.js +10 -0
  136. package/v2Containers/TemplatesV2/index.js +15 -7
  137. package/v2Containers/TemplatesV2/messages.js +4 -0
  138. package/v2Containers/WebPush/Create/components/BrandIconSection.js +108 -0
  139. package/v2Containers/WebPush/Create/components/ButtonForm.js +172 -0
  140. package/v2Containers/WebPush/Create/components/ButtonItem.js +101 -0
  141. package/v2Containers/WebPush/Create/components/ButtonList.js +145 -0
  142. package/v2Containers/WebPush/Create/components/ButtonsLinksSection.js +164 -0
  143. package/v2Containers/WebPush/Create/components/ButtonsLinksSection.test.js +463 -0
  144. package/v2Containers/WebPush/Create/components/FormActions.js +54 -0
  145. package/v2Containers/WebPush/Create/components/FormActions.test.js +163 -0
  146. package/v2Containers/WebPush/Create/components/MediaSection.js +142 -0
  147. package/v2Containers/WebPush/Create/components/MediaSection.test.js +341 -0
  148. package/v2Containers/WebPush/Create/components/MessageSection.js +103 -0
  149. package/v2Containers/WebPush/Create/components/MessageSection.test.js +268 -0
  150. package/v2Containers/WebPush/Create/components/NotificationTitleSection.js +87 -0
  151. package/v2Containers/WebPush/Create/components/NotificationTitleSection.test.js +210 -0
  152. package/v2Containers/WebPush/Create/components/TemplateNameSection.js +54 -0
  153. package/v2Containers/WebPush/Create/components/TemplateNameSection.test.js +143 -0
  154. package/v2Containers/WebPush/Create/components/__snapshots__/ButtonsLinksSection.test.js.snap +86 -0
  155. package/v2Containers/WebPush/Create/components/__snapshots__/FormActions.test.js.snap +16 -0
  156. package/v2Containers/WebPush/Create/components/__snapshots__/MediaSection.test.js.snap +41 -0
  157. package/v2Containers/WebPush/Create/components/__snapshots__/MessageSection.test.js.snap +54 -0
  158. package/v2Containers/WebPush/Create/components/__snapshots__/NotificationTitleSection.test.js.snap +37 -0
  159. package/v2Containers/WebPush/Create/components/__snapshots__/TemplateNameSection.test.js.snap +21 -0
  160. package/v2Containers/WebPush/Create/components/_buttons.scss +246 -0
  161. package/v2Containers/WebPush/Create/components/tests/ButtonForm.test.js +554 -0
  162. package/v2Containers/WebPush/Create/components/tests/ButtonItem.test.js +607 -0
  163. package/v2Containers/WebPush/Create/components/tests/ButtonList.test.js +633 -0
  164. package/v2Containers/WebPush/Create/components/tests/__snapshots__/ButtonForm.test.js.snap +666 -0
  165. package/v2Containers/WebPush/Create/components/tests/__snapshots__/ButtonItem.test.js.snap +74 -0
  166. package/v2Containers/WebPush/Create/components/tests/__snapshots__/ButtonList.test.js.snap +78 -0
  167. package/v2Containers/WebPush/Create/hooks/useButtonManagement.js +138 -0
  168. package/v2Containers/WebPush/Create/hooks/useButtonManagement.test.js +406 -0
  169. package/v2Containers/WebPush/Create/hooks/useCharacterCount.js +30 -0
  170. package/v2Containers/WebPush/Create/hooks/useCharacterCount.test.js +151 -0
  171. package/v2Containers/WebPush/Create/hooks/useImageUpload.js +104 -0
  172. package/v2Containers/WebPush/Create/hooks/useImageUpload.test.js +538 -0
  173. package/v2Containers/WebPush/Create/hooks/useTagManagement.js +122 -0
  174. package/v2Containers/WebPush/Create/hooks/useTagManagement.test.js +633 -0
  175. package/v2Containers/WebPush/Create/index.js +1148 -0
  176. package/v2Containers/WebPush/Create/index.scss +134 -0
  177. package/v2Containers/WebPush/Create/messages.js +203 -0
  178. package/v2Containers/WebPush/Create/preview/DevicePreviewContent.js +228 -0
  179. package/v2Containers/WebPush/Create/preview/NotificationContainer.js +294 -0
  180. package/v2Containers/WebPush/Create/preview/PreviewContent.js +90 -0
  181. package/v2Containers/WebPush/Create/preview/PreviewControls.js +305 -0
  182. package/v2Containers/WebPush/Create/preview/PreviewDisclaimer.js +23 -0
  183. package/v2Containers/WebPush/Create/preview/WebPushPreview.js +155 -0
  184. package/v2Containers/WebPush/Create/preview/assets/Light.svg +53 -0
  185. package/v2Containers/WebPush/Create/preview/assets/Top.svg +5 -0
  186. package/v2Containers/WebPush/Create/preview/assets/android-arrow-down.svg +9 -0
  187. package/v2Containers/WebPush/Create/preview/assets/android-arrow-up.svg +9 -0
  188. package/v2Containers/WebPush/Create/preview/assets/chrome-icon.png +0 -0
  189. package/v2Containers/WebPush/Create/preview/assets/edge-icon.png +0 -0
  190. package/v2Containers/WebPush/Create/preview/assets/firefox-icon.svg +106 -0
  191. package/v2Containers/WebPush/Create/preview/assets/iOS.svg +26 -0
  192. package/v2Containers/WebPush/Create/preview/assets/macos-arrow-down-icon.svg +9 -0
  193. package/v2Containers/WebPush/Create/preview/assets/macos-triple-dot-icon.svg +9 -0
  194. package/v2Containers/WebPush/Create/preview/assets/opera-icon.svg +18 -0
  195. package/v2Containers/WebPush/Create/preview/assets/safari-icon.svg +29 -0
  196. package/v2Containers/WebPush/Create/preview/assets/windows-close-icon.svg +9 -0
  197. package/v2Containers/WebPush/Create/preview/assets/windows-triple-dot-icon.svg +9 -0
  198. package/v2Containers/WebPush/Create/preview/components/AndroidMobileChromeHeader.js +47 -0
  199. package/v2Containers/WebPush/Create/preview/components/AndroidMobileExpanded.js +141 -0
  200. package/v2Containers/WebPush/Create/preview/components/IOSHeader.js +45 -0
  201. package/v2Containers/WebPush/Create/preview/components/NotificationExpandedContent.js +68 -0
  202. package/v2Containers/WebPush/Create/preview/components/NotificationHeader.js +61 -0
  203. package/v2Containers/WebPush/Create/preview/components/WindowsChromeExpanded.js +99 -0
  204. package/v2Containers/WebPush/Create/preview/components/tests/AndroidMobileExpanded.test.js +733 -0
  205. package/v2Containers/WebPush/Create/preview/components/tests/WindowsChromeExpanded.test.js +571 -0
  206. package/v2Containers/WebPush/Create/preview/components/tests/__snapshots__/AndroidMobileExpanded.test.js.snap +81 -0
  207. package/v2Containers/WebPush/Create/preview/components/tests/__snapshots__/WindowsChromeExpanded.test.js.snap +81 -0
  208. package/v2Containers/WebPush/Create/preview/config/notificationMappings.js +50 -0
  209. package/v2Containers/WebPush/Create/preview/constants.js +637 -0
  210. package/v2Containers/WebPush/Create/preview/notification-container.scss +79 -0
  211. package/v2Containers/WebPush/Create/preview/preview.scss +351 -0
  212. package/v2Containers/WebPush/Create/preview/styles/_android-mobile-chrome.scss +370 -0
  213. package/v2Containers/WebPush/Create/preview/styles/_android-mobile-edge.scss +12 -0
  214. package/v2Containers/WebPush/Create/preview/styles/_android-mobile-firefox.scss +12 -0
  215. package/v2Containers/WebPush/Create/preview/styles/_android-mobile-opera.scss +12 -0
  216. package/v2Containers/WebPush/Create/preview/styles/_android-tablet-chrome.scss +47 -0
  217. package/v2Containers/WebPush/Create/preview/styles/_android-tablet-edge.scss +11 -0
  218. package/v2Containers/WebPush/Create/preview/styles/_android-tablet-firefox.scss +11 -0
  219. package/v2Containers/WebPush/Create/preview/styles/_android-tablet-opera.scss +11 -0
  220. package/v2Containers/WebPush/Create/preview/styles/_base.scss +207 -0
  221. package/v2Containers/WebPush/Create/preview/styles/_ios.scss +153 -0
  222. package/v2Containers/WebPush/Create/preview/styles/_ipados.scss +107 -0
  223. package/v2Containers/WebPush/Create/preview/styles/_macos-chrome.scss +101 -0
  224. package/v2Containers/WebPush/Create/preview/styles/_windows-chrome.scss +229 -0
  225. package/v2Containers/WebPush/Create/preview/tests/DevicePreviewContent.test.js +909 -0
  226. package/v2Containers/WebPush/Create/preview/tests/NotificationContainer.test.js +1081 -0
  227. package/v2Containers/WebPush/Create/preview/tests/PreviewControls.test.js +723 -0
  228. package/v2Containers/WebPush/Create/preview/tests/WebPushPreview.test.js +1327 -0
  229. package/v2Containers/WebPush/Create/preview/tests/__snapshots__/DevicePreviewContent.test.js.snap +131 -0
  230. package/v2Containers/WebPush/Create/preview/tests/__snapshots__/NotificationContainer.test.js.snap +112 -0
  231. package/v2Containers/WebPush/Create/preview/tests/__snapshots__/PreviewControls.test.js.snap +144 -0
  232. package/v2Containers/WebPush/Create/preview/tests/__snapshots__/WebPushPreview.test.js.snap +129 -0
  233. package/v2Containers/WebPush/Create/utils/payloadBuilder.js +96 -0
  234. package/v2Containers/WebPush/Create/utils/payloadBuilder.test.js +396 -0
  235. package/v2Containers/WebPush/Create/utils/previewUtils.js +89 -0
  236. package/v2Containers/WebPush/Create/utils/urlValidation.js +115 -0
  237. package/v2Containers/WebPush/Create/utils/urlValidation.test.js +449 -0
  238. package/v2Containers/WebPush/Create/utils/validation.js +75 -0
  239. package/v2Containers/WebPush/Create/utils/validation.test.js +283 -0
  240. package/v2Containers/WebPush/actions.js +60 -0
  241. package/v2Containers/WebPush/constants.js +132 -0
  242. package/v2Containers/WebPush/index.js +2 -0
  243. package/v2Containers/WebPush/reducer.js +104 -0
  244. package/v2Containers/WebPush/sagas.js +119 -0
  245. package/v2Containers/WebPush/selectors.js +65 -0
  246. package/v2Containers/WebPush/tests/reducer.test.js +863 -0
  247. package/v2Containers/WebPush/tests/sagas.test.js +566 -0
  248. package/v2Containers/WebPush/tests/selectors.test.js +960 -0
  249. package/v2Containers/Whatsapp/tests/__snapshots__/index.test.js.snap +1272 -768
  250. package/v2Components/ErrorInfoNote/constants.js +0 -1
  251. package/v2Components/HtmlEditor/__tests__/HTMLEditor.apiErrors.test.js +0 -874
  252. package/v2Components/HtmlEditor/components/ValidationPanel/constants.js +0 -6
  253. package/v2Components/HtmlEditor/components/ValidationTabs/_validationTabs.scss +0 -255
  254. package/v2Components/HtmlEditor/components/ValidationTabs/index.js +0 -364
  255. package/v2Components/HtmlEditor/components/ValidationTabs/messages.js +0 -51
  256. package/v2Components/HtmlEditor/utils/validationConstants.js +0 -40
  257. package/v2Containers/BeePopupEditor/_beePopupEditor.scss +0 -14
  258. package/v2Containers/BeePopupEditor/constants.js +0 -10
  259. package/v2Containers/BeePopupEditor/index.js +0 -194
  260. package/v2Containers/BeePopupEditor/tests/index.test.js +0 -627
  261. package/v2Containers/EmailWrapper/components/EmailHTMLEditor.js +0 -1285
  262. package/v2Containers/EmailWrapper/components/__tests__/EmailHTMLEditor.test.js +0 -1880
  263. package/v2Containers/EmailWrapper/components/__tests__/EmailWrapperView.test.js +0 -520
  264. package/v2Containers/EmailWrapper/tests/useEmailWrapper.edgeCases.test.js +0 -643
  265. package/v2Containers/InApp/__tests__/InAppHTMLEditor.test.js +0 -376
  266. package/v2Containers/InApp/__tests__/sagas.test.js +0 -363
  267. package/v2Containers/InApp/tests/selectors.test.js +0 -612
  268. package/v2Containers/InAppWrapper/components/InAppWrapperView.js +0 -151
  269. package/v2Containers/InAppWrapper/components/__tests__/InAppWrapperView.test.js +0 -267
  270. package/v2Containers/InAppWrapper/components/inAppWrapperView.scss +0 -23
  271. package/v2Containers/InAppWrapper/constants.js +0 -16
  272. package/v2Containers/InAppWrapper/hooks/__tests__/useInAppWrapper.test.js +0 -473
  273. package/v2Containers/InAppWrapper/hooks/useInAppWrapper.js +0 -198
  274. package/v2Containers/InAppWrapper/index.js +0 -148
  275. package/v2Containers/InAppWrapper/messages.js +0 -49
  276. package/v2Containers/InappAdvance/index.js +0 -1099
  277. package/v2Containers/InappAdvance/index.scss +0 -10
  278. package/v2Containers/InappAdvance/tests/index.test.js +0 -448
@@ -0,0 +1,142 @@
1
+ import React from 'react';
2
+ import PropTypes from 'prop-types';
3
+ import { FormattedMessage } from 'react-intl';
4
+ import CapRow from '@capillarytech/cap-ui-library/CapRow';
5
+ import CapHeading from '@capillarytech/cap-ui-library/CapHeading';
6
+ import CapSelect from '@capillarytech/cap-ui-library/CapSelect';
7
+ import CapDivider from '@capillarytech/cap-ui-library/CapDivider';
8
+ import CapImageUpload from '../../../../v2Components/CapImageUpload';
9
+ import CapImageUrlUpload from '../../../../v2Components/CapImageUrlUpload';
10
+ import { WEBPUSH } from '../../../CreativesContainer/constants';
11
+ import {
12
+ WEBPUSH_MEDIA_TYPES,
13
+ WEBPUSH_MEDIA_TYPES_OPTIONS,
14
+ ALLOWED_IMAGE_EXTENSIONS_REGEX,
15
+ WEBPUSH_IMG_SIZE,
16
+ WEBPUSH_RECOMMENDED_DIMENSIONS,
17
+ IMAGE_UPLOAD_METHODS,
18
+ } from '../../constants';
19
+
20
+ /**
21
+ * MediaSection component - Media type selection and image upload
22
+ */
23
+ export const MediaSection = ({
24
+ mediaType,
25
+ onMediaTypeChange,
26
+ imageUpload,
27
+ isLocked,
28
+ isAnyUploadActive,
29
+ formatMessage,
30
+ messages,
31
+ webPush,
32
+ isFullMode,
33
+ }) => {
34
+ const {
35
+ uploadMethod,
36
+ imageSrc,
37
+ imageUrl,
38
+ isValidating,
39
+ isUploading,
40
+ uploadAsset,
41
+ setUpdateImageSrc,
42
+ updateOnReUpload,
43
+ handleUrlChange,
44
+ handleValidationStateChange,
45
+ handleUploadStateChange,
46
+ } = imageUpload;
47
+
48
+ const isImageFieldActive = isValidating || isUploading;
49
+
50
+ return (
51
+ <>
52
+ <CapRow className="creatives-webpush-media">
53
+ <CapHeading type="h4" className="webpush-media-type">
54
+ <FormattedMessage {...messages.mediaType} />
55
+ </CapHeading>
56
+ <CapSelect.CapCustomSelect
57
+ width="100%"
58
+ className="margin-t-4"
59
+ options={WEBPUSH_MEDIA_TYPES_OPTIONS}
60
+ value={mediaType}
61
+ onChange={onMediaTypeChange}
62
+ disabled={isAnyUploadActive}
63
+ />
64
+ </CapRow>
65
+ {mediaType === WEBPUSH_MEDIA_TYPES.IMAGE && (
66
+ <>
67
+ {uploadMethod === IMAGE_UPLOAD_METHODS.UPLOAD_IMAGE && (
68
+ <CapRow
69
+ className="webpush-image-upload-section"
70
+ style={isLocked ? { pointerEvents: 'none', opacity: 0.5 } : undefined}
71
+ aria-disabled={isLocked}
72
+ >
73
+ <CapImageUpload
74
+ allowedExtensionsRegex={ALLOWED_IMAGE_EXTENSIONS_REGEX}
75
+ imgSize={WEBPUSH_IMG_SIZE}
76
+ uploadAsset={uploadAsset}
77
+ isFullMode={isFullMode}
78
+ imageSrc={imageSrc}
79
+ updateImageSrc={setUpdateImageSrc}
80
+ updateOnReUpload={updateOnReUpload}
81
+ index={0}
82
+ className="cap-custom-image-upload"
83
+ key="webpush-uploaded-image"
84
+ imageData={webPush}
85
+ channel={WEBPUSH}
86
+ showReUploadButton
87
+ recommendedDimensions={WEBPUSH_RECOMMENDED_DIMENSIONS}
88
+ />
89
+ </CapRow>
90
+ )}
91
+ {uploadMethod === IMAGE_UPLOAD_METHODS.ADD_IMAGE_URL && (
92
+ <CapRow
93
+ className="webpush-image-url-section"
94
+ style={isLocked ? { pointerEvents: 'none', opacity: 0.5 } : undefined}
95
+ aria-disabled={isLocked}
96
+ >
97
+ <CapImageUrlUpload
98
+ uploadAsset={uploadAsset}
99
+ imgSize={WEBPUSH_IMG_SIZE}
100
+ recommendedDimensions={WEBPUSH_RECOMMENDED_DIMENSIONS}
101
+ sizeLabel={formatMessage(messages.sizeLimit)}
102
+ formatLabel={formatMessage(messages.formatTypes)}
103
+ imageUrl={imageUrl}
104
+ imageSrc={imageSrc}
105
+ onUrlChange={handleUrlChange}
106
+ onValidationStateChange={handleValidationStateChange}
107
+ onUploadStateChange={handleUploadStateChange}
108
+ isExternalUploading={isImageFieldActive}
109
+ disabled={isLocked}
110
+ className="webpush-image-url-upload"
111
+ fileNamePrefix="webpush-image"
112
+ />
113
+ </CapRow>
114
+ )}
115
+ </>
116
+ )}
117
+ <CapDivider />
118
+ </>
119
+ );
120
+ };
121
+
122
+ MediaSection.propTypes = {
123
+ mediaType: PropTypes.string.isRequired,
124
+ onMediaTypeChange: PropTypes.func.isRequired,
125
+ imageUpload: PropTypes.object.isRequired,
126
+ isLocked: PropTypes.bool,
127
+ isAnyUploadActive: PropTypes.bool,
128
+ formatMessage: PropTypes.func.isRequired,
129
+ messages: PropTypes.object.isRequired,
130
+ webPush: PropTypes.object,
131
+ isFullMode: PropTypes.bool,
132
+ };
133
+
134
+ MediaSection.defaultProps = {
135
+ isLocked: false,
136
+ isAnyUploadActive: false,
137
+ webPush: {},
138
+ isFullMode: true,
139
+ };
140
+
141
+ export default MediaSection;
142
+
@@ -0,0 +1,341 @@
1
+ import React from 'react';
2
+ import { mountWithIntl, shallowWithIntl } from '../../../../helpers/intl-enzym-test-helpers';
3
+ import MediaSection from './MediaSection';
4
+ import CapRow from '@capillarytech/cap-ui-library/CapRow';
5
+ import CapHeading from '@capillarytech/cap-ui-library/CapHeading';
6
+ import CapSelect from '@capillarytech/cap-ui-library/CapSelect';
7
+ import CapDivider from '@capillarytech/cap-ui-library/CapDivider';
8
+ import {
9
+ WEBPUSH_MEDIA_TYPES,
10
+ IMAGE_UPLOAD_METHODS,
11
+ } from '../../constants';
12
+
13
+ // Mock CapImageUpload
14
+ jest.mock('../../../../v2Components/CapImageUpload', () => {
15
+ const React = require('react');
16
+ return function MockCapImageUpload(props) {
17
+ return (
18
+ <div data-testid="cap-image-upload" className="cap-image-upload-mock">
19
+ <div>CapImageUpload</div>
20
+ <div>imageSrc: {props.imageSrc}</div>
21
+ <div>channel: {props.channel}</div>
22
+ </div>
23
+ );
24
+ };
25
+ });
26
+
27
+ // Mock CapImageUrlUpload
28
+ jest.mock('../../../../v2Components/CapImageUrlUpload', () => {
29
+ const React = require('react');
30
+ return function MockCapImageUrlUpload(props) {
31
+ return (
32
+ <div data-testid="cap-image-url-upload" className="cap-image-url-upload-mock">
33
+ <div>CapImageUrlUpload</div>
34
+ <div>imageUrl: {props.imageUrl}</div>
35
+ <div>imageSrc: {props.imageSrc}</div>
36
+ <div>disabled: {props.disabled ? 'true' : 'false'}</div>
37
+ </div>
38
+ );
39
+ };
40
+ });
41
+
42
+ describe('MediaSection', () => {
43
+ const mockFormatMessage = jest.fn((message) => message?.defaultMessage || message?.id || message);
44
+ const mockOnMediaTypeChange = jest.fn();
45
+
46
+ const mockMessages = {
47
+ mediaType: {
48
+ id: 'app.webpush.mediaType',
49
+ defaultMessage: 'Media Type',
50
+ },
51
+ sizeLimit: {
52
+ id: 'app.webpush.sizeLimit',
53
+ defaultMessage: 'Size limit',
54
+ },
55
+ formatTypes: {
56
+ id: 'app.webpush.formatTypes',
57
+ defaultMessage: 'Format types',
58
+ },
59
+ };
60
+
61
+ const mockImageUpload = {
62
+ uploadMethod: IMAGE_UPLOAD_METHODS.UPLOAD_IMAGE,
63
+ imageSrc: '',
64
+ imageUrl: '',
65
+ isValidating: false,
66
+ isUploading: false,
67
+ uploadAsset: jest.fn(),
68
+ setUpdateImageSrc: jest.fn(),
69
+ updateOnReUpload: jest.fn(),
70
+ handleUrlChange: jest.fn(),
71
+ handleValidationStateChange: jest.fn(),
72
+ handleUploadStateChange: jest.fn(),
73
+ };
74
+
75
+ const defaultProps = {
76
+ mediaType: WEBPUSH_MEDIA_TYPES.NONE,
77
+ onMediaTypeChange: mockOnMediaTypeChange,
78
+ imageUpload: mockImageUpload,
79
+ isLocked: false,
80
+ isAnyUploadActive: false,
81
+ formatMessage: mockFormatMessage,
82
+ messages: mockMessages,
83
+ webPush: {},
84
+ isFullMode: true,
85
+ };
86
+
87
+ beforeEach(() => {
88
+ jest.clearAllMocks();
89
+ });
90
+
91
+ describe('Rendering', () => {
92
+ it('should render correctly with default props', () => {
93
+ const wrapper = shallowWithIntl(<MediaSection {...defaultProps} />);
94
+ expect(wrapper).toMatchSnapshot();
95
+ });
96
+
97
+ it('should render main container with correct class', () => {
98
+ const wrapper = mountWithIntl(<MediaSection {...defaultProps} />);
99
+ const container = wrapper.find('.creatives-webpush-media');
100
+ expect(container.exists()).toBe(true);
101
+ });
102
+
103
+ it('should render heading', () => {
104
+ const wrapper = mountWithIntl(<MediaSection {...defaultProps} />);
105
+ const heading = wrapper.find(CapHeading).find('.webpush-media-type');
106
+ expect(heading.exists()).toBe(true);
107
+ });
108
+
109
+ it('should render CapSelect.CapCustomSelect with correct props', () => {
110
+ const wrapper = mountWithIntl(<MediaSection {...defaultProps} />);
111
+ const select = wrapper.find(CapSelect.CapCustomSelect);
112
+ expect(select.exists()).toBe(true);
113
+ expect(select.prop('width')).toBe('100%');
114
+ expect(select.prop('className')).toBe('margin-t-4');
115
+ expect(select.prop('value')).toBe(WEBPUSH_MEDIA_TYPES.NONE);
116
+ expect(select.prop('disabled')).toBe(false);
117
+ });
118
+
119
+ it('should render Divider', () => {
120
+ const wrapper = mountWithIntl(<MediaSection {...defaultProps} />);
121
+ const divider = wrapper.find(CapDivider);
122
+ expect(divider.exists()).toBe(true);
123
+ });
124
+ });
125
+
126
+ describe('Media Type Selection', () => {
127
+ it('should call onMediaTypeChange when select changes', () => {
128
+ const wrapper = mountWithIntl(<MediaSection {...defaultProps} />);
129
+ const select = wrapper.find(CapSelect.CapCustomSelect);
130
+ select.prop('onChange')(WEBPUSH_MEDIA_TYPES.IMAGE);
131
+ expect(mockOnMediaTypeChange).toHaveBeenCalledWith(WEBPUSH_MEDIA_TYPES.IMAGE);
132
+ });
133
+
134
+ it('should disable select when isAnyUploadActive is true', () => {
135
+ const wrapper = mountWithIntl(
136
+ <MediaSection {...defaultProps} isAnyUploadActive={true} />
137
+ );
138
+ const select = wrapper.find(CapSelect.CapCustomSelect);
139
+ expect(select.prop('disabled')).toBe(true);
140
+ });
141
+ });
142
+
143
+ describe('Image Upload - UPLOAD_IMAGE Method', () => {
144
+ it('should render CapImageUpload when mediaType is IMAGE and uploadMethod is UPLOAD_IMAGE', () => {
145
+ const wrapper = mountWithIntl(
146
+ <MediaSection
147
+ {...defaultProps}
148
+ mediaType={WEBPUSH_MEDIA_TYPES.IMAGE}
149
+ imageUpload={{
150
+ ...mockImageUpload,
151
+ uploadMethod: IMAGE_UPLOAD_METHODS.UPLOAD_IMAGE,
152
+ imageSrc: 'path/to/image.jpg',
153
+ }}
154
+ />
155
+ );
156
+ const imageUpload = wrapper.find('[data-testid="cap-image-upload"]');
157
+ expect(imageUpload.exists()).toBe(true);
158
+ });
159
+
160
+ it('should not render CapImageUpload when mediaType is NONE', () => {
161
+ const wrapper = mountWithIntl(<MediaSection {...defaultProps} />);
162
+ const imageUpload = wrapper.find('[data-testid="cap-image-upload"]');
163
+ expect(imageUpload.exists()).toBe(false);
164
+ });
165
+
166
+ it('should apply locked styles when isLocked is true', () => {
167
+ const wrapper = mountWithIntl(
168
+ <MediaSection
169
+ {...defaultProps}
170
+ mediaType={WEBPUSH_MEDIA_TYPES.IMAGE}
171
+ isLocked={true}
172
+ />
173
+ );
174
+ const uploadSection = wrapper.find('.webpush-image-upload-section').first();
175
+ expect(uploadSection.exists()).toBe(true);
176
+ expect(uploadSection.prop('aria-disabled')).toBe(true);
177
+ });
178
+
179
+ it('should not apply locked styles when isLocked is false', () => {
180
+ const wrapper = mountWithIntl(
181
+ <MediaSection
182
+ {...defaultProps}
183
+ mediaType={WEBPUSH_MEDIA_TYPES.IMAGE}
184
+ isLocked={false}
185
+ />
186
+ );
187
+ const uploadSection = wrapper.find('.webpush-image-upload-section');
188
+ expect(uploadSection.exists()).toBe(true);
189
+ });
190
+ });
191
+
192
+ describe('Image Upload - ADD_IMAGE_URL Method', () => {
193
+ it('should render CapImageUrlUpload when mediaType is IMAGE and uploadMethod is ADD_IMAGE_URL', () => {
194
+ const wrapper = mountWithIntl(
195
+ <MediaSection
196
+ {...defaultProps}
197
+ mediaType={WEBPUSH_MEDIA_TYPES.IMAGE}
198
+ imageUpload={{
199
+ ...mockImageUpload,
200
+ uploadMethod: IMAGE_UPLOAD_METHODS.ADD_IMAGE_URL,
201
+ imageUrl: 'https://example.com/image.jpg',
202
+ }}
203
+ />
204
+ );
205
+ const imageUrlUpload = wrapper.find('[data-testid="cap-image-url-upload"]');
206
+ expect(imageUrlUpload.exists()).toBe(true);
207
+ });
208
+
209
+ it('should not render CapImageUrlUpload when mediaType is NONE', () => {
210
+ const wrapper = mountWithIntl(<MediaSection {...defaultProps} />);
211
+ const imageUrlUpload = wrapper.find('[data-testid="cap-image-url-upload"]');
212
+ expect(imageUrlUpload.exists()).toBe(false);
213
+ });
214
+
215
+ it('should apply locked styles when isLocked is true', () => {
216
+ const wrapper = mountWithIntl(
217
+ <MediaSection
218
+ {...defaultProps}
219
+ mediaType={WEBPUSH_MEDIA_TYPES.IMAGE}
220
+ imageUpload={{
221
+ ...mockImageUpload,
222
+ uploadMethod: IMAGE_UPLOAD_METHODS.ADD_IMAGE_URL,
223
+ }}
224
+ isLocked={true}
225
+ />
226
+ );
227
+ const urlSection = wrapper.find('.webpush-image-url-section').first();
228
+ expect(urlSection.exists()).toBe(true);
229
+ expect(urlSection.prop('aria-disabled')).toBe(true);
230
+ });
231
+
232
+ it('should pass disabled prop to CapImageUrlUpload when isLocked is true', () => {
233
+ const wrapper = mountWithIntl(
234
+ <MediaSection
235
+ {...defaultProps}
236
+ mediaType={WEBPUSH_MEDIA_TYPES.IMAGE}
237
+ imageUpload={{
238
+ ...mockImageUpload,
239
+ uploadMethod: IMAGE_UPLOAD_METHODS.ADD_IMAGE_URL,
240
+ }}
241
+ isLocked={true}
242
+ />
243
+ );
244
+ const imageUrlUpload = wrapper.find('[data-testid="cap-image-url-upload"]');
245
+ expect(imageUrlUpload.exists()).toBe(true);
246
+ });
247
+ });
248
+
249
+ describe('Image Upload State', () => {
250
+ it('should set isImageFieldActive when isValidating is true', () => {
251
+ const wrapper = mountWithIntl(
252
+ <MediaSection
253
+ {...defaultProps}
254
+ mediaType={WEBPUSH_MEDIA_TYPES.IMAGE}
255
+ imageUpload={{
256
+ ...mockImageUpload,
257
+ uploadMethod: IMAGE_UPLOAD_METHODS.ADD_IMAGE_URL,
258
+ isValidating: true,
259
+ }}
260
+ />
261
+ );
262
+ const imageUrlUpload = wrapper.find('[data-testid="cap-image-url-upload"]');
263
+ expect(imageUrlUpload.exists()).toBe(true);
264
+ });
265
+
266
+ it('should set isImageFieldActive when isUploading is true', () => {
267
+ const wrapper = mountWithIntl(
268
+ <MediaSection
269
+ {...defaultProps}
270
+ mediaType={WEBPUSH_MEDIA_TYPES.IMAGE}
271
+ imageUpload={{
272
+ ...mockImageUpload,
273
+ uploadMethod: IMAGE_UPLOAD_METHODS.ADD_IMAGE_URL,
274
+ isUploading: true,
275
+ }}
276
+ />
277
+ );
278
+ const imageUrlUpload = wrapper.find('[data-testid="cap-image-url-upload"]');
279
+ expect(imageUrlUpload.exists()).toBe(true);
280
+ });
281
+ });
282
+
283
+ describe('Default Props', () => {
284
+ it('should use default isLocked when not provided', () => {
285
+ const wrapper = mountWithIntl(
286
+ <MediaSection
287
+ mediaType={WEBPUSH_MEDIA_TYPES.NONE}
288
+ onMediaTypeChange={mockOnMediaTypeChange}
289
+ imageUpload={mockImageUpload}
290
+ formatMessage={mockFormatMessage}
291
+ messages={mockMessages}
292
+ />
293
+ );
294
+ const select = wrapper.find(CapSelect.CapCustomSelect);
295
+ expect(select.prop('disabled')).toBe(false);
296
+ });
297
+
298
+ it('should use default isAnyUploadActive when not provided', () => {
299
+ const wrapper = mountWithIntl(
300
+ <MediaSection
301
+ mediaType={WEBPUSH_MEDIA_TYPES.NONE}
302
+ onMediaTypeChange={mockOnMediaTypeChange}
303
+ imageUpload={mockImageUpload}
304
+ formatMessage={mockFormatMessage}
305
+ messages={mockMessages}
306
+ />
307
+ );
308
+ const select = wrapper.find(CapSelect.CapCustomSelect);
309
+ expect(select.prop('disabled')).toBe(false);
310
+ });
311
+
312
+ it('should use default webPush when not provided', () => {
313
+ const wrapper = mountWithIntl(
314
+ <MediaSection
315
+ mediaType={WEBPUSH_MEDIA_TYPES.IMAGE}
316
+ onMediaTypeChange={mockOnMediaTypeChange}
317
+ imageUpload={mockImageUpload}
318
+ formatMessage={mockFormatMessage}
319
+ messages={mockMessages}
320
+ />
321
+ );
322
+ const imageUpload = wrapper.find('[data-testid="cap-image-upload"]');
323
+ expect(imageUpload.exists()).toBe(true);
324
+ });
325
+
326
+ it('should use default isFullMode when not provided', () => {
327
+ const wrapper = mountWithIntl(
328
+ <MediaSection
329
+ mediaType={WEBPUSH_MEDIA_TYPES.IMAGE}
330
+ onMediaTypeChange={mockOnMediaTypeChange}
331
+ imageUpload={mockImageUpload}
332
+ formatMessage={mockFormatMessage}
333
+ messages={mockMessages}
334
+ />
335
+ );
336
+ const imageUpload = wrapper.find('[data-testid="cap-image-upload"]');
337
+ expect(imageUpload.exists()).toBe(true);
338
+ });
339
+ });
340
+ });
341
+
@@ -0,0 +1,103 @@
1
+ import React from 'react';
2
+ import PropTypes from 'prop-types';
3
+ import { FormattedMessage } from 'react-intl';
4
+ import CapInput from '@capillarytech/cap-ui-library/CapInput';
5
+ import CapRow from '@capillarytech/cap-ui-library/CapRow';
6
+ import CapHeading from '@capillarytech/cap-ui-library/CapHeading';
7
+ import CapError from '@capillarytech/cap-ui-library/CapError';
8
+ import CapLabel from '@capillarytech/cap-ui-library/CapLabel';
9
+ import CapEmojiPicker from '@capillarytech/cap-ui-library/CapEmojiPicker';
10
+ import CapDivider from '@capillarytech/cap-ui-library/CapDivider';
11
+ import { MESSAGE_MAX_LENGTH, SHOW_CHARACTER_COUNT } from '../../constants';
12
+
13
+ /**
14
+ * MessageSection component - Message textarea with tags, emoji picker, and character count
15
+ */
16
+ export const MessageSection = ({
17
+ value,
18
+ error,
19
+ onChange,
20
+ formatMessage,
21
+ messages,
22
+ tagList,
23
+ messageCountRef,
24
+ messageTextAreaRef,
25
+ handleMessageTextAreaRef,
26
+ }) => {
27
+ const renderCharacterCount = () => {
28
+ if (!SHOW_CHARACTER_COUNT) return null;
29
+
30
+ const maxLength = MESSAGE_MAX_LENGTH;
31
+
32
+ return (
33
+ <CapLabel type="label2" className="webpush-character-count">
34
+ <span ref={messageCountRef}>
35
+ {formatMessage(messages.characterCount, {
36
+ currentLength: value.length,
37
+ maxLength,
38
+ })}
39
+ </span>
40
+ </CapLabel>
41
+ );
42
+ };
43
+
44
+ return (
45
+ <>
46
+ <CapRow className="creatives-webpush-message">
47
+ <CapRow className="tooltip-add-label-container webpush-message-taglist">
48
+ {tagList}
49
+ </CapRow>
50
+ <CapHeading type="h4" className="webpush-message">
51
+ <FormattedMessage {...messages.message} />
52
+ </CapHeading>
53
+ <CapEmojiPicker.Wrapper
54
+ value={value}
55
+ onChange={onChange}
56
+ textAreaRef={messageTextAreaRef}
57
+ >
58
+ <CapInput.TextArea
59
+ id="webpush-message-input"
60
+ value={value}
61
+ onChange={onChange}
62
+ placeholder={formatMessage(messages.messagePlaceholder)}
63
+ size="default"
64
+ isRequired
65
+ autosize={{ minRows: 3, maxRows: 5 }}
66
+ setInputRef={handleMessageTextAreaRef}
67
+ errorMessage={
68
+ error && (
69
+ <CapError className="webpush-template-message-error">
70
+ {error}
71
+ </CapError>
72
+ )
73
+ }
74
+ />
75
+ </CapEmojiPicker.Wrapper>
76
+ {renderCharacterCount()}
77
+ </CapRow>
78
+ <CapDivider className="webpush-message-divider" />
79
+ </>
80
+ );
81
+ };
82
+
83
+ MessageSection.propTypes = {
84
+ value: PropTypes.string.isRequired,
85
+ error: PropTypes.string,
86
+ onChange: PropTypes.func.isRequired,
87
+ formatMessage: PropTypes.func.isRequired,
88
+ messages: PropTypes.object.isRequired,
89
+ tagList: PropTypes.node,
90
+ messageCountRef: PropTypes.object,
91
+ messageTextAreaRef: PropTypes.object,
92
+ handleMessageTextAreaRef: PropTypes.func.isRequired,
93
+ };
94
+
95
+ MessageSection.defaultProps = {
96
+ error: '',
97
+ tagList: null,
98
+ messageCountRef: null,
99
+ messageTextAreaRef: null,
100
+ };
101
+
102
+ export default MessageSection;
103
+