@capillarytech/creatives-library 8.0.271 → 8.0.273
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/assets/Android.png +0 -0
- package/assets/iOS.png +0 -0
- package/constants/unified.js +2 -1
- package/initialReducer.js +2 -0
- package/package.json +1 -1
- package/services/api.js +10 -0
- package/services/tests/api.test.js +34 -0
- package/tests/integration/TemplateCreation/TemplateCreation.integration.test.js +17 -35
- package/tests/integration/TemplateCreation/api-response.js +31 -1
- package/tests/integration/TemplateCreation/msw-handler.js +2 -0
- package/utils/common.js +5 -0
- package/utils/commonUtils.js +28 -5
- package/utils/imageUrlUpload.js +13 -14
- package/utils/tests/commonUtil.test.js +224 -0
- package/utils/tests/imageUrlUpload.test.js +298 -0
- package/utils/transformTemplateConfig.js +0 -10
- package/v2Components/CapDeviceContent/index.js +61 -56
- package/v2Components/CapTagList/index.js +6 -1
- package/v2Components/CapTagListWithInput/index.js +5 -1
- package/v2Components/CapTagListWithInput/messages.js +1 -1
- package/v2Components/CapWhatsappCTA/tests/index.test.js +5 -0
- package/v2Components/ErrorInfoNote/constants.js +1 -0
- package/v2Components/ErrorInfoNote/index.js +402 -72
- package/v2Components/ErrorInfoNote/messages.js +32 -6
- package/v2Components/ErrorInfoNote/style.scss +278 -6
- package/v2Components/FormBuilder/tests/index.test.js +13 -4
- package/v2Components/HtmlEditor/HTMLEditor.js +418 -99
- package/v2Components/HtmlEditor/__tests__/HTMLEditor.apiErrors.test.js +870 -0
- package/v2Components/HtmlEditor/__tests__/HTMLEditor.test.js +1882 -133
- package/v2Components/HtmlEditor/__tests__/index.lazy.test.js +27 -16
- package/v2Components/HtmlEditor/_htmlEditor.scss +108 -45
- package/v2Components/HtmlEditor/_index.lazy.scss +0 -1
- package/v2Components/HtmlEditor/components/CodeEditorPane/_codeEditorPane.scss +23 -102
- package/v2Components/HtmlEditor/components/CodeEditorPane/index.js +148 -140
- package/v2Components/HtmlEditor/components/DeviceToggle/_deviceToggle.scss +2 -1
- package/v2Components/HtmlEditor/components/DeviceToggle/index.js +3 -3
- package/v2Components/HtmlEditor/components/EditorToolbar/_editorToolbar.scss +9 -1
- package/v2Components/HtmlEditor/components/EditorToolbar/index.js +31 -6
- package/v2Components/HtmlEditor/components/FullscreenModal/_fullscreenModal.scss +22 -0
- package/v2Components/HtmlEditor/components/InAppPreviewPane/DeviceFrame.js +4 -7
- package/v2Components/HtmlEditor/components/InAppPreviewPane/__tests__/DeviceFrame.test.js +35 -45
- package/v2Components/HtmlEditor/components/InAppPreviewPane/_inAppPreviewPane.scss +1 -3
- package/v2Components/HtmlEditor/components/InAppPreviewPane/constants.js +33 -33
- package/v2Components/HtmlEditor/components/InAppPreviewPane/index.js +7 -6
- package/v2Components/HtmlEditor/components/PreviewPane/_previewPane.scss +7 -10
- package/v2Components/HtmlEditor/components/PreviewPane/index.js +22 -43
- package/v2Components/HtmlEditor/components/SplitContainer/_splitContainer.scss +1 -1
- package/v2Components/HtmlEditor/components/ValidationErrorDisplay/_validationErrorDisplay.scss +18 -0
- package/v2Components/HtmlEditor/components/ValidationErrorDisplay/index.js +36 -31
- package/v2Components/HtmlEditor/components/ValidationPanel/_validationPanel.scss +46 -34
- package/v2Components/HtmlEditor/components/ValidationPanel/constants.js +6 -0
- package/v2Components/HtmlEditor/components/ValidationPanel/index.js +52 -46
- package/v2Components/HtmlEditor/components/ValidationTabs/_validationTabs.scss +277 -0
- package/v2Components/HtmlEditor/components/ValidationTabs/index.js +295 -0
- package/v2Components/HtmlEditor/components/ValidationTabs/messages.js +51 -0
- package/v2Components/HtmlEditor/constants.js +45 -20
- package/v2Components/HtmlEditor/hooks/__tests__/useInAppContent.test.js +373 -16
- package/v2Components/HtmlEditor/hooks/__tests__/useValidation.test.js +351 -16
- package/v2Components/HtmlEditor/hooks/useEditorContent.js +5 -2
- package/v2Components/HtmlEditor/hooks/useInAppContent.js +88 -146
- package/v2Components/HtmlEditor/hooks/useValidation.js +213 -56
- package/v2Components/HtmlEditor/index.js +1 -1
- package/v2Components/HtmlEditor/messages.js +102 -94
- package/v2Components/HtmlEditor/utils/__tests__/htmlValidator.enhanced.test.js +214 -45
- package/v2Components/HtmlEditor/utils/__tests__/validationAdapter.test.js +134 -0
- package/v2Components/HtmlEditor/utils/contentSanitizer.js +40 -41
- package/v2Components/HtmlEditor/utils/htmlValidator.js +71 -72
- package/v2Components/HtmlEditor/utils/liquidTemplateSupport.js +158 -124
- package/v2Components/HtmlEditor/utils/properSyntaxHighlighting.js +23 -25
- package/v2Components/HtmlEditor/utils/validationAdapter.js +66 -41
- package/v2Components/HtmlEditor/utils/validationConstants.js +38 -0
- package/v2Components/MobilePushPreviewV2/constants.js +6 -0
- package/v2Components/MobilePushPreviewV2/index.js +33 -7
- package/v2Components/TemplatePreview/_templatePreview.scss +55 -24
- package/v2Components/TemplatePreview/index.js +47 -32
- package/v2Components/TemplatePreview/messages.js +4 -0
- package/v2Components/TestAndPreviewSlidebox/_testAndPreviewSlidebox.scss +1 -0
- package/v2Containers/BeeEditor/index.js +172 -90
- package/v2Containers/BeePopupEditor/_beePopupEditor.scss +14 -0
- package/v2Containers/BeePopupEditor/constants.js +10 -0
- package/v2Containers/BeePopupEditor/index.js +194 -0
- package/v2Containers/BeePopupEditor/tests/index.test.js +627 -0
- package/v2Containers/CreativesContainer/SlideBoxContent.js +127 -51
- package/v2Containers/CreativesContainer/SlideBoxFooter.js +156 -13
- package/v2Containers/CreativesContainer/SlideBoxHeader.js +2 -1
- package/v2Containers/CreativesContainer/constants.js +1 -0
- package/v2Containers/CreativesContainer/index.js +251 -47
- package/v2Containers/CreativesContainer/messages.js +8 -0
- package/v2Containers/CreativesContainer/tests/SlideBoxFooter.test.js +11 -2
- package/v2Containers/CreativesContainer/tests/__snapshots__/SlideBoxContent.test.js.snap +38 -50
- package/v2Containers/CreativesContainer/tests/__snapshots__/index.test.js.snap +103 -0
- package/v2Containers/Email/actions.js +7 -0
- package/v2Containers/Email/constants.js +5 -1
- package/v2Containers/Email/index.js +234 -29
- package/v2Containers/Email/messages.js +32 -0
- package/v2Containers/Email/reducer.js +12 -1
- package/v2Containers/Email/sagas.js +61 -7
- package/v2Containers/Email/tests/__snapshots__/reducer.test.js.snap +2 -0
- package/v2Containers/Email/tests/reducer.test.js +46 -0
- package/v2Containers/Email/tests/sagas.test.js +320 -29
- package/v2Containers/EmailWrapper/components/EmailHTMLEditor.js +1246 -0
- package/v2Containers/EmailWrapper/components/EmailWrapperView.js +212 -21
- package/v2Containers/EmailWrapper/components/HTMLEditorTesting.js +40 -74
- package/v2Containers/EmailWrapper/components/__tests__/EmailHTMLEditor.test.js +2614 -0
- package/v2Containers/EmailWrapper/components/__tests__/EmailWrapperView.test.js +520 -0
- package/v2Containers/EmailWrapper/components/__tests__/HTMLEditorTesting.test.js +2 -67
- package/v2Containers/EmailWrapper/constants.js +2 -0
- package/v2Containers/EmailWrapper/hooks/useEmailWrapper.js +627 -79
- package/v2Containers/EmailWrapper/index.js +103 -23
- package/v2Containers/EmailWrapper/messages.js +65 -1
- package/v2Containers/EmailWrapper/tests/useEmailWrapper.edgeCases.test.js +955 -0
- package/v2Containers/EmailWrapper/tests/useEmailWrapper.test.js +596 -82
- package/v2Containers/InApp/__tests__/InAppHTMLEditor.test.js +376 -0
- package/v2Containers/InApp/__tests__/sagas.test.js +363 -0
- package/v2Containers/InApp/actions.js +7 -0
- package/v2Containers/InApp/constants.js +20 -4
- package/v2Containers/InApp/index.js +802 -360
- package/v2Containers/InApp/index.scss +4 -3
- package/v2Containers/InApp/messages.js +7 -3
- package/v2Containers/InApp/reducer.js +21 -3
- package/v2Containers/InApp/sagas.js +29 -9
- package/v2Containers/InApp/selectors.js +25 -5
- package/v2Containers/InApp/tests/index.test.js +154 -50
- package/v2Containers/InApp/tests/reducer.test.js +34 -0
- package/v2Containers/InApp/tests/sagas.test.js +61 -9
- package/v2Containers/InApp/tests/selectors.test.js +612 -0
- package/v2Containers/InAppWrapper/components/InAppWrapperView.js +151 -0
- package/v2Containers/InAppWrapper/components/__tests__/InAppWrapperView.test.js +267 -0
- package/v2Containers/InAppWrapper/components/inAppWrapperView.scss +23 -0
- package/v2Containers/InAppWrapper/constants.js +16 -0
- package/v2Containers/InAppWrapper/hooks/__tests__/useInAppWrapper.test.js +473 -0
- package/v2Containers/InAppWrapper/hooks/useInAppWrapper.js +198 -0
- package/v2Containers/InAppWrapper/index.js +148 -0
- package/v2Containers/InAppWrapper/messages.js +49 -0
- package/v2Containers/InappAdvance/index.js +1099 -0
- package/v2Containers/InappAdvance/index.scss +10 -0
- package/v2Containers/InappAdvance/tests/index.test.js +448 -0
- package/v2Containers/Line/Container/ImageCarousel/tests/__snapshots__/content.test.js.snap +3 -0
- package/v2Containers/Line/Container/ImageCarousel/tests/__snapshots__/index.test.js.snap +2 -0
- package/v2Containers/Line/Container/Wrapper/tests/__snapshots__/index.test.js.snap +2 -0
- package/v2Containers/Line/Container/tests/__snapshots__/index.test.js.snap +9 -0
- package/v2Containers/MobilePush/Create/index.js +1 -1
- package/v2Containers/MobilePush/Edit/index.js +10 -6
- package/v2Containers/Rcs/tests/__snapshots__/index.test.js.snap +12 -0
- package/v2Containers/SmsTrai/Edit/tests/__snapshots__/index.test.js.snap +4 -0
- package/v2Containers/TagList/index.js +62 -19
- package/v2Containers/Templates/_templates.scss +60 -1
- package/v2Containers/Templates/index.js +89 -4
- package/v2Containers/Templates/messages.js +4 -0
- package/v2Containers/TemplatesV2/TemplatesV2.style.js +4 -2
- package/v2Containers/Whatsapp/tests/__snapshots__/index.test.js.snap +34 -0
- package/v2Components/HtmlEditor/components/ValidationErrorDisplay/__tests__/index.test.js +0 -152
- package/v2Containers/EmailWrapper/tests/EmailWrapperView.test.js +0 -214
package/assets/Android.png
CHANGED
|
Binary file
|
package/assets/iOS.png
CHANGED
|
Binary file
|
package/constants/unified.js
CHANGED
|
@@ -45,6 +45,7 @@ export const GIFT_CARDS = 'GIFT_CARDS';
|
|
|
45
45
|
export const PROMO_ENGINE = 'PROMO_ENGINE';
|
|
46
46
|
export const LIQUID_SUPPORT = 'ENABLE_LIQUID_SUPPORT';
|
|
47
47
|
export const ENABLE_NEW_MPUSH = 'ENABLE_NEW_MPUSH';
|
|
48
|
+
export const SUPPORT_CK_EDITOR = 'SUPPORT_CK_EDITOR';
|
|
48
49
|
export const CUSTOM_TAG = 'CustomTagMessage';
|
|
49
50
|
export const CUSTOMER_EXTENDED_FIELD = 'Customer extended fields';
|
|
50
51
|
export const EXTENDED_TAG = 'ExtendedTagMessage';
|
|
@@ -169,7 +170,7 @@ export const JAPANESE_HIDE_DATE_TAGS = [
|
|
|
169
170
|
"dd.mm.yy",
|
|
170
171
|
"dd Mon",
|
|
171
172
|
"dd/m/yyyy",
|
|
172
|
-
];
|
|
173
|
+
];
|
|
173
174
|
|
|
174
175
|
export const LIQUID_SUPPORTED_CHANNELS = [EMAIL, SMS, MOBILE_PUSH, INAPP];
|
|
175
176
|
|
package/initialReducer.js
CHANGED
|
@@ -15,6 +15,7 @@ import galleryReducer from './v2Containers/Assets/Gallery/reducer';
|
|
|
15
15
|
import CapCollapsibleLeftNavigationReducer from '@capillarytech/cap-ui-library/CapCollapsibleLeftNavigation/reducer';
|
|
16
16
|
import { AIRA_REDUCER_DOMAIN, askAiraReducer } from '@capillarytech/cap-ui-library/CapAskAira';
|
|
17
17
|
import previewAndTestReducer from './v2Components/TestAndPreviewSlidebox/reducer';
|
|
18
|
+
import inAppReducer from './v2Containers/InApp/reducer';
|
|
18
19
|
|
|
19
20
|
export const initialReducer = {
|
|
20
21
|
language: languageProviderReducer,
|
|
@@ -33,4 +34,5 @@ export const initialReducer = {
|
|
|
33
34
|
gallery: galleryReducer,
|
|
34
35
|
navigationConfig: CapCollapsibleLeftNavigationReducer,
|
|
35
36
|
previewAndTest: previewAndTestReducer,
|
|
37
|
+
inApp: inAppReducer,
|
|
36
38
|
};
|
package/package.json
CHANGED
package/services/api.js
CHANGED
|
@@ -470,6 +470,11 @@ export const getCmsTemplateSettingsV2 = (cmsType, projectId, cmsMode, langId, is
|
|
|
470
470
|
return API.get(url);
|
|
471
471
|
};
|
|
472
472
|
|
|
473
|
+
export const getCmsAccounts = (cmsType) => {
|
|
474
|
+
const url = `${API_ENDPOINT}/cms/accounts?type=${cmsType}`;
|
|
475
|
+
return API.get(url);
|
|
476
|
+
};
|
|
477
|
+
|
|
473
478
|
export const getCmsTemplateData = (cmsType, projectId, langId) => {
|
|
474
479
|
const url = `${API_ENDPOINT}/cms/getContent?type=${cmsType}&projectId=${projectId}&langId=${langId}`;
|
|
475
480
|
return API.get(url);
|
|
@@ -725,4 +730,9 @@ export const getAssetStatus = (type, assetId) => {
|
|
|
725
730
|
return request(url, getAPICallObject('GET'));
|
|
726
731
|
};
|
|
727
732
|
|
|
733
|
+
export const getBeePopupBuilderToken = () => {
|
|
734
|
+
const url = `${API_ENDPOINT}/common/getInappTokenData`;
|
|
735
|
+
return request(url, getAPICallObject('GET'));
|
|
736
|
+
};
|
|
737
|
+
|
|
728
738
|
export {request, getAPICallObject};
|
|
@@ -26,6 +26,8 @@ import {
|
|
|
26
26
|
updateMetaConfig,
|
|
27
27
|
getMediaDetails,
|
|
28
28
|
getAssetStatus,
|
|
29
|
+
getBeePopupBuilderToken,
|
|
30
|
+
getCmsAccounts,
|
|
29
31
|
} from '../api';
|
|
30
32
|
import { mockData } from './mockData';
|
|
31
33
|
import getSchema from '../getSchema';
|
|
@@ -973,3 +975,35 @@ describe('getAssetStatus', () => {
|
|
|
973
975
|
expect(callArgs[0]).toContain('/assets/video/asset-456/status');
|
|
974
976
|
});
|
|
975
977
|
});
|
|
978
|
+
|
|
979
|
+
describe('getBeePopupBuilderToken', () => {
|
|
980
|
+
it('should return correct response', async () => {
|
|
981
|
+
global.fetch.mockReturnValue(Promise.resolve({
|
|
982
|
+
status: 200,
|
|
983
|
+
json: () => Promise.resolve({
|
|
984
|
+
status: 200,
|
|
985
|
+
response: 'test',
|
|
986
|
+
}),
|
|
987
|
+
}));
|
|
988
|
+
const result = await getBeePopupBuilderToken();
|
|
989
|
+
expect(result).toEqual({
|
|
990
|
+
status: 200,
|
|
991
|
+
response: 'test',
|
|
992
|
+
});
|
|
993
|
+
});
|
|
994
|
+
});
|
|
995
|
+
|
|
996
|
+
describe('getCmsAccounts', () => {
|
|
997
|
+
it('should return a promise (line 473-476)', () => {
|
|
998
|
+
// Similar to other API tests, just verify it returns a Promise
|
|
999
|
+
const result = getCmsAccounts('bee');
|
|
1000
|
+
expect(result).toBeInstanceOf(Promise);
|
|
1001
|
+
});
|
|
1002
|
+
|
|
1003
|
+
it('should be callable with cmsType parameter', () => {
|
|
1004
|
+
// Verify function exists and can be called
|
|
1005
|
+
expect(typeof getCmsAccounts).toBe('function');
|
|
1006
|
+
const result = getCmsAccounts('bee');
|
|
1007
|
+
expect(result).toBeInstanceOf(Promise);
|
|
1008
|
+
});
|
|
1009
|
+
});
|
|
@@ -17,6 +17,7 @@ import { initialReducer } from '../../../initialReducer';
|
|
|
17
17
|
import { mockInitialState } from './mocks/initialState';
|
|
18
18
|
import TemplatesV2 from '../../../v2Containers/TemplatesV2';
|
|
19
19
|
import globalMessages from '../../../v2Containers/Cap/messages';
|
|
20
|
+
import rcsMessages from '../../../v2Containers/Rcs/messages';
|
|
20
21
|
import * as helper from './helper';
|
|
21
22
|
|
|
22
23
|
jest.mock('@capillarytech/cap-ui-utils', () => ({
|
|
@@ -147,13 +148,16 @@ describe("Creatives testing template creation", () => {
|
|
|
147
148
|
fireEvent.change(templateNameInput, {
|
|
148
149
|
target: { value: 'rcs_test_template' },
|
|
149
150
|
});
|
|
150
|
-
const
|
|
151
|
+
const richCardOption = await creativesScreen.findByText(/rich card/i);
|
|
152
|
+
fireEvent.click(richCardOption);
|
|
153
|
+
const imageRadioBtn = await creativesScreen.findByRole('radio', {
|
|
154
|
+
name: /image/i,
|
|
155
|
+
});
|
|
151
156
|
fireEvent.click(imageRadioBtn);
|
|
152
|
-
const imageUploader = await creativesScreen.findByText(
|
|
157
|
+
const imageUploader = await creativesScreen.findByText(
|
|
158
|
+
/drag and drop image here/i,
|
|
159
|
+
);
|
|
153
160
|
expect(imageUploader).toBeInTheDocument();
|
|
154
|
-
const noneRadioBtn = await creativesScreen.findByText(/none/i);
|
|
155
|
-
fireEvent.click(noneRadioBtn);
|
|
156
|
-
expect(imageUploader).not.toBeInTheDocument();
|
|
157
161
|
|
|
158
162
|
// Enter the template title
|
|
159
163
|
const templateTitleInput = await creativesScreen.findByTestId(
|
|
@@ -175,7 +179,7 @@ describe("Creatives testing template creation", () => {
|
|
|
175
179
|
|
|
176
180
|
// click on Done button
|
|
177
181
|
const doneButton = creativesScreen.getByRole('button', {
|
|
178
|
-
name:
|
|
182
|
+
name: rcsMessages.sendForApprovalButtonLabel.defaultMessage,
|
|
179
183
|
});
|
|
180
184
|
await userEvent.click(doneButton);
|
|
181
185
|
expect(createButton).toBeInTheDocument();
|
|
@@ -188,11 +192,13 @@ describe("Creatives testing template creation", () => {
|
|
|
188
192
|
name: /rcs/i,
|
|
189
193
|
});
|
|
190
194
|
await userEvent.click(rcs);
|
|
191
|
-
const createButton = creativesScreen.
|
|
195
|
+
const createButton = await creativesScreen.findByRole('button', {
|
|
192
196
|
name: /create new/i,
|
|
193
197
|
});
|
|
194
198
|
await waitFor(() => expect(createButton).toBeEnabled(),{ timeout: 5000, interval: 100 });
|
|
195
199
|
await userEvent.click(createButton);
|
|
200
|
+
const richCardOption = await creativesScreen.findByText(/rich card/i);
|
|
201
|
+
fireEvent.click(richCardOption);
|
|
196
202
|
const templateNameInput = await creativesScreen.findByTestId(
|
|
197
203
|
/template_name/,
|
|
198
204
|
{},
|
|
@@ -217,38 +223,14 @@ describe("Creatives testing template creation", () => {
|
|
|
217
223
|
fireEvent.change(templateTextInput, {
|
|
218
224
|
target: { value: 'rcs_template_text' },
|
|
219
225
|
});
|
|
220
|
-
const
|
|
221
|
-
|
|
222
|
-
const doneButton = creativesScreen.getByRole('button', {
|
|
223
|
-
name: globalMessages.done.defaultMessage,
|
|
226
|
+
const addButton = creativesScreen.getByRole('button', {
|
|
227
|
+
name: /add button/i,
|
|
224
228
|
});
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
//configuring cta button
|
|
229
|
-
//accessing cta button textBox by char limit
|
|
230
|
-
const textLimit = creativesScreen.getByText(/0 \/ 20/i);
|
|
231
|
-
const ctaBtnText = textLimit.parentElement.previousSibling;
|
|
232
|
-
|
|
233
|
-
//configuring CTA button text and URL
|
|
229
|
+
fireEvent.click(addButton);
|
|
230
|
+
const ctaBtnText = creativesScreen.getByPlaceholderText(/enter button text/i);
|
|
234
231
|
fireEvent.change(ctaBtnText, {
|
|
235
232
|
target: { value: 'cta_button' },
|
|
236
233
|
});
|
|
237
|
-
const ctaLink = await creativesScreen.findByTestId(
|
|
238
|
-
/cta_btn_link/,
|
|
239
|
-
{},
|
|
240
|
-
);
|
|
241
|
-
fireEvent.change(ctaLink, {
|
|
242
|
-
target: { value: 'invalid url' },
|
|
243
|
-
});
|
|
244
|
-
const urlError = creativesScreen.getByText(/button url is not valid/i);
|
|
245
|
-
expect(urlError).toBeInTheDocument();
|
|
246
|
-
fireEvent.change(ctaLink, {
|
|
247
|
-
target: { value: 'https://capillarytech.com/' },
|
|
248
|
-
});
|
|
249
|
-
expect(urlError).not.toBeInTheDocument();
|
|
250
|
-
//saving the template
|
|
251
|
-
fireEvent.click(doneButton);
|
|
252
234
|
expect(createButton).toBeInTheDocument();
|
|
253
235
|
});
|
|
254
236
|
|
|
@@ -1558,6 +1558,25 @@ export const whatsappAccount = {
|
|
|
1558
1558
|
response: []
|
|
1559
1559
|
};
|
|
1560
1560
|
|
|
1561
|
+
export const rcsAccount = {
|
|
1562
|
+
success: true,
|
|
1563
|
+
status: {
|
|
1564
|
+
isError: false,
|
|
1565
|
+
code: 200,
|
|
1566
|
+
message: "success",
|
|
1567
|
+
},
|
|
1568
|
+
message: "WeCRM Account details fetched",
|
|
1569
|
+
response: [
|
|
1570
|
+
{
|
|
1571
|
+
name: "RCS Account",
|
|
1572
|
+
sourceAccountIdentifier: "rcs_account",
|
|
1573
|
+
configs: {
|
|
1574
|
+
accessToken: "rcs_access_token",
|
|
1575
|
+
},
|
|
1576
|
+
},
|
|
1577
|
+
],
|
|
1578
|
+
};
|
|
1579
|
+
|
|
1561
1580
|
export const mpushAccount = {
|
|
1562
1581
|
success: true,
|
|
1563
1582
|
status: {
|
|
@@ -1716,7 +1735,18 @@ export const emailTemplates = {
|
|
|
1716
1735
|
|
|
1717
1736
|
export const domainProperties = {
|
|
1718
1737
|
entity:{
|
|
1719
|
-
WHATSAPP:[]
|
|
1738
|
+
WHATSAPP:[],
|
|
1739
|
+
RCS: [
|
|
1740
|
+
{
|
|
1741
|
+
domainProperties: {
|
|
1742
|
+
hostName: "rcs-host",
|
|
1743
|
+
connectionProperties: {
|
|
1744
|
+
sourceAccountIdentifier: "rcs_account",
|
|
1745
|
+
account_sid: "rcs_account",
|
|
1746
|
+
},
|
|
1747
|
+
},
|
|
1748
|
+
},
|
|
1749
|
+
],
|
|
1720
1750
|
},
|
|
1721
1751
|
warnings:[]
|
|
1722
1752
|
};
|
|
@@ -36,6 +36,8 @@ export const server = setupServer(
|
|
|
36
36
|
rest.get(`${API_ENDPOINT}/meta/wecrm`, (req, res, ctx) => {
|
|
37
37
|
const sourceName = req.url.searchParams.get('source_name');
|
|
38
38
|
switch (sourceName) {
|
|
39
|
+
case 'RCS':
|
|
40
|
+
return res(ctx.status(200), ctx.json(apiResponse.rcsAccount));
|
|
39
41
|
case 'WHATSAPP':
|
|
40
42
|
return res(ctx.status(200), ctx.json(apiResponse.whatsappAccount));
|
|
41
43
|
case 'VIBER':
|
package/utils/common.js
CHANGED
|
@@ -23,6 +23,7 @@ import {
|
|
|
23
23
|
ENABLE_WECHAT,
|
|
24
24
|
ENABLE_WEBPUSH,
|
|
25
25
|
LIQUID_SUPPORT,
|
|
26
|
+
SUPPORT_CK_EDITOR,
|
|
26
27
|
ENABLE_NEW_MPUSH
|
|
27
28
|
} from '../constants/unified';
|
|
28
29
|
import { apiMessageFormatHandler } from './commonUtils';
|
|
@@ -96,6 +97,10 @@ export const hasLiquidSupportFeature = Auth.hasFeatureAccess.bind(
|
|
|
96
97
|
LIQUID_SUPPORT,
|
|
97
98
|
);
|
|
98
99
|
|
|
100
|
+
export const hasSupportCKEditor = Auth.hasFeatureAccess.bind(
|
|
101
|
+
null,
|
|
102
|
+
SUPPORT_CK_EDITOR,
|
|
103
|
+
);
|
|
99
104
|
|
|
100
105
|
export const hasGiftVoucherFeature = Auth.hasFeatureAccess.bind(
|
|
101
106
|
null,
|
package/utils/commonUtils.js
CHANGED
|
@@ -173,20 +173,30 @@ export const validateLiquidTemplateContent = async (
|
|
|
173
173
|
// Handle API errors or empty content
|
|
174
174
|
if (result?.errors?.length > 0 || !validString || isError) {
|
|
175
175
|
let standardErrors = [];
|
|
176
|
+
let liquidErrors = [];
|
|
177
|
+
|
|
178
|
+
// Empty content errors are NOT from liquid endpoints, so they go to standardErrors
|
|
176
179
|
if (!validString) {
|
|
177
180
|
standardErrors = [emptyBodyError];
|
|
178
181
|
}
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
+
|
|
183
|
+
// IMPORTANT: Errors from extractTags and liquidValidation endpoints should ALWAYS be categorized as liquidErrors
|
|
184
|
+
// These endpoints are: /extractTags and /ask-aira-service/creatives_ai/liquidValidation
|
|
185
|
+
// Only errors from these endpoints should appear in Liquid Issues tab
|
|
186
|
+
if (result && Array.isArray(result?.errors) && result.errors.length > 0) {
|
|
187
|
+
// Errors from extractTags or liquidValidation endpoints
|
|
188
|
+
liquidErrors = result.errors.map((error) => {
|
|
182
189
|
const message = typeof error?.message === "string"
|
|
183
190
|
? error.message
|
|
184
191
|
: somethingWrongMsg;
|
|
185
192
|
return message;
|
|
186
193
|
});
|
|
187
|
-
} else {
|
|
194
|
+
} else if (isError) {
|
|
195
|
+
// If isError is true, it means the API call to extractTags/liquidValidation failed
|
|
196
|
+
// This is also a liquid endpoint error, so categorize as liquidErrors
|
|
188
197
|
liquidErrors = [somethingWrongMsg];
|
|
189
198
|
}
|
|
199
|
+
|
|
190
200
|
onError({
|
|
191
201
|
standardErrors,
|
|
192
202
|
liquidErrors,
|
|
@@ -398,7 +408,20 @@ export const validateMobilePushContent = async (formData, options) => {
|
|
|
398
408
|
// Helper function to extract content for a platform
|
|
399
409
|
export const extractContent = (platformData) => {
|
|
400
410
|
if (!platformData) return '';
|
|
401
|
-
const { title, message, ctas } = platformData;
|
|
411
|
+
const { title, message, ctas, isBEEeditor, beeHtml } = platformData;
|
|
412
|
+
|
|
413
|
+
// For BEE editor, extract content from beeHtml
|
|
414
|
+
if (isBEEeditor && beeHtml) {
|
|
415
|
+
// beeHtml can be an object with value property or a string
|
|
416
|
+
const beeHtmlContent = typeof beeHtml === 'string' ? beeHtml : (beeHtml?.value || '');
|
|
417
|
+
return [
|
|
418
|
+
title,
|
|
419
|
+
beeHtmlContent,
|
|
420
|
+
...((ctas?.map((cta) => cta?.text || cta?.actionLink)) || []),
|
|
421
|
+
].filter(Boolean).join(' ');
|
|
422
|
+
}
|
|
423
|
+
|
|
424
|
+
// For regular content
|
|
402
425
|
return [
|
|
403
426
|
title,
|
|
404
427
|
message,
|
package/utils/imageUrlUpload.js
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Utility functions for uploading images from URLs
|
|
3
|
-
*
|
|
3
|
+
*
|
|
4
4
|
* NOTE: CORS-limited; will be replaced with backend implementation.
|
|
5
5
|
* Flow currently hidden (not removed) from frontend.
|
|
6
6
|
*/
|
|
@@ -13,14 +13,14 @@ import {
|
|
|
13
13
|
|
|
14
14
|
/**
|
|
15
15
|
* Fetches an image from a URL
|
|
16
|
-
*
|
|
16
|
+
*
|
|
17
17
|
* @param {string} url - The image URL to fetch
|
|
18
18
|
* @returns {Promise<Response>} - The fetch response object
|
|
19
19
|
* @throws {Error} - If the fetch fails (network/CORS error)
|
|
20
20
|
*/
|
|
21
21
|
export const fetchImageFromUrl = async (url) => {
|
|
22
22
|
const trimmedUrl = url?.trim() || '';
|
|
23
|
-
|
|
23
|
+
|
|
24
24
|
if (!trimmedUrl) {
|
|
25
25
|
throw new Error('URL is required');
|
|
26
26
|
}
|
|
@@ -42,7 +42,7 @@ export const fetchImageFromUrl = async (url) => {
|
|
|
42
42
|
/**
|
|
43
43
|
* Helper function to upload image from URL
|
|
44
44
|
* Fetches image, validates content type and size, converts to File, and uploads via uploadAsset
|
|
45
|
-
*
|
|
45
|
+
*
|
|
46
46
|
* @param {string} url - The image URL to upload
|
|
47
47
|
* @param {Function} formatMessage - React Intl formatMessage function
|
|
48
48
|
* @param {Object} messages - React Intl messages object
|
|
@@ -51,7 +51,7 @@ export const fetchImageFromUrl = async (url) => {
|
|
|
51
51
|
* @param {number} maxSize - Maximum file size in bytes
|
|
52
52
|
* @param {string[]} allowedContentTypes - Array of allowed MIME types (defaults to DEFAULT_ALLOWED_CONTENT_TYPES)
|
|
53
53
|
* @returns {Promise<{success: boolean, error: string}>} - Result object with success status and error message
|
|
54
|
-
*
|
|
54
|
+
*
|
|
55
55
|
* @example
|
|
56
56
|
* const result = await uploadImageFromUrlHelper(
|
|
57
57
|
* 'https://example.com/image.jpg',
|
|
@@ -73,14 +73,14 @@ export const uploadImageFromUrlHelper = async (
|
|
|
73
73
|
allowedContentTypes = DEFAULT_ALLOWED_CONTENT_TYPES,
|
|
74
74
|
) => {
|
|
75
75
|
const trimmedUrl = url?.trim() || '';
|
|
76
|
-
|
|
76
|
+
|
|
77
77
|
try {
|
|
78
78
|
const response = await fetchImageFromUrl(trimmedUrl);
|
|
79
79
|
|
|
80
80
|
// Validate Content-Type
|
|
81
81
|
const contentType = response.headers?.get('Content-Type') || '';
|
|
82
82
|
const normalizedContentType = contentType.split(';')[0].toLowerCase().trim();
|
|
83
|
-
|
|
83
|
+
|
|
84
84
|
if (!allowedContentTypes.includes(normalizedContentType)) {
|
|
85
85
|
return {
|
|
86
86
|
success: false,
|
|
@@ -89,7 +89,7 @@ export const uploadImageFromUrlHelper = async (
|
|
|
89
89
|
}
|
|
90
90
|
|
|
91
91
|
const blob = await response.blob();
|
|
92
|
-
|
|
92
|
+
|
|
93
93
|
if (blob.size > maxSize) {
|
|
94
94
|
return {
|
|
95
95
|
success: false,
|
|
@@ -101,7 +101,7 @@ export const uploadImageFromUrlHelper = async (
|
|
|
101
101
|
return new Promise((resolve) => {
|
|
102
102
|
const img = new Image();
|
|
103
103
|
const objectUrl = URL.createObjectURL(blob);
|
|
104
|
-
|
|
104
|
+
|
|
105
105
|
img.onload = () => {
|
|
106
106
|
const extension = MIME_TYPE_TO_EXTENSION[normalizedContentType] || DEFAULT_IMAGE_EXTENSION;
|
|
107
107
|
const fileName = `${fileNamePrefix}.${extension}`;
|
|
@@ -111,16 +111,16 @@ export const uploadImageFromUrlHelper = async (
|
|
|
111
111
|
height: img.height,
|
|
112
112
|
error: false,
|
|
113
113
|
};
|
|
114
|
-
|
|
114
|
+
|
|
115
115
|
uploadAssetFn(file, 'image', fileParams);
|
|
116
116
|
URL.revokeObjectURL(objectUrl);
|
|
117
|
-
|
|
117
|
+
|
|
118
118
|
resolve({
|
|
119
119
|
success: true,
|
|
120
120
|
error: '',
|
|
121
121
|
});
|
|
122
122
|
};
|
|
123
|
-
|
|
123
|
+
|
|
124
124
|
img.onerror = () => {
|
|
125
125
|
URL.revokeObjectURL(objectUrl);
|
|
126
126
|
resolve({
|
|
@@ -128,7 +128,7 @@ export const uploadImageFromUrlHelper = async (
|
|
|
128
128
|
error: formatMessage(messages.imageLoadError),
|
|
129
129
|
});
|
|
130
130
|
};
|
|
131
|
-
|
|
131
|
+
|
|
132
132
|
img.src = objectUrl;
|
|
133
133
|
});
|
|
134
134
|
} catch (error) {
|
|
@@ -138,4 +138,3 @@ export const uploadImageFromUrlHelper = async (
|
|
|
138
138
|
};
|
|
139
139
|
}
|
|
140
140
|
};
|
|
141
|
-
|
|
@@ -1376,4 +1376,228 @@ describe("validateCarouselCards", () => {
|
|
|
1376
1376
|
expect(result.isValid).toBe(true);
|
|
1377
1377
|
});
|
|
1378
1378
|
});
|
|
1379
|
+
|
|
1380
|
+
describe('extractContent BEE Editor Logic (L404-L412)', () => {
|
|
1381
|
+
it('extracts content from beeHtml as string', () => {
|
|
1382
|
+
const platformData = {
|
|
1383
|
+
title: 'Test Title',
|
|
1384
|
+
isBEEeditor: true,
|
|
1385
|
+
beeHtml: '<p>BEE HTML Content</p>',
|
|
1386
|
+
ctas: [
|
|
1387
|
+
{ text: 'Click Here', actionLink: 'https://example.com' },
|
|
1388
|
+
],
|
|
1389
|
+
};
|
|
1390
|
+
|
|
1391
|
+
const result = extractContent(platformData);
|
|
1392
|
+
|
|
1393
|
+
expect(result).toContain(platformData.title);
|
|
1394
|
+
expect(result).toContain(platformData.beeHtml);
|
|
1395
|
+
expect(result).toContain(platformData.ctas[0].text);
|
|
1396
|
+
});
|
|
1397
|
+
|
|
1398
|
+
it('extracts content from beeHtml as object with value property', () => {
|
|
1399
|
+
const platformData = {
|
|
1400
|
+
title: 'Test Title',
|
|
1401
|
+
isBEEeditor: true,
|
|
1402
|
+
beeHtml: { value: '<p>BEE HTML from Object</p>' },
|
|
1403
|
+
ctas: [
|
|
1404
|
+
{ text: 'Button Text' },
|
|
1405
|
+
],
|
|
1406
|
+
};
|
|
1407
|
+
|
|
1408
|
+
const result = extractContent(platformData);
|
|
1409
|
+
|
|
1410
|
+
expect(result).toContain(platformData.title);
|
|
1411
|
+
expect(result).toContain(platformData.beeHtml.value);
|
|
1412
|
+
expect(result).toContain(platformData.ctas[0].text);
|
|
1413
|
+
});
|
|
1414
|
+
|
|
1415
|
+
it('handles beeHtml as object without value property', () => {
|
|
1416
|
+
const platformData = {
|
|
1417
|
+
title: 'Test Title',
|
|
1418
|
+
isBEEeditor: true,
|
|
1419
|
+
beeHtml: { someOtherProperty: 'data' },
|
|
1420
|
+
ctas: [],
|
|
1421
|
+
};
|
|
1422
|
+
|
|
1423
|
+
const result = extractContent(platformData);
|
|
1424
|
+
|
|
1425
|
+
// Should extract title and empty beeHtml (since value is undefined)
|
|
1426
|
+
expect(result).toContain('Test Title');
|
|
1427
|
+
expect(result).not.toContain('someOtherProperty');
|
|
1428
|
+
});
|
|
1429
|
+
|
|
1430
|
+
it('extracts ctas with text property', () => {
|
|
1431
|
+
const platformData = {
|
|
1432
|
+
title: 'Title',
|
|
1433
|
+
isBEEeditor: true,
|
|
1434
|
+
beeHtml: '<p>Content</p>',
|
|
1435
|
+
ctas: [
|
|
1436
|
+
{ text: 'CTA Text 1' },
|
|
1437
|
+
{ text: 'CTA Text 2' },
|
|
1438
|
+
],
|
|
1439
|
+
};
|
|
1440
|
+
|
|
1441
|
+
const result = extractContent(platformData);
|
|
1442
|
+
|
|
1443
|
+
expect(result).toContain('CTA Text 1');
|
|
1444
|
+
expect(result).toContain('CTA Text 2');
|
|
1445
|
+
});
|
|
1446
|
+
|
|
1447
|
+
it('extracts ctas with actionLink when text is missing', () => {
|
|
1448
|
+
const platformData = {
|
|
1449
|
+
title: 'Title',
|
|
1450
|
+
isBEEeditor: true,
|
|
1451
|
+
beeHtml: '<p>Content</p>',
|
|
1452
|
+
ctas: [
|
|
1453
|
+
{ actionLink: 'https://link1.com' },
|
|
1454
|
+
{ actionLink: 'https://link2.com' },
|
|
1455
|
+
],
|
|
1456
|
+
};
|
|
1457
|
+
|
|
1458
|
+
const result = extractContent(platformData);
|
|
1459
|
+
|
|
1460
|
+
expect(result).toContain('https://link1.com');
|
|
1461
|
+
expect(result).toContain('https://link2.com');
|
|
1462
|
+
});
|
|
1463
|
+
|
|
1464
|
+
it('filters out falsy values with filter(Boolean)', () => {
|
|
1465
|
+
const platformData = {
|
|
1466
|
+
title: '',
|
|
1467
|
+
isBEEeditor: true,
|
|
1468
|
+
beeHtml: null,
|
|
1469
|
+
ctas: [
|
|
1470
|
+
{ text: null, actionLink: null },
|
|
1471
|
+
{ text: 'Valid Text' },
|
|
1472
|
+
],
|
|
1473
|
+
};
|
|
1474
|
+
|
|
1475
|
+
const result = extractContent(platformData);
|
|
1476
|
+
|
|
1477
|
+
// Should only contain 'Valid Text' after filtering
|
|
1478
|
+
expect(result).toBe('Valid Text');
|
|
1479
|
+
});
|
|
1480
|
+
|
|
1481
|
+
it('handles empty ctas array', () => {
|
|
1482
|
+
const platformData = {
|
|
1483
|
+
title: 'Title Only',
|
|
1484
|
+
isBEEeditor: true,
|
|
1485
|
+
beeHtml: '<p>BEE Content</p>',
|
|
1486
|
+
ctas: [],
|
|
1487
|
+
};
|
|
1488
|
+
|
|
1489
|
+
const result = extractContent(platformData);
|
|
1490
|
+
|
|
1491
|
+
expect(result).toContain('Title Only');
|
|
1492
|
+
expect(result).toContain('<p>BEE Content</p>');
|
|
1493
|
+
});
|
|
1494
|
+
|
|
1495
|
+
it('handles undefined ctas', () => {
|
|
1496
|
+
const platformData = {
|
|
1497
|
+
title: 'Title',
|
|
1498
|
+
isBEEeditor: true,
|
|
1499
|
+
beeHtml: '<p>Content</p>',
|
|
1500
|
+
ctas: undefined,
|
|
1501
|
+
};
|
|
1502
|
+
|
|
1503
|
+
const result = extractContent(platformData);
|
|
1504
|
+
|
|
1505
|
+
expect(result).toContain('Title');
|
|
1506
|
+
expect(result).toContain('<p>Content</p>');
|
|
1507
|
+
});
|
|
1508
|
+
|
|
1509
|
+
it('joins all content with spaces', () => {
|
|
1510
|
+
const platformData = {
|
|
1511
|
+
title: 'Title',
|
|
1512
|
+
isBEEeditor: true,
|
|
1513
|
+
beeHtml: 'HTML',
|
|
1514
|
+
ctas: [
|
|
1515
|
+
{ text: 'CTA1' },
|
|
1516
|
+
{ text: 'CTA2' },
|
|
1517
|
+
],
|
|
1518
|
+
};
|
|
1519
|
+
|
|
1520
|
+
const result = extractContent(platformData);
|
|
1521
|
+
|
|
1522
|
+
expect(result).toBe('Title HTML CTA1 CTA2');
|
|
1523
|
+
});
|
|
1524
|
+
|
|
1525
|
+
it('falls back to regular content when not BEE editor', () => {
|
|
1526
|
+
const platformData = {
|
|
1527
|
+
title: 'Title',
|
|
1528
|
+
message: 'Message',
|
|
1529
|
+
isBEEeditor: false,
|
|
1530
|
+
beeHtml: '<p>Should be ignored</p>',
|
|
1531
|
+
ctas: [
|
|
1532
|
+
{ text: 'CTA' },
|
|
1533
|
+
],
|
|
1534
|
+
};
|
|
1535
|
+
|
|
1536
|
+
const result = extractContent(platformData);
|
|
1537
|
+
|
|
1538
|
+
expect(result).toContain('Title');
|
|
1539
|
+
expect(result).toContain('Message');
|
|
1540
|
+
expect(result).toContain('CTA');
|
|
1541
|
+
expect(result).not.toContain('<p>Should be ignored</p>');
|
|
1542
|
+
});
|
|
1543
|
+
|
|
1544
|
+
it('handles null beeHtml', () => {
|
|
1545
|
+
const platformData = {
|
|
1546
|
+
title: 'Title',
|
|
1547
|
+
isBEEeditor: true,
|
|
1548
|
+
beeHtml: null,
|
|
1549
|
+
ctas: [{ text: 'CTA' }],
|
|
1550
|
+
};
|
|
1551
|
+
|
|
1552
|
+
const result = extractContent(platformData);
|
|
1553
|
+
|
|
1554
|
+
expect(result).toContain('Title');
|
|
1555
|
+
expect(result).toContain('CTA');
|
|
1556
|
+
});
|
|
1557
|
+
|
|
1558
|
+
it('handles undefined beeHtml', () => {
|
|
1559
|
+
const platformData = {
|
|
1560
|
+
title: 'Title',
|
|
1561
|
+
isBEEeditor: true,
|
|
1562
|
+
beeHtml: undefined,
|
|
1563
|
+
ctas: [],
|
|
1564
|
+
};
|
|
1565
|
+
|
|
1566
|
+
const result = extractContent(platformData);
|
|
1567
|
+
|
|
1568
|
+
expect(result).toBe('Title');
|
|
1569
|
+
});
|
|
1570
|
+
|
|
1571
|
+
it('handles complex ctas with both text and actionLink', () => {
|
|
1572
|
+
const platformData = {
|
|
1573
|
+
title: 'Title',
|
|
1574
|
+
isBEEeditor: true,
|
|
1575
|
+
beeHtml: 'Content',
|
|
1576
|
+
ctas: [
|
|
1577
|
+
{ text: 'CTA Text', actionLink: 'https://example.com' },
|
|
1578
|
+
],
|
|
1579
|
+
};
|
|
1580
|
+
|
|
1581
|
+
const result = extractContent(platformData);
|
|
1582
|
+
|
|
1583
|
+
// Should prefer text over actionLink
|
|
1584
|
+
expect(result).toContain('CTA Text');
|
|
1585
|
+
expect(result).toContain('Title');
|
|
1586
|
+
expect(result).toContain('Content');
|
|
1587
|
+
});
|
|
1588
|
+
|
|
1589
|
+
it('handles empty string beeHtml', () => {
|
|
1590
|
+
const platformData = {
|
|
1591
|
+
title: 'Title',
|
|
1592
|
+
isBEEeditor: true,
|
|
1593
|
+
beeHtml: '',
|
|
1594
|
+
ctas: [{ text: 'CTA' }],
|
|
1595
|
+
};
|
|
1596
|
+
|
|
1597
|
+
const result = extractContent(platformData);
|
|
1598
|
+
|
|
1599
|
+
expect(result).toBe('Title CTA');
|
|
1600
|
+
});
|
|
1601
|
+
});
|
|
1379
1602
|
});
|
|
1603
|
+
|