@capillarytech/creatives-library 8.0.345-alpha.12 → 8.0.345-alpha.13

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 (138) hide show
  1. package/constants/unified.js +0 -29
  2. package/package.json +1 -1
  3. package/services/api.js +20 -0
  4. package/services/tests/api.test.js +59 -13
  5. package/utils/commonUtils.js +1 -19
  6. package/v2Components/CapActionButton/constants.js +0 -7
  7. package/v2Components/CapActionButton/index.js +109 -167
  8. package/v2Components/CapActionButton/index.scss +6 -157
  9. package/v2Components/CapActionButton/messages.js +3 -19
  10. package/v2Components/CapActionButton/tests/index.test.js +17 -41
  11. package/v2Components/CapCustomSkeleton/index.js +1 -1
  12. package/v2Components/CapCustomSkeleton/tests/__snapshots__/index.test.js.snap +12 -12
  13. package/v2Components/CapTagList/index.js +0 -10
  14. package/v2Components/CommonTestAndPreview/CustomValuesEditor.js +49 -70
  15. package/v2Components/CommonTestAndPreview/DeliverySettings/DeliverySettings.scss +2 -8
  16. package/v2Components/CommonTestAndPreview/DeliverySettings/ModifyDeliverySettings.js +21 -207
  17. package/v2Components/CommonTestAndPreview/DeliverySettings/constants.js +0 -16
  18. package/v2Components/CommonTestAndPreview/DeliverySettings/index.js +10 -85
  19. package/v2Components/CommonTestAndPreview/DeliverySettings/messages.js +0 -30
  20. package/v2Components/CommonTestAndPreview/DeliverySettings/utils/parseSenderDetailsResponse.js +11 -79
  21. package/v2Components/CommonTestAndPreview/SendTestMessage.js +5 -10
  22. package/v2Components/CommonTestAndPreview/UnifiedPreview/RcsPreviewContent.js +15 -160
  23. package/v2Components/CommonTestAndPreview/UnifiedPreview/_unifiedPreview.scss +76 -341
  24. package/v2Components/CommonTestAndPreview/UnifiedPreview/index.js +4 -133
  25. package/v2Components/CommonTestAndPreview/_commonTestAndPreview.scss +0 -11
  26. package/v2Components/CommonTestAndPreview/constants.js +2 -38
  27. package/v2Components/CommonTestAndPreview/index.js +186 -676
  28. package/v2Components/CommonTestAndPreview/messages.js +3 -49
  29. package/v2Components/CommonTestAndPreview/sagas.js +6 -15
  30. package/v2Components/CommonTestAndPreview/tests/CustomValuesEditor.test.js +284 -308
  31. package/v2Components/CommonTestAndPreview/tests/DeliverySettings/ModifyDeliverySettings.test.js +65 -231
  32. package/v2Components/CommonTestAndPreview/tests/DeliverySettings/index.test.js +5 -118
  33. package/v2Components/CommonTestAndPreview/tests/DeliverySettings/utils/parseSenderDetailsResponse.test.js +0 -341
  34. package/v2Components/CommonTestAndPreview/tests/PreviewSection.test.js +1 -8
  35. package/v2Components/CommonTestAndPreview/tests/SendTestMessage.test.js +13 -34
  36. package/v2Components/CommonTestAndPreview/tests/UnifiedPreview/RcsPreviewContent.test.js +283 -281
  37. package/v2Components/CommonTestAndPreview/tests/UnifiedPreview/index.test.js +1 -199
  38. package/v2Components/CommonTestAndPreview/tests/index.test.js +4 -132
  39. package/v2Components/CommonTestAndPreview/tests/sagas.test.js +2 -2
  40. package/v2Components/FormBuilder/index.js +10 -8
  41. package/v2Components/TemplatePreview/_templatePreview.scss +23 -33
  42. package/v2Components/TemplatePreview/index.js +28 -143
  43. package/v2Components/TemplatePreview/tests/index.test.js +0 -142
  44. package/v2Components/TestAndPreviewSlidebox/index.js +1 -13
  45. package/v2Components/TestAndPreviewSlidebox/sagas.js +4 -11
  46. package/v2Components/TestAndPreviewSlidebox/tests/saga.test.js +1 -3
  47. package/v2Containers/Assets/images/archive_Empty_Illustration.svg +9 -0
  48. package/v2Containers/CreativesContainer/SlideBoxContent.js +4 -36
  49. package/v2Containers/CreativesContainer/SlideBoxFooter.js +4 -11
  50. package/v2Containers/CreativesContainer/SlideBoxHeader.js +4 -29
  51. package/v2Containers/CreativesContainer/constants.js +0 -9
  52. package/v2Containers/CreativesContainer/index.js +108 -300
  53. package/v2Containers/CreativesContainer/index.scss +1 -51
  54. package/v2Containers/CreativesContainer/messages.js +4 -0
  55. package/v2Containers/CreativesContainer/tests/SlideBoxFooter.test.js +34 -78
  56. package/v2Containers/CreativesContainer/tests/SlideBoxHeader.test.js +16 -79
  57. package/v2Containers/CreativesContainer/tests/__snapshots__/SlideBoxContent.test.js.snap +0 -8
  58. package/v2Containers/CreativesContainer/tests/__snapshots__/SlideBoxHeader.test.js.snap +98 -357
  59. package/v2Containers/CreativesContainer/tests/__snapshots__/index.test.js.snap +18 -20
  60. package/v2Containers/CreativesContainer/tests/index.test.js +9 -71
  61. package/v2Containers/Rcs/constants.js +8 -119
  62. package/v2Containers/Rcs/index.js +812 -2375
  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 +70345 -98302
  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/ChannelTypeIllustration.js +23 -6
  80. package/v2Containers/Templates/_templates.scss +126 -181
  81. package/v2Containers/Templates/actions.js +36 -11
  82. package/v2Containers/Templates/constants.js +23 -2
  83. package/v2Containers/Templates/index.js +333 -142
  84. package/v2Containers/Templates/messages.js +68 -0
  85. package/v2Containers/Templates/reducer.js +68 -0
  86. package/v2Containers/Templates/sagas.js +98 -55
  87. package/v2Containers/Templates/selectors.js +12 -0
  88. package/v2Containers/Templates/tests/ChannelTypeIllustration.test.js +12 -0
  89. package/v2Containers/Templates/tests/__snapshots__/index.test.js.snap +1256 -1042
  90. package/v2Containers/Templates/tests/index.test.js +6 -0
  91. package/v2Containers/Templates/tests/reducer.test.js +178 -0
  92. package/v2Containers/Templates/tests/sagas.test.js +436 -200
  93. package/v2Containers/Templates/tests/selector.test.js +32 -0
  94. package/v2Containers/TemplatesV2/TemplatesV2.style.js +1 -72
  95. package/v2Containers/TemplatesV2/index.js +23 -86
  96. package/v2Containers/Whatsapp/index.js +20 -3
  97. package/v2Containers/Whatsapp/tests/__snapshots__/index.test.js.snap +34 -578
  98. package/utils/rcsPayloadUtils.js +0 -92
  99. package/utils/templateVarUtils.js +0 -201
  100. package/utils/tests/templateVarUtils.test.js +0 -204
  101. package/v2Components/CommonTestAndPreview/UnifiedPreview/RcsPreviewContent.js.rej +0 -18
  102. package/v2Components/CommonTestAndPreview/previewApiUtils.js +0 -59
  103. package/v2Components/CommonTestAndPreview/tests/previewApiUtils.test.js +0 -67
  104. package/v2Components/SmsFallback/SmsFallbackLocalSelector.js +0 -87
  105. package/v2Components/SmsFallback/constants.js +0 -73
  106. package/v2Components/SmsFallback/index.js +0 -955
  107. package/v2Components/SmsFallback/index.scss +0 -265
  108. package/v2Components/SmsFallback/messages.js +0 -78
  109. package/v2Components/SmsFallback/smsFallbackUtils.js +0 -118
  110. package/v2Components/SmsFallback/tests/SmsFallbackLocalSelector.test.js +0 -50
  111. package/v2Components/SmsFallback/tests/rcsSmsFallback.acceptance.test.js +0 -147
  112. package/v2Components/SmsFallback/tests/smsFallbackHandlers.test.js +0 -304
  113. package/v2Components/SmsFallback/tests/smsFallbackUi.test.js +0 -197
  114. package/v2Components/SmsFallback/tests/smsFallbackUtils.test.js +0 -277
  115. package/v2Components/SmsFallback/tests/useLocalTemplateList.test.js +0 -422
  116. package/v2Components/SmsFallback/useLocalTemplateList.js +0 -92
  117. package/v2Components/TemplatePreview/constants.js +0 -2
  118. package/v2Components/VarSegmentMessageEditor/constants.js +0 -2
  119. package/v2Components/VarSegmentMessageEditor/index.js +0 -125
  120. package/v2Components/VarSegmentMessageEditor/index.scss +0 -46
  121. package/v2Containers/CreativesContainer/CreativesSlideBoxWrapper.js +0 -43
  122. package/v2Containers/CreativesContainer/embeddedSlideboxUtils.js +0 -67
  123. package/v2Containers/CreativesContainer/tests/SlideBoxContent.localTemplates.test.js +0 -90
  124. package/v2Containers/CreativesContainer/tests/embeddedSlideboxUtils.test.js +0 -258
  125. package/v2Containers/CreativesContainer/tests/useLocalTemplatesProp.test.js +0 -125
  126. package/v2Containers/Rcs/index.js.rej +0 -1336
  127. package/v2Containers/Rcs/index.scss.rej +0 -74
  128. package/v2Containers/Rcs/rcsLibraryHydrationUtils.js +0 -225
  129. package/v2Containers/Rcs/tests/__snapshots__/utils.test.js.snap.rej +0 -128
  130. package/v2Containers/Rcs/tests/rcsLibraryHydrationUtils.test.js +0 -318
  131. package/v2Containers/Sms/smsFormDataHelpers.js +0 -67
  132. package/v2Containers/Sms/tests/smsFormDataHelpers.test.js +0 -253
  133. package/v2Containers/SmsTrai/Edit/index.scss +0 -121
  134. package/v2Containers/Templates/TemplatesActionBar.js +0 -101
  135. package/v2Containers/Templates/tests/TemplatesActionBar.test.js +0 -120
  136. package/v2Containers/Templates/tests/smsTemplatesListApi.test.js +0 -180
  137. package/v2Containers/Templates/utils/smsTemplatesListApi.js +0 -79
  138. 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
- });