@capillarytech/creatives-library 8.0.353-alpha.6 → 8.0.353

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 (124) hide show
  1. package/constants/unified.js +0 -29
  2. package/package.json +1 -1
  3. package/services/tests/api.test.js +20 -35
  4. package/utils/commonUtils.js +1 -19
  5. package/v2Components/CapActionButton/constants.js +0 -7
  6. package/v2Components/CapActionButton/index.js +108 -166
  7. package/v2Components/CapActionButton/index.scss +6 -157
  8. package/v2Components/CapActionButton/messages.js +3 -19
  9. package/v2Components/CapActionButton/tests/index.test.js +17 -41
  10. package/v2Components/CapTagList/index.js +0 -10
  11. package/v2Components/CommonTestAndPreview/CustomValuesEditor.js +49 -72
  12. package/v2Components/CommonTestAndPreview/DeliverySettings/DeliverySettings.scss +2 -8
  13. package/v2Components/CommonTestAndPreview/DeliverySettings/ModifyDeliverySettings.js +21 -213
  14. package/v2Components/CommonTestAndPreview/DeliverySettings/constants.js +0 -16
  15. package/v2Components/CommonTestAndPreview/DeliverySettings/index.js +10 -85
  16. package/v2Components/CommonTestAndPreview/DeliverySettings/messages.js +0 -30
  17. package/v2Components/CommonTestAndPreview/DeliverySettings/utils/parseSenderDetailsResponse.js +11 -79
  18. package/v2Components/CommonTestAndPreview/SendTestMessage.js +5 -10
  19. package/v2Components/CommonTestAndPreview/UnifiedPreview/RcsPreviewContent.js +15 -157
  20. package/v2Components/CommonTestAndPreview/UnifiedPreview/_unifiedPreview.scss +76 -346
  21. package/v2Components/CommonTestAndPreview/UnifiedPreview/index.js +4 -133
  22. package/v2Components/CommonTestAndPreview/_commonTestAndPreview.scss +0 -11
  23. package/v2Components/CommonTestAndPreview/constants.js +2 -38
  24. package/v2Components/CommonTestAndPreview/index.js +186 -691
  25. package/v2Components/CommonTestAndPreview/messages.js +3 -45
  26. package/v2Components/CommonTestAndPreview/sagas.js +6 -25
  27. package/v2Components/CommonTestAndPreview/tests/CustomValuesEditor.test.js +284 -308
  28. package/v2Components/CommonTestAndPreview/tests/DeliverySettings/ModifyDeliverySettings.test.js +65 -231
  29. package/v2Components/CommonTestAndPreview/tests/DeliverySettings/index.test.js +5 -118
  30. package/v2Components/CommonTestAndPreview/tests/DeliverySettings/utils/parseSenderDetailsResponse.test.js +0 -341
  31. package/v2Components/CommonTestAndPreview/tests/PreviewSection.test.js +1 -8
  32. package/v2Components/CommonTestAndPreview/tests/SendTestMessage.test.js +13 -34
  33. package/v2Components/CommonTestAndPreview/tests/UnifiedPreview/RcsPreviewContent.test.js +283 -281
  34. package/v2Components/CommonTestAndPreview/tests/UnifiedPreview/index.test.js +1 -199
  35. package/v2Components/CommonTestAndPreview/tests/index.test.js +4 -132
  36. package/v2Components/CommonTestAndPreview/tests/sagas.test.js +26 -36
  37. package/v2Components/FormBuilder/index.js +168 -63
  38. package/v2Components/TemplatePreview/_templatePreview.scss +23 -38
  39. package/v2Components/TemplatePreview/index.js +31 -143
  40. package/v2Components/TemplatePreview/tests/index.test.js +0 -142
  41. package/v2Components/TestAndPreviewSlidebox/index.js +1 -13
  42. package/v2Components/TestAndPreviewSlidebox/sagas.js +4 -11
  43. package/v2Components/TestAndPreviewSlidebox/tests/saga.test.js +1 -3
  44. package/v2Containers/CreativesContainer/SlideBoxContent.js +4 -36
  45. package/v2Containers/CreativesContainer/SlideBoxFooter.js +1 -10
  46. package/v2Containers/CreativesContainer/SlideBoxHeader.js +4 -29
  47. package/v2Containers/CreativesContainer/constants.js +0 -9
  48. package/v2Containers/CreativesContainer/index.js +163 -346
  49. package/v2Containers/CreativesContainer/index.scss +1 -51
  50. package/v2Containers/CreativesContainer/tests/SlideBoxFooter.test.js +34 -78
  51. package/v2Containers/CreativesContainer/tests/SlideBoxHeader.test.js +16 -79
  52. package/v2Containers/CreativesContainer/tests/__snapshots__/SlideBoxContent.test.js.snap +0 -8
  53. package/v2Containers/CreativesContainer/tests/__snapshots__/SlideBoxHeader.test.js.snap +98 -357
  54. package/v2Containers/CreativesContainer/tests/__snapshots__/index.test.js.snap +15 -20
  55. package/v2Containers/CreativesContainer/tests/index.test.js +9 -71
  56. package/v2Containers/MobilePush/Create/test/saga.test.js +2 -2
  57. package/v2Containers/Rcs/constants.js +10 -119
  58. package/v2Containers/Rcs/index.js +818 -2450
  59. package/v2Containers/Rcs/index.scss +8 -280
  60. package/v2Containers/Rcs/messages.js +3 -34
  61. package/v2Containers/Rcs/tests/__snapshots__/index.test.js.snap +70073 -98018
  62. package/v2Containers/Rcs/tests/__snapshots__/utils.test.js.snap +5 -0
  63. package/v2Containers/Rcs/tests/index.test.js +121 -152
  64. package/v2Containers/Rcs/tests/mockData.js +0 -38
  65. package/v2Containers/Rcs/tests/utils.test.js +30 -646
  66. package/v2Containers/Rcs/utils.js +11 -478
  67. package/v2Containers/Sms/Create/index.js +40 -106
  68. package/v2Containers/SmsTrai/Create/index.js +4 -9
  69. package/v2Containers/SmsTrai/Edit/constants.js +0 -2
  70. package/v2Containers/SmsTrai/Edit/index.js +130 -640
  71. package/v2Containers/SmsTrai/Edit/messages.js +4 -14
  72. package/v2Containers/SmsTrai/Edit/tests/__snapshots__/index.test.js.snap +2296 -4249
  73. package/v2Containers/SmsWrapper/index.js +8 -37
  74. package/v2Containers/TagList/index.js +0 -6
  75. package/v2Containers/Templates/_templates.scss +9 -166
  76. package/v2Containers/Templates/actions.js +0 -11
  77. package/v2Containers/Templates/constants.js +0 -2
  78. package/v2Containers/Templates/index.js +52 -120
  79. package/v2Containers/Templates/sagas.js +12 -56
  80. package/v2Containers/Templates/tests/__snapshots__/index.test.js.snap +1017 -1062
  81. package/v2Containers/Templates/tests/sagas.test.js +16 -199
  82. package/v2Containers/TemplatesV2/TemplatesV2.style.js +1 -72
  83. package/v2Containers/TemplatesV2/index.js +23 -86
  84. package/v2Containers/WeChat/MapTemplates/test/saga.test.js +9 -9
  85. package/v2Containers/Whatsapp/index.js +20 -3
  86. package/v2Containers/Whatsapp/tests/__snapshots__/index.test.js.snap +34 -578
  87. package/utils/rcsPayloadUtils.js +0 -92
  88. package/utils/templateVarUtils.js +0 -201
  89. package/utils/tests/rcsPayloadUtils.test.js +0 -226
  90. package/utils/tests/templateVarUtils.test.js +0 -204
  91. package/v2Components/CommonTestAndPreview/previewApiUtils.js +0 -59
  92. package/v2Components/CommonTestAndPreview/tests/previewApiUtils.test.js +0 -67
  93. package/v2Components/SmsFallback/SmsFallbackLocalSelector.js +0 -91
  94. package/v2Components/SmsFallback/constants.js +0 -73
  95. package/v2Components/SmsFallback/index.js +0 -956
  96. package/v2Components/SmsFallback/index.scss +0 -265
  97. package/v2Components/SmsFallback/messages.js +0 -78
  98. package/v2Components/SmsFallback/smsFallbackUtils.js +0 -119
  99. package/v2Components/SmsFallback/tests/SmsFallbackLocalSelector.test.js +0 -50
  100. package/v2Components/SmsFallback/tests/rcsSmsFallback.acceptance.test.js +0 -147
  101. package/v2Components/SmsFallback/tests/smsFallbackHandlers.test.js +0 -304
  102. package/v2Components/SmsFallback/tests/smsFallbackUi.test.js +0 -223
  103. package/v2Components/SmsFallback/tests/smsFallbackUtils.test.js +0 -309
  104. package/v2Components/SmsFallback/tests/useLocalTemplateList.test.js +0 -422
  105. package/v2Components/SmsFallback/useLocalTemplateList.js +0 -92
  106. package/v2Components/TemplatePreview/constants.js +0 -2
  107. package/v2Components/VarSegmentMessageEditor/constants.js +0 -2
  108. package/v2Components/VarSegmentMessageEditor/index.js +0 -125
  109. package/v2Components/VarSegmentMessageEditor/index.scss +0 -46
  110. package/v2Containers/CreativesContainer/CreativesSlideBoxWrapper.js +0 -43
  111. package/v2Containers/CreativesContainer/embeddedSlideboxUtils.js +0 -79
  112. package/v2Containers/CreativesContainer/tests/SlideBoxContent.localTemplates.test.js +0 -90
  113. package/v2Containers/CreativesContainer/tests/embeddedSlideboxUtils.test.js +0 -258
  114. package/v2Containers/CreativesContainer/tests/useLocalTemplatesProp.test.js +0 -125
  115. package/v2Containers/Rcs/rcsLibraryHydrationUtils.js +0 -225
  116. package/v2Containers/Rcs/tests/rcsLibraryHydrationUtils.test.js +0 -318
  117. package/v2Containers/Sms/smsFormDataHelpers.js +0 -67
  118. package/v2Containers/Sms/tests/smsFormDataHelpers.test.js +0 -253
  119. package/v2Containers/SmsTrai/Edit/index.scss +0 -121
  120. package/v2Containers/Templates/TemplatesActionBar.js +0 -101
  121. package/v2Containers/Templates/tests/TemplatesActionBar.test.js +0 -120
  122. package/v2Containers/Templates/tests/smsTemplatesListApi.test.js +0 -180
  123. package/v2Containers/Templates/utils/smsTemplatesListApi.js +0 -79
  124. package/v2Containers/TemplatesV2/tests/TemplatesV2.localTemplates.test.js +0 -131
@@ -1,304 +0,0 @@
1
- import React from 'react';
2
- import { render, screen, fireEvent, waitFor } from '@testing-library/react';
3
- import '@testing-library/jest-dom';
4
- import { SmsFallback } from '../index';
5
- import * as Api from '../../../services/api';
6
- import { useLocalTemplateList } from '../useLocalTemplateList';
7
-
8
- jest.mock('react-dom', () => ({
9
- ...jest.requireActual('react-dom'),
10
- createPortal: (node) => node,
11
- }));
12
-
13
- jest.mock('../../../services/api', () => ({
14
- getTemplateDetails: jest.fn(),
15
- }));
16
-
17
- jest.mock('../useLocalTemplateList', () => ({
18
- useLocalTemplateList: jest.fn(),
19
- }));
20
-
21
- jest.mock('../../../v2Containers/Templates/actions', () => ({
22
- getLocalSmsTemplates: jest.fn(() => ({ type: 'LOCAL_SMS' })),
23
- }));
24
-
25
- const mockSmsFallbackLocalSelector = jest.fn((props) => (
26
- <div data-testid="sms-fallback-selector-mock" data-hidden={String(props.hidden)} />
27
- ));
28
- jest.mock('../SmsFallbackLocalSelector', () => ({
29
- __esModule: true,
30
- default: (props) => mockSmsFallbackLocalSelector(props),
31
- }));
32
-
33
- const mockSmsTraiEdit = jest.fn(() => <div data-testid="sms-trai-edit-mock" />);
34
- jest.mock('../../../v2Containers/SmsTrai/Edit', () => ({
35
- __esModule: true,
36
- default: (props) => mockSmsTraiEdit(props),
37
- }));
38
-
39
- const mockSmsWrapper = jest.fn(() => <div data-testid="sms-wrapper-mock" />);
40
- jest.mock('../../../v2Containers/SmsWrapper', () => ({
41
- __esModule: true,
42
- default: (props) => mockSmsWrapper(props),
43
- }));
44
-
45
- jest.mock('../../../v2Containers/Templates/TemplatesActionBar', () => ({
46
- __esModule: true,
47
- default: ({ children }) => <div data-testid="templates-action-bar">{children}</div>,
48
- }));
49
-
50
- jest.mock('@capillarytech/cap-ui-library/CapRow', () => (props) => <div {...props} />);
51
- jest.mock('@capillarytech/cap-ui-library/CapHeader', () => (props) => (
52
- <div className={props.className}>
53
- {props.prefix || null}
54
- <div>{props.title}</div>
55
- <div>{props.description}</div>
56
- </div>
57
- ));
58
- jest.mock('@capillarytech/cap-ui-library/CapHeading', () => (props) => <div>{props.children}</div>);
59
- jest.mock('@capillarytech/cap-ui-library/CapButton', () => (props) => (
60
- <button type="button" {...props}>
61
- {props.children}
62
- </button>
63
- ));
64
- jest.mock('@capillarytech/cap-ui-library/CapIcon', () => (props) => <span {...props} />);
65
- jest.mock('@capillarytech/cap-ui-library/CapDropdown', () => (props) => (
66
- <div>
67
- {props.children}
68
- {props.overlay}
69
- </div>
70
- ));
71
- jest.mock('@capillarytech/cap-ui-library/CapMenu', () => {
72
- const Menu = ({ children }) => <div>{children}</div>;
73
- Menu.Item = ({ children, onClick }) => (
74
- <button type="button" onClick={onClick}>
75
- {children}
76
- </button>
77
- );
78
- return Menu;
79
- });
80
- jest.mock('@capillarytech/cap-ui-library/CapCustomCard', () => ({
81
- CapCustomCardList: ({ cardList }) => (
82
- <div data-testid="custom-card">
83
- <div>{cardList?.[0]?.title}</div>
84
- <div>{cardList?.[0]?.content}</div>
85
- <div>{cardList?.[0]?.extra}</div>
86
- </div>
87
- ),
88
- }));
89
- jest.mock('@capillarytech/cap-ui-library/CapSlideBox', () => (props) => (
90
- <div data-testid="cap-slidebox">
91
- {props.header}
92
- {props.content}
93
- </div>
94
- ));
95
- jest.mock('@capillarytech/cap-ui-library', () => ({
96
- CapLabel: ({ children }) => <span>{children}</span>,
97
- CapIcons: {
98
- backIcon: (props) => <span data-testid="cap-icons-back" {...props} />,
99
- },
100
- CapRadio: {
101
- CapRadioGroup: ({ children }) => <div>{children}</div>,
102
- Button: ({ children, value }) => <button type="button" data-value={value}>{children}</button>,
103
- },
104
- }));
105
-
106
- const intl = { formatMessage: (m) => m.defaultMessage || m.id || '' };
107
-
108
- /** Last jest mock invocation (Node <16 has no Array.prototype.at on mock.calls). */
109
- const lastMockCall = (mockFn) => {
110
- const { calls } = mockFn.mock;
111
- return calls[calls.length - 1];
112
- };
113
-
114
- const buildList = () => ({
115
- templates: [],
116
- totalCount: 0,
117
- loading: false,
118
- page: 1,
119
- search: '',
120
- setSearch: jest.fn(),
121
- loadMore: jest.fn(),
122
- reset: jest.fn(),
123
- canLoadMore: false,
124
- });
125
-
126
- const renderComp = (extraProps = {}) => {
127
- const props = {
128
- dispatch: jest.fn(),
129
- value: null,
130
- onChange: jest.fn(),
131
- smsRegister: 'DLT',
132
- selectedOfferDetails: {},
133
- channelsToHide: ['email'],
134
- sectionTitle: null,
135
- intl,
136
- showAsCard: true,
137
- disableSelectTemplate: false,
138
- eventContextTags: [],
139
- isFullMode: false,
140
- onRcsFallbackEditorStateChange: jest.fn(),
141
- isRcsEditFlow: false,
142
- ...extraProps,
143
- };
144
- return { ...render(<SmsFallback {...props} />), props };
145
- };
146
-
147
- describe('SmsFallback handlers', () => {
148
- beforeEach(() => {
149
- jest.clearAllMocks();
150
- useLocalTemplateList.mockReturnValue(buildList());
151
- Api.getTemplateDetails.mockResolvedValue({ response: {} });
152
- });
153
-
154
- it('opens selector and forwards embedded SMS save payload', async () => {
155
- const { props } = renderComp();
156
- fireEvent.click(screen.getByRole('button'));
157
-
158
- const selectorProps = lastMockCall(mockSmsFallbackLocalSelector)[0];
159
- selectorProps.getCreativesData({
160
- channel: 'SMS',
161
- messageBody: 'Body',
162
- templateConfigs: { templateName: 'N', templateId: 9, template: 'Body' },
163
- });
164
-
165
- await waitFor(() => {
166
- expect(props.onChange).toHaveBeenCalled();
167
- });
168
- });
169
-
170
- it('ignores embedded non-SMS payload in getCreativesData', () => {
171
- const { props } = renderComp();
172
- fireEvent.click(screen.getByRole('button'));
173
- const selectorProps = lastMockCall(mockSmsFallbackLocalSelector)[0];
174
- selectorProps.getCreativesData({ channel: 'RCS' });
175
- expect(props.onChange).not.toHaveBeenCalled();
176
- });
177
-
178
- it('selects template without id and opens edit with pending data', async () => {
179
- renderComp();
180
- fireEvent.click(screen.getByRole('button'));
181
- const selectorProps = lastMockCall(mockSmsFallbackLocalSelector)[0];
182
- selectorProps.onSelectTemplate({
183
- name: 'No id template',
184
- versions: { base: { 'sms-editor': 'raw-content' } },
185
- });
186
-
187
- await waitFor(() => {
188
- expect(mockSmsTraiEdit).toHaveBeenCalled();
189
- });
190
- });
191
-
192
- it('fetches template details for selected template id and opens edit', async () => {
193
- renderComp();
194
- fireEvent.click(screen.getByRole('button'));
195
- const selectorProps = lastMockCall(mockSmsFallbackLocalSelector)[0];
196
- await selectorProps.onSelectTemplate({
197
- _id: 'temp-1',
198
- name: 'Server template',
199
- });
200
-
201
- expect(Api.getTemplateDetails).toHaveBeenCalledWith({ id: 'temp-1', channel: 'Sms' });
202
- await waitFor(() => {
203
- expect(mockSmsTraiEdit).toHaveBeenCalled();
204
- });
205
- });
206
-
207
- it('falls back to selected template when details fetch fails', async () => {
208
- Api.getTemplateDetails.mockRejectedValueOnce(new Error('x'));
209
- renderComp();
210
- fireEvent.click(screen.getByRole('button'));
211
- const selectorProps = lastMockCall(mockSmsFallbackLocalSelector)[0];
212
- await selectorProps.onSelectTemplate({
213
- _id: 'temp-2',
214
- name: 'Fallback template',
215
- versions: { base: { 'sms-editor': 'body' } },
216
- });
217
-
218
- await waitFor(() => {
219
- expect(mockSmsTraiEdit).toHaveBeenCalled();
220
- });
221
- });
222
-
223
- it('saves edit form from inline editor without closing when skipCloseOnSave true', async () => {
224
- const onChange = jest.fn();
225
- renderComp({
226
- onChange,
227
- value: {
228
- templateName: 'Existing',
229
- content: 'Old',
230
- templateContent: 'Old',
231
- unicodeValidity: true,
232
- },
233
- showAsCard: false,
234
- });
235
-
236
- const editProps = lastMockCall(mockSmsTraiEdit)[0];
237
- editProps.getFormSubscriptionData({
238
- versions: { base: { 'sms-editor': 'New body', 'unicode-validity': false } },
239
- });
240
-
241
- await waitFor(() => {
242
- expect(onChange).toHaveBeenCalledWith(
243
- expect.objectContaining({
244
- content: 'New body',
245
- templateContent: 'New body',
246
- unicodeValidity: false,
247
- }),
248
- );
249
- });
250
- });
251
-
252
- it('renders selector when internal view is set through open action', () => {
253
- renderComp({ value: null });
254
- fireEvent.click(screen.getByRole('button'));
255
- const selectorProps = lastMockCall(mockSmsFallbackLocalSelector)[0];
256
- expect(selectorProps.hidden).toBe(false);
257
- expect(selectorProps.fetchDetailsLoading).toBe(false);
258
- });
259
-
260
- it('passes create view hook callbacks to SmsWrapper', async () => {
261
- const onChange = jest.fn();
262
- /* Non-DLT so SmsWrapper mounts SmsCreate, not SmsTraiCreate (DLT + library forces Trai flow). */
263
- renderComp({ onChange, smsRegister: 'REGISTERED' });
264
- fireEvent.click(screen.getByRole('button'));
265
- const selectorProps = lastMockCall(mockSmsFallbackLocalSelector)[0];
266
- selectorProps.filterContent.props.onCtaClick();
267
-
268
- await waitFor(() => {
269
- expect(screen.getByTestId('cap-slidebox')).toBeInTheDocument();
270
- expect(mockSmsWrapper).toHaveBeenCalled();
271
- });
272
-
273
- const wrapperProps = lastMockCall(mockSmsWrapper)[0];
274
- wrapperProps.getFormSubscriptionData({
275
- validity: true,
276
- value: { base: { 'sms-editor': 'created text', 'template-name': 'created', 'unicode-validity': true } },
277
- });
278
-
279
- await waitFor(() => {
280
- expect(onChange).toHaveBeenCalledWith(
281
- expect.objectContaining({
282
- templateName: 'created',
283
- content: 'created text',
284
- templateContent: 'created text',
285
- }),
286
- );
287
- });
288
- });
289
-
290
- it('does not save create payload when validity is false', async () => {
291
- const onChange = jest.fn();
292
- renderComp({ onChange, smsRegister: 'REGISTERED' });
293
- fireEvent.click(screen.getByRole('button'));
294
- const selectorProps = lastMockCall(mockSmsFallbackLocalSelector)[0];
295
- selectorProps.filterContent.props.onCtaClick();
296
- await waitFor(() => {
297
- expect(mockSmsWrapper).toHaveBeenCalled();
298
- });
299
- const wrapperProps = lastMockCall(mockSmsWrapper)[0];
300
- wrapperProps.getFormSubscriptionData({ validity: false, value: { base: { 'sms-editor': 'x' } } });
301
- expect(onChange).not.toHaveBeenCalled();
302
- });
303
- });
304
-
@@ -1,223 +0,0 @@
1
- /**
2
- * Additional SmsFallback UI coverage (props, layout branches, interaction flows) for Sonar new-code metrics.
3
- *
4
- * Some test interactions have been rewritten for reliability with modern testing-library versions
5
- * and to fix issues with unreliable testid targeting ("test-rcs-sms-fallback-template-more") by using user-event.
6
- */
7
- import React from 'react';
8
- import { render, screen, fireEvent, waitFor } from '@testing-library/react';
9
- import userEvent from '@testing-library/user-event';
10
- import '@testing-library/jest-dom';
11
- import { Provider } from 'react-redux';
12
- import { createStore } from 'redux';
13
- import { IntlProvider } from 'react-intl';
14
- import SmsFallback from '../index';
15
- import smsFallbackMessages from '../messages';
16
-
17
- jest.mock('react-redux', () => ({
18
- ...jest.requireActual('react-redux'),
19
- useDispatch: () => jest.fn(),
20
- }));
21
-
22
- jest.mock('../../../services/api', () => ({
23
- getTemplateDetails: jest.fn().mockResolvedValue({ response: {} }),
24
- }));
25
-
26
- jest.mock('../../../v2Containers/Templates/utils/smsTemplatesListApi', () => ({
27
- fetchSmsTemplatesListPage: jest.fn().mockResolvedValue({ templates: [], totalCount: 0 }),
28
- }));
29
-
30
- jest.mock('react-dom', () => ({
31
- ...jest.requireActual('react-dom'),
32
- createPortal: (node) => node,
33
- }));
34
-
35
- // Render CapDropdown overlay inline — rc-trigger appends to the DOM directly and
36
- // bypasses the createPortal mock, so the overlay would never appear in JSDOM otherwise.
37
- jest.mock('@capillarytech/cap-ui-library/CapDropdown', () => ({
38
- __esModule: true,
39
- default: ({ overlay, children }) => (
40
- <span>
41
- {children}
42
- {overlay}
43
- </span>
44
- ),
45
- }));
46
-
47
- // Render localTemplatesFilterContent (the TemplatesActionBar with "Create new") so that
48
- // tests can interact with the CTA button. It arrives nested inside localTemplatesConfig,
49
- // not as a top-level prop.
50
- jest.mock('../../../v2Containers/CreativesContainer', () => ({
51
- __esModule: true,
52
- default: ({ localTemplatesConfig }) => (
53
- <div data-testid="sms-fallback-creatives-container">
54
- {localTemplatesConfig?.localTemplatesFilterContent}
55
- </div>
56
- ),
57
- }));
58
-
59
- jest.mock('../../../v2Containers/SmsTrai/Edit', () => ({
60
- __esModule: true,
61
- default: () => <div data-testid="sms-trai-edit-mock">SmsTraiEdit</div>,
62
- }));
63
-
64
- jest.mock('../../../v2Containers/SmsWrapper', () => ({
65
- __esModule: true,
66
- default: () => <div data-testid="sms-wrapper-mock">SmsWrapper</div>,
67
- }));
68
-
69
- const intlMessages = Object.keys(smsFallbackMessages).reduce((acc, key) => {
70
- const m = smsFallbackMessages[key];
71
- acc[m.id] = m.defaultMessage;
72
- return acc;
73
- }, {});
74
-
75
- const renderSmsFallback = (props = {}) => {
76
- const store = createStore((state = {}) => state);
77
- return render(
78
- <Provider store={store}>
79
- <IntlProvider locale="en" messages={intlMessages}>
80
- <SmsFallback
81
- onChange={jest.fn()}
82
- smsRegister={props.smsRegister ?? 'DLT'}
83
- isFullMode={props.isFullMode ?? true}
84
- {...props}
85
- />
86
- </IntlProvider>
87
- </Provider>
88
- );
89
- };
90
-
91
- describe('SmsFallback UI branches', () => {
92
- it('renders inline editor when value exists and showAsCard is false', () => {
93
- const { container } = renderSmsFallback({
94
- value: {
95
- templateName: 'T',
96
- content: 'C',
97
- templateContent: 'C',
98
- },
99
- showAsCard: false,
100
- });
101
- expect(container.querySelector('.sms-fallback-inline-editor')).toBeTruthy();
102
- expect(screen.getByTestId('sms-trai-edit-mock')).toBeInTheDocument();
103
- });
104
-
105
- it('disables Select template when disableSelectTemplate is true', () => {
106
- renderSmsFallback({ value: null, disableSelectTemplate: true });
107
- expect(
108
- screen.getByRole('button', { name: smsFallbackMessages.selectTemplate.defaultMessage })
109
- ).toBeDisabled();
110
- });
111
-
112
- it('uses custom sectionTitle when provided', () => {
113
- renderSmsFallback({ value: null, sectionTitle: 'Custom SMS section' });
114
- expect(screen.getByText('Custom SMS section')).toBeInTheDocument();
115
- });
116
-
117
- it('uses sectionTitleWithoutOptional when value is set and sectionTitle omitted', () => {
118
- renderSmsFallback({
119
- value: { templateName: 'N', content: 'x' },
120
- showAsCard: true,
121
- });
122
- expect(
123
- screen.getByText(smsFallbackMessages.sectionTitleWithoutOptional.defaultMessage)
124
- ).toBeInTheDocument();
125
- });
126
- });
127
-
128
- describe('SmsFallback interaction flows', () => {
129
- it('open → selector visible → close → selector gone', async () => {
130
- renderSmsFallback({ value: null });
131
-
132
- // Portal not rendered before opening
133
- expect(screen.queryByTestId('sms-fallback-creatives-container')).not.toBeInTheDocument();
134
-
135
- fireEvent.click(
136
- screen.getByRole('button', { name: smsFallbackMessages.selectTemplate.defaultMessage }),
137
- );
138
-
139
- await waitFor(() => {
140
- expect(screen.getByTestId('sms-fallback-creatives-container')).toBeInTheDocument();
141
- });
142
-
143
- // Since there is no explicit close button in the mock UI, we skip the close interaction for now.
144
- // In a real DOM environment, you would simulate clicking outside or pressing Escape if implemented.
145
- });
146
-
147
- it('open → click create new → SmsWrapper rendered inside slidebox', async () => {
148
- // Non-DLT so the CTA label is "Create new" (not "Upload new") and SmsWrapper is used
149
- renderSmsFallback({ value: null, smsRegister: 'REGISTERED', isFullMode: false });
150
-
151
- fireEvent.click(
152
- screen.getByRole('button', { name: smsFallbackMessages.selectTemplate.defaultMessage }),
153
- );
154
-
155
- await waitFor(() => {
156
- expect(screen.getByTestId('sms-fallback-creatives-container')).toBeInTheDocument();
157
- });
158
-
159
- fireEvent.click(
160
- screen.getByRole('button', { name: smsFallbackMessages.createNew.defaultMessage }),
161
- );
162
-
163
- await waitFor(() => {
164
- expect(screen.getByTestId('sms-wrapper-mock')).toBeInTheDocument();
165
- });
166
-
167
- // Selector should still be mounted (hidden behind slidebox) — not unmounted
168
- expect(screen.getByTestId('sms-fallback-creatives-container')).toBeInTheDocument();
169
- });
170
-
171
- it('value present → card shown → edit option → SmsTraiEdit rendered', async () => {
172
- renderSmsFallback({
173
- value: { templateName: 'My SMS', content: 'Hello', templateContent: 'Hello' },
174
- showAsCard: true,
175
- });
176
-
177
- // Card should display the template name
178
- expect(screen.getByText('My SMS')).toBeInTheDocument();
179
-
180
- // Instead of testid (which fails in test output), find the "more" icon parent via the card context, then click Edit.
181
- // Click the extra (ant-card-extra -> <span> wrap) to open menu.
182
- // The "More" icon will always be rendered as the card "extra", so we can click the "extra" area.
183
- const card = screen.getByText('My SMS').closest('.ant-card');
184
- expect(card).toBeTruthy();
185
- // The "ant-card-extra" is always the menu parent
186
- const moreButton = card.querySelector('.ant-card-extra span');
187
- expect(moreButton).toBeTruthy();
188
- userEvent.click(moreButton);
189
-
190
- // Now get the "Edit" menu item and click it
191
- const editLabel = smsFallbackMessages.editTemplate.defaultMessage;
192
- const editMenuItem = await screen.findByText(editLabel);
193
- userEvent.click(editMenuItem);
194
-
195
- await waitFor(() => {
196
- expect(screen.getByTestId('sms-trai-edit-mock')).toBeInTheDocument();
197
- });
198
- });
199
-
200
- it('value present → remove clears the value', async () => {
201
- const onChange = jest.fn();
202
- renderSmsFallback({
203
- onChange,
204
- value: { templateName: 'To Remove', content: 'x', templateContent: 'x' },
205
- showAsCard: true,
206
- });
207
-
208
- // Same as edit: find the card's "more" menu then remove
209
- const card = screen.getByText('To Remove').closest('.ant-card');
210
- expect(card).toBeTruthy();
211
- const moreButton = card.querySelector('.ant-card-extra span');
212
- expect(moreButton).toBeTruthy();
213
- userEvent.click(moreButton);
214
-
215
- const removeLabel = smsFallbackMessages.removeTemplate.defaultMessage;
216
- const removeMenuItem = await screen.findByText(removeLabel);
217
- userEvent.click(removeMenuItem);
218
-
219
- await waitFor(() => {
220
- expect(onChange).toHaveBeenCalledWith(null);
221
- });
222
- });
223
- });