@capillarytech/creatives-library 8.0.260 → 8.0.261
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/tests/commonUtil.test.js +224 -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 +457 -72
- package/v2Components/ErrorInfoNote/messages.js +36 -6
- package/v2Components/ErrorInfoNote/style.scss +282 -6
- package/v2Components/FormBuilder/tests/index.test.js +13 -4
- package/v2Components/HtmlEditor/HTMLEditor.js +547 -94
- package/v2Components/HtmlEditor/__tests__/HTMLEditor.apiErrors.test.js +874 -0
- package/v2Components/HtmlEditor/__tests__/HTMLEditor.test.js +1883 -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 -0
- package/v2Components/HtmlEditor/components/EditorToolbar/index.js +4 -4
- 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 +3 -6
- 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 +1 -0
- package/v2Components/HtmlEditor/components/ValidationErrorDisplay/index.js +49 -31
- package/v2Components/HtmlEditor/components/ValidationPanel/_validationPanel.scss +50 -34
- package/v2Components/HtmlEditor/components/ValidationPanel/constants.js +6 -0
- package/v2Components/HtmlEditor/components/ValidationPanel/index.js +70 -41
- package/v2Components/HtmlEditor/components/ValidationTabs/_validationTabs.scss +254 -0
- package/v2Components/HtmlEditor/components/ValidationTabs/index.js +364 -0
- package/v2Components/HtmlEditor/components/ValidationTabs/messages.js +51 -0
- package/v2Components/HtmlEditor/constants.js +42 -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 +189 -53
- package/v2Components/HtmlEditor/index.js +1 -1
- package/v2Components/HtmlEditor/messages.js +92 -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 +134 -102
- package/v2Components/HtmlEditor/utils/properSyntaxHighlighting.js +23 -25
- package/v2Components/HtmlEditor/utils/validationAdapter.js +66 -41
- package/v2Components/HtmlEditor/utils/validationConstants.js +40 -0
- package/v2Components/MobilePushPreviewV2/index.js +32 -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 +163 -13
- package/v2Containers/CreativesContainer/SlideBoxHeader.js +2 -1
- package/v2Containers/CreativesContainer/constants.js +1 -0
- package/v2Containers/CreativesContainer/index.js +239 -46
- 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 +106 -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 +1285 -0
- package/v2Containers/EmailWrapper/components/EmailWrapperView.js +211 -21
- package/v2Containers/EmailWrapper/components/HTMLEditorTesting.js +40 -74
- package/v2Containers/EmailWrapper/components/__tests__/EmailHTMLEditor.test.js +2487 -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 +629 -77
- package/v2Containers/EmailWrapper/index.js +103 -23
- package/v2Containers/EmailWrapper/messages.js +65 -1
- package/v2Containers/EmailWrapper/tests/useEmailWrapper.edgeCases.test.js +956 -0
- package/v2Containers/EmailWrapper/tests/useEmailWrapper.test.js +594 -77
- 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 -359
- 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/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/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,
|
|
@@ -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
|
+
|
|
@@ -101,7 +101,6 @@ export async function transformTemplateConfigWithMediaDetails(templateConfig, ch
|
|
|
101
101
|
const contentType = detectMediaContentType(templateConfig);
|
|
102
102
|
|
|
103
103
|
if (contentType === MEDIA_CONTENT_TYPE.NONE) {
|
|
104
|
-
console.log('No media content detected, returning original config');
|
|
105
104
|
return templateConfig;
|
|
106
105
|
}
|
|
107
106
|
|
|
@@ -109,24 +108,19 @@ export async function transformTemplateConfigWithMediaDetails(templateConfig, ch
|
|
|
109
108
|
let transformDirection;
|
|
110
109
|
if (forceDirection) {
|
|
111
110
|
transformDirection = forceDirection;
|
|
112
|
-
console.log(`Forced transformation direction: ${transformDirection}`);
|
|
113
111
|
} else {
|
|
114
112
|
// Auto-detect based on content
|
|
115
113
|
if (contentType === MEDIA_CONTENT_TYPE.URLS) {
|
|
116
114
|
transformDirection = TRANSFORM_DIRECTION.FORWARD; // URLs → tags
|
|
117
|
-
console.log('Detected URLs, applying forward transformation (URLs → tags)');
|
|
118
115
|
} else if (contentType === MEDIA_CONTENT_TYPE.TAGS) {
|
|
119
116
|
transformDirection = TRANSFORM_DIRECTION.REVERSE; // tags → URLs
|
|
120
|
-
console.log('Detected media_content tags, applying reverse transformation (tags → URLs)');
|
|
121
117
|
} else if (contentType === MEDIA_CONTENT_TYPE.MIXED) {
|
|
122
118
|
transformDirection = TRANSFORM_DIRECTION.SMART; // Handle mixed content intelligently
|
|
123
|
-
console.log('Detected mixed content, applying smart transformation');
|
|
124
119
|
}
|
|
125
120
|
}
|
|
126
121
|
|
|
127
122
|
const newConfig = cloneDeep(templateConfig);
|
|
128
123
|
try {
|
|
129
|
-
console.log(`Fetching media details for ${transformDirection} transformation, ID: ${mediaDetailsId}`);
|
|
130
124
|
|
|
131
125
|
// Fetch media details mapping from API
|
|
132
126
|
const response = await getMediaDetails({ id: mediaDetailsId })
|
|
@@ -145,7 +139,6 @@ export async function transformTemplateConfigWithMediaDetails(templateConfig, ch
|
|
|
145
139
|
idToUrlMapping[mediaId] = url;
|
|
146
140
|
});
|
|
147
141
|
|
|
148
|
-
console.log(`Processing ${Object.keys(urlToIdMapping).length} media mappings`);
|
|
149
142
|
|
|
150
143
|
// Apply transformations based on direction
|
|
151
144
|
newConfig?.cards?.forEach(card => {
|
|
@@ -157,7 +150,6 @@ export async function transformTemplateConfigWithMediaDetails(templateConfig, ch
|
|
|
157
150
|
// Forward transformation: URL → tag
|
|
158
151
|
const mediaDetailId = urlToIdMapping[currentUrl];
|
|
159
152
|
if (mediaDetailId) {
|
|
160
|
-
console.log(`Forward: ${currentUrl} → {{media_content(${mediaDetailId})}}`);
|
|
161
153
|
card.media.url = `{{media_content(${mediaDetailId})}}`;
|
|
162
154
|
} else {
|
|
163
155
|
console.warn(`No media detail ID found for URL: ${currentUrl}`);
|
|
@@ -169,7 +161,6 @@ export async function transformTemplateConfigWithMediaDetails(templateConfig, ch
|
|
|
169
161
|
if (mediaId) {
|
|
170
162
|
const actualUrl = idToUrlMapping[mediaId];
|
|
171
163
|
if (actualUrl) {
|
|
172
|
-
console.log(`Reverse: {{media_content(${mediaId})}} → ${actualUrl}`);
|
|
173
164
|
card.media.url = actualUrl;
|
|
174
165
|
} else {
|
|
175
166
|
console.warn(`No URL found for media ID: ${mediaId}`);
|
|
@@ -185,7 +176,6 @@ export async function transformTemplateConfigWithMediaDetails(templateConfig, ch
|
|
|
185
176
|
console.error('Failed to fetch media details for transformation:', error);
|
|
186
177
|
|
|
187
178
|
// Fallback: try to extract/convert what we can without API
|
|
188
|
-
console.log('Falling back to URL extraction method');
|
|
189
179
|
newConfig.cards.forEach(card => {
|
|
190
180
|
if (card.media && card.media.url) {
|
|
191
181
|
if (transformDirection === TRANSFORM_DIRECTION.FORWARD) {
|