@capillarytech/creatives-library 8.0.345-alpha.15 → 8.0.345-alpha.16

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 (130) hide show
  1. package/constants/unified.js +0 -29
  2. package/package.json +1 -1
  3. package/services/tests/api.test.js +0 -13
  4. package/utils/commonUtils.js +1 -19
  5. package/v2Components/CapActionButton/constants.js +0 -7
  6. package/v2Components/CapActionButton/index.js +109 -167
  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 -70
  12. package/v2Components/CommonTestAndPreview/DeliverySettings/DeliverySettings.scss +2 -8
  13. package/v2Components/CommonTestAndPreview/DeliverySettings/ModifyDeliverySettings.js +21 -207
  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 -160
  20. package/v2Components/CommonTestAndPreview/UnifiedPreview/_unifiedPreview.scss +76 -341
  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 -676
  25. package/v2Components/CommonTestAndPreview/messages.js +3 -49
  26. package/v2Components/CommonTestAndPreview/sagas.js +6 -15
  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 +2 -2
  37. package/v2Components/FormBuilder/index.js +10 -8
  38. package/v2Components/TemplatePreview/_templatePreview.scss +23 -33
  39. package/v2Components/TemplatePreview/index.js +28 -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 +103 -300
  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/Email/reducer.js +12 -3
  57. package/v2Containers/Email/sagas.js +9 -4
  58. package/v2Containers/Email/tests/__snapshots__/reducer.test.js.snap +4 -0
  59. package/v2Containers/Email/tests/reducer.test.js +47 -0
  60. package/v2Containers/Email/tests/sagas.test.js +146 -6
  61. package/v2Containers/Rcs/constants.js +8 -119
  62. package/v2Containers/Rcs/index.js +811 -2383
  63. package/v2Containers/Rcs/index.scss +6 -276
  64. package/v2Containers/Rcs/messages.js +3 -38
  65. package/v2Containers/Rcs/tests/__snapshots__/index.test.js.snap +70073 -98018
  66. package/v2Containers/Rcs/tests/__snapshots__/utils.test.js.snap +5 -0
  67. package/v2Containers/Rcs/tests/index.test.js +121 -152
  68. package/v2Containers/Rcs/tests/mockData.js +0 -38
  69. package/v2Containers/Rcs/tests/utils.test.js +30 -646
  70. package/v2Containers/Rcs/utils.js +11 -478
  71. package/v2Containers/Sms/Create/index.js +40 -100
  72. package/v2Containers/SmsTrai/Create/index.js +4 -9
  73. package/v2Containers/SmsTrai/Edit/constants.js +0 -2
  74. package/v2Containers/SmsTrai/Edit/index.js +130 -636
  75. package/v2Containers/SmsTrai/Edit/messages.js +4 -14
  76. package/v2Containers/SmsTrai/Edit/tests/__snapshots__/index.test.js.snap +2296 -4249
  77. package/v2Containers/SmsWrapper/index.js +8 -37
  78. package/v2Containers/TagList/index.js +0 -6
  79. package/v2Containers/Templates/_templates.scss +2 -163
  80. package/v2Containers/Templates/actions.js +0 -11
  81. package/v2Containers/Templates/constants.js +0 -2
  82. package/v2Containers/Templates/index.js +54 -119
  83. package/v2Containers/Templates/sagas.js +12 -57
  84. package/v2Containers/Templates/tests/__snapshots__/index.test.js.snap +1079 -1043
  85. package/v2Containers/Templates/tests/sagas.test.js +123 -193
  86. package/v2Containers/TemplatesV2/TemplatesV2.style.js +1 -72
  87. package/v2Containers/TemplatesV2/index.js +23 -86
  88. package/v2Containers/Whatsapp/index.js +20 -3
  89. package/v2Containers/Whatsapp/tests/__snapshots__/index.test.js.snap +34 -578
  90. package/utils/rcsPayloadUtils.js +0 -92
  91. package/utils/templateVarUtils.js +0 -201
  92. package/utils/tests/templateVarUtils.test.js +0 -204
  93. package/v2Components/CommonTestAndPreview/UnifiedPreview/RcsPreviewContent.js.rej +0 -18
  94. package/v2Components/CommonTestAndPreview/previewApiUtils.js +0 -59
  95. package/v2Components/CommonTestAndPreview/tests/previewApiUtils.test.js +0 -67
  96. package/v2Components/SmsFallback/SmsFallbackLocalSelector.js +0 -87
  97. package/v2Components/SmsFallback/constants.js +0 -73
  98. package/v2Components/SmsFallback/index.js +0 -955
  99. package/v2Components/SmsFallback/index.scss +0 -265
  100. package/v2Components/SmsFallback/messages.js +0 -78
  101. package/v2Components/SmsFallback/smsFallbackUtils.js +0 -118
  102. package/v2Components/SmsFallback/tests/SmsFallbackLocalSelector.test.js +0 -50
  103. package/v2Components/SmsFallback/tests/rcsSmsFallback.acceptance.test.js +0 -147
  104. package/v2Components/SmsFallback/tests/smsFallbackHandlers.test.js +0 -304
  105. package/v2Components/SmsFallback/tests/smsFallbackUi.test.js +0 -197
  106. package/v2Components/SmsFallback/tests/smsFallbackUtils.test.js +0 -277
  107. package/v2Components/SmsFallback/tests/useLocalTemplateList.test.js +0 -422
  108. package/v2Components/SmsFallback/useLocalTemplateList.js +0 -92
  109. package/v2Components/TemplatePreview/constants.js +0 -2
  110. package/v2Components/VarSegmentMessageEditor/constants.js +0 -2
  111. package/v2Components/VarSegmentMessageEditor/index.js +0 -125
  112. package/v2Components/VarSegmentMessageEditor/index.scss +0 -46
  113. package/v2Containers/CreativesContainer/CreativesSlideBoxWrapper.js +0 -43
  114. package/v2Containers/CreativesContainer/embeddedSlideboxUtils.js +0 -67
  115. package/v2Containers/CreativesContainer/tests/SlideBoxContent.localTemplates.test.js +0 -90
  116. package/v2Containers/CreativesContainer/tests/embeddedSlideboxUtils.test.js +0 -258
  117. package/v2Containers/CreativesContainer/tests/useLocalTemplatesProp.test.js +0 -125
  118. package/v2Containers/Rcs/index.js.rej +0 -1336
  119. package/v2Containers/Rcs/index.scss.rej +0 -74
  120. package/v2Containers/Rcs/rcsLibraryHydrationUtils.js +0 -225
  121. package/v2Containers/Rcs/tests/__snapshots__/utils.test.js.snap.rej +0 -128
  122. package/v2Containers/Rcs/tests/rcsLibraryHydrationUtils.test.js +0 -318
  123. package/v2Containers/Sms/smsFormDataHelpers.js +0 -67
  124. package/v2Containers/Sms/tests/smsFormDataHelpers.test.js +0 -253
  125. package/v2Containers/SmsTrai/Edit/index.scss +0 -121
  126. package/v2Containers/Templates/TemplatesActionBar.js +0 -101
  127. package/v2Containers/Templates/tests/TemplatesActionBar.test.js +0 -120
  128. package/v2Containers/Templates/tests/smsTemplatesListApi.test.js +0 -180
  129. package/v2Containers/Templates/utils/smsTemplatesListApi.js +0 -79
  130. package/v2Containers/TemplatesV2/tests/TemplatesV2.localTemplates.test.js +0 -131
@@ -1,277 +0,0 @@
1
- import {
2
- buildFallbackDataFromTemplate,
3
- mapFallbackValueToEditTemplateData,
4
- getBaseFromSmsTraiFormData,
5
- getSmsFallbackCardDisplayContent,
6
- resolveContentFromTraiBase,
7
- filterSmsTemplatesByCategory,
8
- buildFallbackDataFromCreativesPayload,
9
- } from '../smsFallbackUtils';
10
- import { SMS_CATEGORY_FILTERS } from '../constants';
11
-
12
- describe('smsFallbackUtils', () => {
13
- describe('buildFallbackDataFromTemplate', () => {
14
- it('maps template versions.base and header sender list', () => {
15
- const template = {
16
- _id: 'tid',
17
- name: 'My SMS',
18
- versions: {
19
- base: {
20
- 'sms-editor': 'Hello {{1}}',
21
- header: ['S1', 'S2'],
22
- 'unicode-validity': false,
23
- },
24
- },
25
- };
26
- expect(buildFallbackDataFromTemplate(template)).toEqual({
27
- smsTemplateId: 'tid',
28
- templateName: 'My SMS',
29
- content: 'Hello {{1}}',
30
- templateContent: 'Hello {{1}}',
31
- senderId: 'S1',
32
- registeredSenderIds: ['S1', 'S2'],
33
- unicodeValidity: false,
34
- });
35
- });
36
-
37
- it('falls back to senderId when header is missing', () => {
38
- const template = {
39
- _id: '',
40
- name: '',
41
- versions: { base: { 'sms-editor': 'x', senderId: 'SID' } },
42
- };
43
- expect(buildFallbackDataFromTemplate(template).senderId).toBe('SID');
44
- expect(buildFallbackDataFromTemplate(template).registeredSenderIds).toEqual([]);
45
- });
46
-
47
- it('defaults unicodeValidity to true when not a boolean', () => {
48
- const template = {
49
- versions: { base: { 'sms-editor': '' } },
50
- };
51
- expect(buildFallbackDataFromTemplate(template).unicodeValidity).toBe(true);
52
- });
53
-
54
- it('uses versions.base.template_name when root name is empty (DLT detail shape)', () => {
55
- const template = {
56
- _id: 'dlt1',
57
- name: '',
58
- versions: {
59
- base: {
60
- template_name: 'DLT Registered Name',
61
- 'sms-editor': 'Hi',
62
- header: ['H1'],
63
- },
64
- },
65
- };
66
- expect(buildFallbackDataFromTemplate(template).templateName).toBe('DLT Registered Name');
67
- });
68
- });
69
-
70
- describe('getSmsFallbackCardDisplayContent', () => {
71
- it('returns raw template when rcsSmsFallbackVarMapped is empty', () => {
72
- expect(
73
- getSmsFallbackCardDisplayContent({
74
- templateContent: 'A {#var#} B',
75
- rcsSmsFallbackVarMapped: {},
76
- }),
77
- ).toBe('A {#var#} B');
78
- });
79
-
80
- it('substitutes slot values like preview when var map is set', () => {
81
- expect(
82
- getSmsFallbackCardDisplayContent({
83
- templateContent: 'Hi {{name}}',
84
- rcsSmsFallbackVarMapped: { '{{name}}_1': 'Pat' },
85
- }),
86
- ).toBe('Hi Pat');
87
- });
88
-
89
- it('shows raw {#var#} in card when slots saved empty (DLT, no labels)', () => {
90
- expect(
91
- getSmsFallbackCardDisplayContent({
92
- templateContent: 'Balance {#var#}',
93
- rcsSmsFallbackVarMapped: { '{#var#}_1': '' },
94
- }),
95
- ).toBe('Balance {#var#}');
96
- });
97
- });
98
-
99
- describe('mapFallbackValueToEditTemplateData', () => {
100
- it('returns null when source is missing', () => {
101
- expect(mapFallbackValueToEditTemplateData(null)).toBeNull();
102
- expect(mapFallbackValueToEditTemplateData(undefined)).toBeNull();
103
- });
104
-
105
- it('maps content, optional header and unicode flag', () => {
106
- const source = {
107
- smsTemplateId: 'id1',
108
- templateName: 'N',
109
- templateContent: 'body',
110
- registeredSenderIds: ['h1'],
111
- unicodeValidity: false,
112
- };
113
- expect(mapFallbackValueToEditTemplateData(source)).toEqual({
114
- _id: 'id1',
115
- name: 'N',
116
- versions: {
117
- base: {
118
- 'sms-editor': 'body',
119
- template_name: 'N',
120
- header: ['h1'],
121
- 'unicode-validity': false,
122
- },
123
- },
124
- });
125
- });
126
-
127
- it('uses content when templateContent is empty', () => {
128
- const source = {
129
- smsTemplateId: 'x',
130
- templateName: 'y',
131
- content: 'from-content',
132
- };
133
- expect(mapFallbackValueToEditTemplateData(source).versions.base['sms-editor']).toBe('from-content');
134
- });
135
-
136
- it('omits header when registeredSenderIds is empty', () => {
137
- const source = {
138
- smsTemplateId: 'x',
139
- templateName: 'y',
140
- templateContent: 't',
141
- registeredSenderIds: [],
142
- };
143
- expect(mapFallbackValueToEditTemplateData(source).versions.base.header).toBeUndefined();
144
- });
145
-
146
- it('omits unicode-validity when value is not a boolean', () => {
147
- const source = {
148
- smsTemplateId: 'x',
149
- templateName: 'y',
150
- templateContent: 't',
151
- unicodeValidity: 'yes',
152
- };
153
- expect(mapFallbackValueToEditTemplateData(source).versions.base['unicode-validity']).toBeUndefined();
154
- });
155
- });
156
-
157
- describe('getBaseFromSmsTraiFormData', () => {
158
- it('reads versions.base', () => {
159
- const fd = { versions: { base: { 'sms-editor': 'a' } } };
160
- expect(getBaseFromSmsTraiFormData(fd)).toEqual({ base: { 'sms-editor': 'a' } });
161
- });
162
-
163
- it('falls back to value.base', () => {
164
- const fd = { value: { base: { 'sms-editor': 'b' } } };
165
- expect(getBaseFromSmsTraiFormData(fd)).toEqual({ base: { 'sms-editor': 'b' } });
166
- });
167
-
168
- it('uses empty base when neither path exists', () => {
169
- expect(getBaseFromSmsTraiFormData({})).toEqual({ base: {} });
170
- });
171
-
172
- it('normalizes null versions.base to empty object', () => {
173
- expect(getBaseFromSmsTraiFormData({ versions: { base: null } })).toEqual({ base: {} });
174
- });
175
- });
176
-
177
- describe('resolveContentFromTraiBase', () => {
178
- it('returns sms-editor string', () => {
179
- expect(resolveContentFromTraiBase({ 'sms-editor': 'z' })).toBe('z');
180
- });
181
-
182
- it('returns empty string when missing', () => {
183
- expect(resolveContentFromTraiBase({})).toBe('');
184
- });
185
- });
186
-
187
- describe('filterSmsTemplatesByCategory', () => {
188
- const templates = [
189
- { versions: { base: { type: 'Promo' } } },
190
- { versions: { base: { type: 'Explicit' } } },
191
- ];
192
-
193
- it('returns empty array for non-array input', () => {
194
- expect(filterSmsTemplatesByCategory(null, SMS_CATEGORY_FILTERS.PROMOTIONAL)).toEqual([]);
195
- });
196
-
197
- it('returns all templates for ALL or missing filter', () => {
198
- expect(filterSmsTemplatesByCategory(templates, SMS_CATEGORY_FILTERS.ALL)).toEqual(templates);
199
- expect(filterSmsTemplatesByCategory(templates, null)).toEqual(templates);
200
- });
201
-
202
- it('filters by lowercase type substring', () => {
203
- const filtered = filterSmsTemplatesByCategory(templates, SMS_CATEGORY_FILTERS.PROMOTIONAL);
204
- expect(filtered).toHaveLength(1);
205
- expect(filtered[0].versions.base.type).toBe('Promo');
206
- });
207
-
208
- it('treats missing type as empty string for filtering', () => {
209
- const mixed = [{}, { versions: { base: {} } }];
210
- expect(filterSmsTemplatesByCategory(mixed, SMS_CATEGORY_FILTERS.PROMOTIONAL)).toEqual([]);
211
- });
212
- });
213
-
214
- describe('buildFallbackDataFromCreativesPayload', () => {
215
- it('returns null for missing or non-SMS channel', () => {
216
- expect(buildFallbackDataFromCreativesPayload(null)).toBeNull();
217
- expect(buildFallbackDataFromCreativesPayload({ channel: 'RCS' })).toBeNull();
218
- });
219
-
220
- it('maps SMS payload with messageBody and templateConfigs', () => {
221
- const creativesData = {
222
- channel: 'SMS',
223
- messageBody: 'Hello',
224
- templateConfigs: {
225
- templateId: 99,
226
- templateName: 'T',
227
- template: 'Template str',
228
- registeredSenderIds: ['R1'],
229
- },
230
- };
231
- expect(buildFallbackDataFromCreativesPayload(creativesData)).toEqual({
232
- smsTemplateId: '99',
233
- templateName: 'T',
234
- content: 'Hello',
235
- templateContent: 'Template str',
236
- senderId: 'R1',
237
- registeredSenderIds: ['R1'],
238
- unicodeValidity: true,
239
- });
240
- });
241
-
242
- it('uses template string as body when messageBody is not a string', () => {
243
- const creativesData = {
244
- channel: 'SMS',
245
- messageBody: null,
246
- templateConfigs: {
247
- template: 'only-template',
248
- },
249
- };
250
- const out = buildFallbackDataFromCreativesPayload(creativesData);
251
- expect(out.content).toBe('only-template');
252
- expect(out.templateContent).toBe('only-template');
253
- });
254
-
255
- it('stringifies templateId and templateName when present', () => {
256
- const creativesData = {
257
- channel: 'SMS',
258
- messageBody: 'm',
259
- templateConfigs: { templateId: 0, templateName: null },
260
- };
261
- const out = buildFallbackDataFromCreativesPayload(creativesData);
262
- expect(out.smsTemplateId).toBe('0');
263
- expect(out.templateName).toBe('');
264
- });
265
-
266
- it('uses empty templateConfigs when null', () => {
267
- const creativesData = {
268
- channel: 'SMS',
269
- messageBody: 'only-body',
270
- templateConfigs: null,
271
- };
272
- const out = buildFallbackDataFromCreativesPayload(creativesData);
273
- expect(out.templateContent).toBe('only-body');
274
- expect(out.content).toBe('only-body');
275
- });
276
- });
277
- });
@@ -1,422 +0,0 @@
1
- import { renderHook, act } from '@testing-library/react-hooks';
2
- import { waitFor } from '@testing-library/react';
3
- import { useLocalTemplateList } from '../useLocalTemplateList';
4
-
5
- const makeDeferred = () => {
6
- let resolve;
7
- let reject;
8
- const promise = new Promise((res, rej) => {
9
- resolve = res;
10
- reject = rej;
11
- });
12
- return { promise, resolve, reject };
13
- };
14
-
15
- describe('useLocalTemplateList', () => {
16
- it('drops stale response when a newer reset starts', async () => {
17
- const first = makeDeferred();
18
- const second = makeDeferred();
19
- const fetchTemplates = jest
20
- .fn()
21
- .mockImplementationOnce(() => first.promise)
22
- .mockImplementationOnce(() => second.promise);
23
-
24
- const { result } = renderHook(() =>
25
- useLocalTemplateList({ fetchTemplates, perPage: 2 })
26
- );
27
-
28
- act(() => {
29
- result.current.reset('old-search');
30
- });
31
-
32
- act(() => {
33
- result.current.reset('new-search');
34
- });
35
-
36
- await act(async () => {
37
- second.resolve({
38
- templates: [{ _id: 'new-1' }],
39
- totalCount: 1,
40
- });
41
- await Promise.resolve();
42
- });
43
-
44
- expect(result.current.templates).toEqual([{ _id: 'new-1' }]);
45
- expect(result.current.totalCount).toBe(1);
46
-
47
- await act(async () => {
48
- first.resolve({
49
- templates: [{ _id: 'old-1' }],
50
- totalCount: 1,
51
- });
52
- await Promise.resolve();
53
- });
54
-
55
- // Older response should not overwrite latest state.
56
- expect(result.current.templates).toEqual([{ _id: 'new-1' }]);
57
- expect(result.current.search).toBe('new-search');
58
- });
59
-
60
- it('keeps existing templates when loadMore fails', async () => {
61
- const fetchTemplates = jest.fn(({ page }) => {
62
- if (page === 1) {
63
- return Promise.resolve({
64
- templates: [{ _id: 't1' }, { _id: 't2' }],
65
- totalCount: 4,
66
- });
67
- }
68
- return Promise.reject(new Error('load-more-failed'));
69
- });
70
-
71
- const { result } = renderHook(() =>
72
- useLocalTemplateList({ fetchTemplates, perPage: 2 })
73
- );
74
-
75
- act(() => {
76
- result.current.reset('');
77
- });
78
-
79
- await waitFor(() => {
80
- expect(result.current.loading).toBe(false);
81
- expect(result.current.templates).toHaveLength(2);
82
- expect(result.current.page).toBe(1);
83
- });
84
-
85
- act(() => {
86
- result.current.loadMore();
87
- });
88
-
89
- await waitFor(() => {
90
- expect(result.current.loading).toBe(false);
91
- });
92
-
93
- // Non-reset failure should not wipe already loaded data.
94
- expect(result.current.templates).toEqual([{ _id: 't1' }, { _id: 't2' }]);
95
- expect(result.current.totalCount).toBe(4);
96
- expect(result.current.page).toBe(1);
97
- });
98
-
99
- it('clears list on reset failure', async () => {
100
- const fetchTemplates = jest
101
- .fn()
102
- .mockResolvedValueOnce({
103
- templates: [{ _id: 'seed-1' }, { _id: 'seed-2' }],
104
- totalCount: 2,
105
- })
106
- .mockRejectedValueOnce(new Error('reset-failed'));
107
-
108
- const { result } = renderHook(() =>
109
- useLocalTemplateList({ fetchTemplates, perPage: 2 })
110
- );
111
-
112
- act(() => {
113
- result.current.reset('');
114
- });
115
-
116
- await waitFor(() => {
117
- expect(result.current.loading).toBe(false);
118
- expect(result.current.templates).toHaveLength(2);
119
- });
120
-
121
- act(() => {
122
- result.current.reset('fresh-search');
123
- });
124
-
125
- await waitFor(() => {
126
- expect(result.current.loading).toBe(false);
127
- });
128
-
129
- expect(result.current.templates).toEqual([]);
130
- expect(result.current.totalCount).toBe(0);
131
- expect(result.current.page).toBe(1);
132
- expect(result.current.search).toBe('fresh-search');
133
- });
134
-
135
- it('keeps previous totalCount when loadMore returns zero totalCount', async () => {
136
- const fetchTemplates = jest.fn(({ page }) => {
137
- if (page === 1) {
138
- return Promise.resolve({
139
- templates: [{ _id: 't1' }],
140
- totalCount: 10,
141
- });
142
- }
143
- return Promise.resolve({
144
- templates: [{ _id: 't2' }],
145
- totalCount: 0,
146
- });
147
- });
148
-
149
- const { result } = renderHook(() =>
150
- useLocalTemplateList({ fetchTemplates, perPage: 1 })
151
- );
152
-
153
- act(() => {
154
- result.current.reset('');
155
- });
156
-
157
- await waitFor(() => {
158
- expect(result.current.loading).toBe(false);
159
- expect(result.current.templates).toHaveLength(1);
160
- expect(result.current.totalCount).toBe(10);
161
- });
162
-
163
- act(() => {
164
- result.current.loadMore();
165
- });
166
-
167
- await waitFor(() => {
168
- expect(result.current.loading).toBe(false);
169
- });
170
-
171
- expect(result.current.templates.map((t) => t._id)).toEqual(['t1', 't2']);
172
- expect(result.current.totalCount).toBe(10);
173
- });
174
-
175
- it('ignores rejection from a superseded fetch (stale generation)', async () => {
176
- const slowFirst = makeDeferred();
177
- const fetchTemplates = jest
178
- .fn()
179
- .mockImplementationOnce(() => slowFirst.promise)
180
- .mockImplementationOnce(() =>
181
- Promise.resolve({ templates: [{ _id: 'from-second' }], totalCount: 1 }),
182
- );
183
-
184
- const { result } = renderHook(() =>
185
- useLocalTemplateList({ fetchTemplates, perPage: 25 })
186
- );
187
-
188
- act(() => {
189
- result.current.reset('first-search');
190
- });
191
- act(() => {
192
- result.current.reset('second-search');
193
- });
194
-
195
- await act(async () => {
196
- await Promise.resolve();
197
- });
198
-
199
- await waitFor(() => {
200
- expect(result.current.loading).toBe(false);
201
- expect(result.current.templates).toEqual([{ _id: 'from-second' }]);
202
- });
203
-
204
- await act(async () => {
205
- slowFirst.reject(new Error('stale'));
206
- await Promise.resolve();
207
- });
208
-
209
- expect(result.current.templates).toEqual([{ _id: 'from-second' }]);
210
- expect(result.current.search).toBe('second-search');
211
- });
212
-
213
- it('does not leave loading stuck when stale resolve completes after newer fetch (finally gen check)', async () => {
214
- const d1 = makeDeferred();
215
- const fetchTemplates = jest
216
- .fn()
217
- .mockImplementationOnce(() => d1.promise)
218
- .mockResolvedValueOnce({ templates: [{ _id: 'new' }], totalCount: 1 });
219
-
220
- const { result } = renderHook(() =>
221
- useLocalTemplateList({ fetchTemplates, perPage: 2 }),
222
- );
223
-
224
- act(() => {
225
- result.current.reset('a');
226
- });
227
- act(() => {
228
- result.current.reset('b');
229
- });
230
-
231
- await act(async () => {
232
- d1.resolve({ templates: [{ _id: 'old' }], totalCount: 1 });
233
- await Promise.resolve();
234
- });
235
-
236
- await waitFor(() => {
237
- expect(result.current.loading).toBe(false);
238
- expect(result.current.templates).toEqual([{ _id: 'new' }]);
239
- });
240
- });
241
-
242
- it('coerces setSearch non-string values to empty search term', () => {
243
- const fetchTemplates = jest.fn().mockResolvedValue({ templates: [], totalCount: 0 });
244
- const { result } = renderHook(() =>
245
- useLocalTemplateList({ fetchTemplates, perPage: 25 }),
246
- );
247
-
248
- act(() => {
249
- result.current.setSearch(null);
250
- });
251
- expect(result.current.search).toBe('');
252
-
253
- act(() => {
254
- result.current.setSearch(123);
255
- });
256
- expect(result.current.search).toBe('');
257
- });
258
-
259
- it('does not call fetch when loadMore runs but canLoadMore is false', async () => {
260
- const fetchTemplates = jest.fn().mockResolvedValue({
261
- templates: [{ _id: 'only' }],
262
- totalCount: 1,
263
- });
264
-
265
- const { result } = renderHook(() =>
266
- useLocalTemplateList({ fetchTemplates, perPage: 1 }),
267
- );
268
-
269
- act(() => {
270
- result.current.reset('');
271
- });
272
-
273
- await waitFor(() => {
274
- expect(result.current.loading).toBe(false);
275
- expect(result.current.canLoadMore).toBe(false);
276
- });
277
-
278
- const callsAfterFirst = fetchTemplates.mock.calls.length;
279
-
280
- act(() => {
281
- result.current.loadMore();
282
- });
283
-
284
- expect(fetchTemplates.mock.calls.length).toBe(callsAfterFirst);
285
- });
286
-
287
- it('passes explicit search term from reset() through to fetchTemplates', async () => {
288
- const fetchTemplates = jest.fn().mockResolvedValue({
289
- templates: [{ _id: 'r1' }],
290
- totalCount: 5,
291
- });
292
-
293
- const { result } = renderHook(() =>
294
- useLocalTemplateList({ fetchTemplates, perPage: 25 }),
295
- );
296
-
297
- act(() => {
298
- result.current.reset('explicit-search');
299
- });
300
-
301
- await waitFor(() => {
302
- expect(result.current.loading).toBe(false);
303
- });
304
-
305
- expect(fetchTemplates).toHaveBeenCalledWith(
306
- expect.objectContaining({ search: 'explicit-search', page: 1, reset: true }),
307
- );
308
- expect(result.current.search).toBe('explicit-search');
309
- });
310
-
311
- it('stores a positive totalCount returned by the API', async () => {
312
- const fetchTemplates = jest.fn().mockResolvedValue({
313
- templates: [{ _id: 'a' }, { _id: 'b' }],
314
- totalCount: 42,
315
- });
316
-
317
- const { result } = renderHook(() =>
318
- useLocalTemplateList({ fetchTemplates, perPage: 25 }),
319
- );
320
-
321
- act(() => {
322
- result.current.reset('');
323
- });
324
-
325
- await waitFor(() => {
326
- expect(result.current.loading).toBe(false);
327
- });
328
-
329
- expect(result.current.totalCount).toBe(42);
330
- expect(result.current.templates).toHaveLength(2);
331
- });
332
-
333
- it('canLoadMore is false while a fetch is in progress even when more items exist', async () => {
334
- let resolveFirst;
335
- const fetchTemplates = jest.fn(
336
- () => new Promise((res) => { resolveFirst = res; }),
337
- );
338
-
339
- const { result } = renderHook(() =>
340
- useLocalTemplateList({ fetchTemplates, perPage: 2 }),
341
- );
342
-
343
- act(() => {
344
- result.current.reset('');
345
- });
346
-
347
- // Loading is true → canLoadMore must be false regardless of hasMoreByTotal
348
- expect(result.current.loading).toBe(true);
349
- expect(result.current.canLoadMore).toBe(false);
350
-
351
- await act(async () => {
352
- resolveFirst({ templates: [{ _id: 'x' }, { _id: 'y' }], totalCount: 10 });
353
- await Promise.resolve();
354
- });
355
-
356
- expect(result.current.loading).toBe(false);
357
- expect(result.current.canLoadMore).toBe(true);
358
- });
359
-
360
- it('canLoadMore is false when templates is empty (hasMoreByFullPage requires length > 0)', async () => {
361
- const fetchTemplates = jest.fn().mockResolvedValue({
362
- templates: [],
363
- totalCount: 0,
364
- });
365
-
366
- const { result } = renderHook(() =>
367
- useLocalTemplateList({ fetchTemplates, perPage: 0 }),
368
- );
369
-
370
- act(() => {
371
- result.current.reset('');
372
- });
373
-
374
- await waitFor(() => {
375
- expect(result.current.loading).toBe(false);
376
- });
377
-
378
- // templates.length === 0 means hasMoreByFullPage is false even if lastFetchFullPage would be true
379
- expect(result.current.canLoadMore).toBe(false);
380
- });
381
-
382
- it('allows loadMore when totalCount is unknown but first page is full (hasMoreByFullPage)', async () => {
383
- const fetchTemplates = jest.fn(({ page }) => {
384
- if (page === 1) {
385
- return Promise.resolve({
386
- templates: [{ _id: 'a' }, { _id: 'b' }],
387
- totalCount: 0,
388
- });
389
- }
390
- return Promise.resolve({
391
- templates: [{ _id: 'c' }],
392
- totalCount: 0,
393
- });
394
- });
395
-
396
- const { result } = renderHook(() =>
397
- useLocalTemplateList({ fetchTemplates, perPage: 2 }),
398
- );
399
-
400
- act(() => {
401
- result.current.reset('');
402
- });
403
-
404
- await waitFor(() => {
405
- expect(result.current.loading).toBe(false);
406
- expect(result.current.canLoadMore).toBe(true);
407
- });
408
-
409
- act(() => {
410
- result.current.loadMore();
411
- });
412
-
413
- await waitFor(() => {
414
- expect(result.current.loading).toBe(false);
415
- });
416
-
417
- expect(result.current.templates.map((t) => t._id)).toEqual(['a', 'b', 'c']);
418
- expect(fetchTemplates).toHaveBeenCalledWith(
419
- expect.objectContaining({ page: 2, reset: false }),
420
- );
421
- });
422
- });