@windstream/react-shared-components 0.1.93 → 0.1.95
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/dist/contentful/index.esm.js +2 -2
- package/dist/contentful/index.esm.js.map +1 -1
- package/dist/contentful/index.js +3 -3
- package/dist/contentful/index.js.map +1 -1
- package/dist/core.d.ts +2 -2
- package/dist/index.d.ts +2 -2
- package/dist/index.esm.js +1 -1
- package/dist/index.esm.js.map +1 -1
- package/dist/index.js +3 -3
- package/dist/index.js.map +1 -1
- package/dist/styles.css +1 -1
- package/dist/utils/index.esm.js +1 -1
- package/dist/utils/index.js +1 -1
- package/package.json +14 -8
- package/src/components/accordion/index.test.tsx +270 -0
- package/src/components/alert-card/index.test.tsx +152 -0
- package/src/components/animation-wrapper/index.test.tsx +424 -0
- package/src/components/brand-button/index.test.tsx +292 -0
- package/src/components/button/index.test.tsx +91 -0
- package/src/components/call-button/index.test.tsx +260 -0
- package/src/components/checkbox/index.test.tsx +252 -0
- package/src/components/checklist/index.test.tsx +231 -0
- package/src/components/checklist/index.tsx +64 -29
- package/src/components/checklist/types.ts +7 -1
- package/src/components/collapse/index.test.tsx +277 -0
- package/src/components/collapse/index.tsx +1 -0
- package/src/components/divider/index.test.tsx +53 -0
- package/src/components/image/index.test.tsx +174 -0
- package/src/components/input/index.test.tsx +348 -0
- package/src/components/link/index.test.tsx +199 -0
- package/src/components/list/index.test.tsx +166 -0
- package/src/components/material-icon/index.test.tsx +130 -0
- package/src/components/modal/index.test.tsx +310 -0
- package/src/components/next-image/index.test.tsx +406 -0
- package/src/components/pagination/index.test.tsx +521 -0
- package/src/components/radio-button/index.test.tsx +151 -0
- package/src/components/see-more/index.test.tsx +96 -0
- package/src/components/select/index.test.tsx +256 -0
- package/src/components/select-plan-button/index.test.tsx +173 -0
- package/src/components/skeleton/index.test.tsx +74 -0
- package/src/components/spinner/index.test.tsx +76 -0
- package/src/components/text/index.test.tsx +65 -0
- package/src/components/tooltip/index.test.tsx +50 -0
- package/src/components/view-cart-button/index.test.tsx +57 -0
- package/src/contentful/blocks/accordion/index.test.tsx +218 -0
- package/src/contentful/blocks/accordion/index.tsx +3 -1
- package/src/contentful/blocks/address-input-banner/index.test.tsx +132 -0
- package/src/contentful/blocks/anchored-bottom-banner/index.test.tsx +287 -0
- package/src/contentful/blocks/blogs-grid/BlogGrid.stories.tsx +5 -4
- package/src/contentful/blocks/blogs-grid/index.test.tsx +355 -0
- package/src/contentful/blocks/blogs-grid-base/index.test.tsx +274 -0
- package/src/contentful/blocks/breadcrumbs/index.test.tsx +281 -0
- package/src/contentful/blocks/button/index.test.tsx +339 -0
- package/src/contentful/blocks/callout/index.test.tsx +539 -0
- package/src/contentful/blocks/cards/blog-card/index.test.tsx +218 -0
- package/src/contentful/blocks/cards/floating-image-card/index.test.tsx +201 -0
- package/src/contentful/blocks/cards/full-image-card/index.test.tsx +216 -0
- package/src/contentful/blocks/cards/index.test.tsx +39 -0
- package/src/contentful/blocks/cards/product-card/index.test.tsx +263 -0
- package/src/contentful/blocks/cards/simple-card/index.test.tsx +364 -0
- package/src/contentful/blocks/cards/simple-card/index.tsx +1 -1
- package/src/contentful/blocks/cards/testimonial-card/index.test.tsx +180 -0
- package/src/contentful/blocks/carousel/helper.test.tsx +539 -0
- package/src/contentful/blocks/carousel/index.test.tsx +308 -0
- package/src/contentful/blocks/carousel/types.test.ts +16 -0
- package/src/contentful/blocks/cart-retention-banner/index.test.tsx +409 -0
- package/src/contentful/blocks/cart-retention-banner/index.tsx +4 -4
- package/src/contentful/blocks/comparison-table/index.test.tsx +114 -0
- package/src/contentful/blocks/cookiebanner/index.test.tsx +277 -0
- package/src/contentful/blocks/cta-callout/index.test.tsx +244 -0
- package/src/contentful/blocks/dynamic-tabs/index.test.tsx +240 -0
- package/src/contentful/blocks/email-input-block/index.test.tsx +213 -0
- package/src/contentful/blocks/email-input-block/index.tsx +40 -35
- package/src/contentful/blocks/find-kinetic/index.test.tsx +269 -0
- package/src/contentful/blocks/floating-banner/index.test.tsx +246 -0
- package/src/contentful/blocks/footer/index.test.tsx +302 -0
- package/src/contentful/blocks/image-promo-bar/helper.test.tsx +61 -0
- package/src/contentful/blocks/image-promo-bar/index.test.tsx +467 -0
- package/src/contentful/blocks/image-promo-bar/index.tsx +248 -246
- package/src/contentful/blocks/image-promo-bar/vimeo-embed.test.tsx +142 -0
- package/src/contentful/blocks/image-promo-bar/youtube-embed.test.tsx +104 -0
- package/src/contentful/blocks/modal/index.test.tsx +209 -0
- package/src/contentful/blocks/navigation/desktop-link-groups.tsx/index.test.tsx +208 -0
- package/src/contentful/blocks/navigation/index.test.tsx +924 -0
- package/src/contentful/blocks/navigation/mobile-link-groups.tsx/index.test.tsx +131 -0
- package/src/contentful/blocks/primary-hero/index.test.tsx +286 -0
- package/src/contentful/blocks/primary-hero/index.tsx +7 -4
- package/src/contentful/blocks/search-block/index.test.tsx +268 -0
- package/src/contentful/blocks/shape-background-wrapper/index.test.tsx +284 -0
- package/src/contentful/blocks/text/index.test.tsx +36 -0
- package/src/contentful/index.test.ts +45 -0
- package/src/global-mocks/contentful/to-document.ts +25 -0
- package/src/global-mocks/cookie.ts +48 -0
- package/src/global-mocks/cx.ts +37 -0
- package/src/global-mocks/index.ts +89 -0
- package/src/global-mocks/speed-card-bg.ts +27 -0
- package/src/global-mocks/utm.ts +49 -0
- package/src/hooks/contentful/use-contentful-rich-text.test.tsx +1758 -0
- package/src/hooks/contentful/use-contentful-rich-text.tsx +1 -1
- package/src/hooks/contentful/use-processed-check-list.test.tsx +277 -0
- package/src/hooks/use-body-scroll-lock.test.ts +134 -0
- package/src/hooks/use-carousel-swipe.test.ts +393 -0
- package/src/hooks/use-outside-click.test.ts +142 -0
- package/src/index.ts +1 -1
- package/src/next/index.test.ts +7 -0
- package/src/setupTests.ts +17 -11
- package/src/utils/contentful/to-document.test.ts +85 -0
- package/src/utils/cookie.test.ts +180 -0
- package/src/utils/cx.test.ts +90 -0
- package/src/utils/index.test.ts +115 -0
- package/src/utils/speed-card-bg.test.ts +46 -0
- package/src/utils/utm.test.ts +359 -0
|
@@ -262,7 +262,7 @@ export function renderContentfulRichTextTable(
|
|
|
262
262
|
|
|
263
263
|
return (
|
|
264
264
|
<td
|
|
265
|
-
className={`rt-table-cell footnote break-words bg-white py-2
|
|
265
|
+
className={`rt-table-cell footnote break-words bg-white py-2 text-center align-top leading-5 text-text md:body2 first:text-left md:py-4 md:leading-7 ${
|
|
266
266
|
isScrollable
|
|
267
267
|
? "w-[50vw] min-w-[50vw] first:sticky first:left-0 first:z-10 first:w-[50vw] first:min-w-[50vw] first:border-r"
|
|
268
268
|
: "w-1/4 first:w-1/2"
|
|
@@ -0,0 +1,277 @@
|
|
|
1
|
+
import {
|
|
2
|
+
label1BoldOptions,
|
|
3
|
+
useProcessedChecklist,
|
|
4
|
+
} from "./use-processed-check-list";
|
|
5
|
+
|
|
6
|
+
import { MARKS } from "@contentful/rich-text-types";
|
|
7
|
+
import { renderHook } from "@testing-library/react";
|
|
8
|
+
|
|
9
|
+
// Mock renderContentfulRichText to return predictable output
|
|
10
|
+
jest.mock("./use-contentful-rich-text", () => ({
|
|
11
|
+
renderContentfulRichText: jest.fn(
|
|
12
|
+
(doc: any, _target?: boolean, className?: string) => {
|
|
13
|
+
// Return a simple string representation based on the doc
|
|
14
|
+
if (!doc) return null;
|
|
15
|
+
// Extract text from the doc content if available
|
|
16
|
+
const text = doc?.content?.[0]?.content?.[0]?.value;
|
|
17
|
+
return text ? `rendered:${text}:${className ?? ""}` : null;
|
|
18
|
+
}
|
|
19
|
+
),
|
|
20
|
+
}));
|
|
21
|
+
|
|
22
|
+
jest.mock("@shared/utils/contentful/to-document", () => ({
|
|
23
|
+
toDocument: jest.fn((v: any) => {
|
|
24
|
+
if (!v) return null;
|
|
25
|
+
if (typeof v === "string") {
|
|
26
|
+
return {
|
|
27
|
+
nodeType: "document",
|
|
28
|
+
data: {},
|
|
29
|
+
content: [
|
|
30
|
+
{
|
|
31
|
+
nodeType: "paragraph",
|
|
32
|
+
data: {},
|
|
33
|
+
content: [{ nodeType: "text", value: v, marks: [], data: {} }],
|
|
34
|
+
},
|
|
35
|
+
],
|
|
36
|
+
};
|
|
37
|
+
}
|
|
38
|
+
return v;
|
|
39
|
+
}),
|
|
40
|
+
}));
|
|
41
|
+
|
|
42
|
+
const makeChecklistData = (items: any[]) => ({
|
|
43
|
+
list: {
|
|
44
|
+
items,
|
|
45
|
+
},
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
describe("useProcessedChecklist", () => {
|
|
49
|
+
describe("Full items mode (titlesOnly=false)", () => {
|
|
50
|
+
it("returns empty array when checklistData is undefined", () => {
|
|
51
|
+
const { result } = renderHook(() => useProcessedChecklist(undefined));
|
|
52
|
+
expect(result.current).toEqual([]);
|
|
53
|
+
});
|
|
54
|
+
|
|
55
|
+
it("returns empty array when checklistData has no list", () => {
|
|
56
|
+
const { result } = renderHook(() => useProcessedChecklist({}));
|
|
57
|
+
expect(result.current).toEqual([]);
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
it("returns empty array when list.items is empty", () => {
|
|
61
|
+
const data = makeChecklistData([]);
|
|
62
|
+
const { result } = renderHook(() => useProcessedChecklist(data));
|
|
63
|
+
expect(result.current).toEqual([]);
|
|
64
|
+
});
|
|
65
|
+
|
|
66
|
+
it("returns ProcessedChecklistItem array with title, iconUrl, anchorId", () => {
|
|
67
|
+
const data = makeChecklistData([
|
|
68
|
+
{
|
|
69
|
+
checkListTitle: "Item 1",
|
|
70
|
+
icon: { url: "https://cdn.com/icon1.png" },
|
|
71
|
+
anchorId: "anchor-1",
|
|
72
|
+
},
|
|
73
|
+
{
|
|
74
|
+
checkListTitle: "Item 2",
|
|
75
|
+
icon: { url: "https://cdn.com/icon2.png" },
|
|
76
|
+
anchorId: "anchor-2",
|
|
77
|
+
},
|
|
78
|
+
]);
|
|
79
|
+
const { result } = renderHook(() => useProcessedChecklist(data));
|
|
80
|
+
expect(result.current).toHaveLength(2);
|
|
81
|
+
expect(result.current[0]).toEqual({
|
|
82
|
+
title: "rendered:Item 1:",
|
|
83
|
+
iconUrl: "https://cdn.com/icon1.png",
|
|
84
|
+
anchorId: "anchor-1",
|
|
85
|
+
});
|
|
86
|
+
expect(result.current[1]).toEqual({
|
|
87
|
+
title: "rendered:Item 2:",
|
|
88
|
+
iconUrl: "https://cdn.com/icon2.png",
|
|
89
|
+
anchorId: "anchor-2",
|
|
90
|
+
});
|
|
91
|
+
});
|
|
92
|
+
|
|
93
|
+
it("sets iconUrl to undefined when icon is missing", () => {
|
|
94
|
+
const data = makeChecklistData([
|
|
95
|
+
{ checkListTitle: "No icon", icon: undefined },
|
|
96
|
+
]);
|
|
97
|
+
const { result } = renderHook(() => useProcessedChecklist(data));
|
|
98
|
+
expect(result.current[0].iconUrl).toBeUndefined();
|
|
99
|
+
});
|
|
100
|
+
|
|
101
|
+
it("sets anchorId to undefined when anchorId is missing", () => {
|
|
102
|
+
const data = makeChecklistData([{ checkListTitle: "No anchor" }]);
|
|
103
|
+
const { result } = renderHook(() => useProcessedChecklist(data));
|
|
104
|
+
expect(result.current[0].anchorId).toBeUndefined();
|
|
105
|
+
});
|
|
106
|
+
|
|
107
|
+
it("uses empty string for title when checkListTitle is null", () => {
|
|
108
|
+
const data = makeChecklistData([{ checkListTitle: null }]);
|
|
109
|
+
const { result } = renderHook(() => useProcessedChecklist(data));
|
|
110
|
+
// toDocument("") returns null, renderContentfulRichText returns null, falls back to ""
|
|
111
|
+
expect(result.current[0].title).toBe("");
|
|
112
|
+
});
|
|
113
|
+
|
|
114
|
+
it("handles item with undefined checkListTitle", () => {
|
|
115
|
+
const data = makeChecklistData([{}]);
|
|
116
|
+
const { result } = renderHook(() => useProcessedChecklist(data));
|
|
117
|
+
expect(result.current[0].title).toBe("");
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
it("handles null item in items array", () => {
|
|
121
|
+
const data = makeChecklistData([null]);
|
|
122
|
+
const { result } = renderHook(() => useProcessedChecklist(data));
|
|
123
|
+
expect(result.current[0].title).toBe("");
|
|
124
|
+
expect(result.current[0].iconUrl).toBeUndefined();
|
|
125
|
+
expect(result.current[0].anchorId).toBeUndefined();
|
|
126
|
+
});
|
|
127
|
+
|
|
128
|
+
it("handles undefined item in items array", () => {
|
|
129
|
+
const data = makeChecklistData([undefined]);
|
|
130
|
+
const { result } = renderHook(() => useProcessedChecklist(data));
|
|
131
|
+
expect(result.current[0].title).toBe("");
|
|
132
|
+
});
|
|
133
|
+
|
|
134
|
+
it("passes richTextClassName to renderContentfulRichText in full mode", () => {
|
|
135
|
+
const data = makeChecklistData([{ checkListTitle: "Styled" }]);
|
|
136
|
+
const { result } = renderHook(() =>
|
|
137
|
+
useProcessedChecklist(data, false, undefined, "full-class")
|
|
138
|
+
);
|
|
139
|
+
expect(result.current[0].title).toBe("rendered:Styled:full-class");
|
|
140
|
+
});
|
|
141
|
+
|
|
142
|
+
it("passes richTextOptions in full mode", () => {
|
|
143
|
+
const data = makeChecklistData([{ checkListTitle: "Opts" }]);
|
|
144
|
+
const customOpts = { renderMark: { [MARKS.BOLD]: (t: any) => t } };
|
|
145
|
+
const { result } = renderHook(() =>
|
|
146
|
+
useProcessedChecklist(data, false, customOpts, "opts-class")
|
|
147
|
+
);
|
|
148
|
+
expect(result.current[0].title).toBe("rendered:Opts:opts-class");
|
|
149
|
+
});
|
|
150
|
+
|
|
151
|
+
it("handles item with icon but no url", () => {
|
|
152
|
+
const data = makeChecklistData([{ checkListTitle: "No url", icon: {} }]);
|
|
153
|
+
const { result } = renderHook(() => useProcessedChecklist(data));
|
|
154
|
+
expect(result.current[0].iconUrl).toBeUndefined();
|
|
155
|
+
});
|
|
156
|
+
});
|
|
157
|
+
|
|
158
|
+
describe("Titles only mode (titlesOnly=true)", () => {
|
|
159
|
+
it("returns string array of rendered titles", () => {
|
|
160
|
+
const data = makeChecklistData([
|
|
161
|
+
{ checkListTitle: "Title A" },
|
|
162
|
+
{ checkListTitle: "Title B" },
|
|
163
|
+
]);
|
|
164
|
+
const { result } = renderHook(() => useProcessedChecklist(data, true));
|
|
165
|
+
expect(result.current).toHaveLength(2);
|
|
166
|
+
expect(result.current[0]).toBe("rendered:Title A:");
|
|
167
|
+
expect(result.current[1]).toBe("rendered:Title B:");
|
|
168
|
+
});
|
|
169
|
+
|
|
170
|
+
it("returns empty string for null checkListTitle", () => {
|
|
171
|
+
const data = makeChecklistData([{ checkListTitle: null }]);
|
|
172
|
+
const { result } = renderHook(() => useProcessedChecklist(data, true));
|
|
173
|
+
expect(result.current[0]).toBe("");
|
|
174
|
+
});
|
|
175
|
+
|
|
176
|
+
it("returns empty array when no items", () => {
|
|
177
|
+
const { result } = renderHook(() =>
|
|
178
|
+
useProcessedChecklist(undefined, true)
|
|
179
|
+
);
|
|
180
|
+
expect(result.current).toEqual([]);
|
|
181
|
+
});
|
|
182
|
+
|
|
183
|
+
it("handles null item in items array in titlesOnly mode", () => {
|
|
184
|
+
const data = makeChecklistData([null]);
|
|
185
|
+
const { result } = renderHook(() => useProcessedChecklist(data, true));
|
|
186
|
+
expect(result.current[0]).toBe("");
|
|
187
|
+
});
|
|
188
|
+
|
|
189
|
+
it("handles undefined item in items array in titlesOnly mode", () => {
|
|
190
|
+
const data = makeChecklistData([undefined]);
|
|
191
|
+
const { result } = renderHook(() => useProcessedChecklist(data, true));
|
|
192
|
+
expect(result.current[0]).toBe("");
|
|
193
|
+
});
|
|
194
|
+
|
|
195
|
+
it("passes richTextOptions in titlesOnly mode", () => {
|
|
196
|
+
const data = makeChecklistData([{ checkListTitle: "Opt" }]);
|
|
197
|
+
const customOpts = { renderMark: {} };
|
|
198
|
+
const { result } = renderHook(() =>
|
|
199
|
+
useProcessedChecklist(data, true, customOpts, "title-class")
|
|
200
|
+
);
|
|
201
|
+
expect(result.current[0]).toBe("rendered:Opt:title-class");
|
|
202
|
+
});
|
|
203
|
+
});
|
|
204
|
+
|
|
205
|
+
describe("richTextOptions and richTextClassName", () => {
|
|
206
|
+
it("passes richTextClassName to renderContentfulRichText", () => {
|
|
207
|
+
const data = makeChecklistData([{ checkListTitle: "Styled" }]);
|
|
208
|
+
const { result } = renderHook(() =>
|
|
209
|
+
useProcessedChecklist(data, true, undefined, "custom-class")
|
|
210
|
+
);
|
|
211
|
+
expect(result.current[0]).toBe("rendered:Styled:custom-class");
|
|
212
|
+
});
|
|
213
|
+
|
|
214
|
+
it("passes richTextOptions through", () => {
|
|
215
|
+
const data = makeChecklistData([{ checkListTitle: "Options" }]);
|
|
216
|
+
const customOpts = { renderMark: {} };
|
|
217
|
+
const { result } = renderHook(() =>
|
|
218
|
+
useProcessedChecklist(data, false, customOpts)
|
|
219
|
+
);
|
|
220
|
+
// Should still produce results (richTextOptions is passed but our mock ignores it)
|
|
221
|
+
expect(result.current).toHaveLength(1);
|
|
222
|
+
});
|
|
223
|
+
});
|
|
224
|
+
|
|
225
|
+
describe("Memoization", () => {
|
|
226
|
+
it("returns same reference when inputs don't change", () => {
|
|
227
|
+
const data = makeChecklistData([{ checkListTitle: "Memo" }]);
|
|
228
|
+
const { result, rerender } = renderHook(
|
|
229
|
+
({ d }) => useProcessedChecklist(d),
|
|
230
|
+
{ initialProps: { d: data } }
|
|
231
|
+
);
|
|
232
|
+
const first = result.current;
|
|
233
|
+
rerender({ d: data });
|
|
234
|
+
expect(result.current).toBe(first);
|
|
235
|
+
});
|
|
236
|
+
|
|
237
|
+
it("recomputes when checklistData changes", () => {
|
|
238
|
+
const data1 = makeChecklistData([{ checkListTitle: "V1" }]);
|
|
239
|
+
const data2 = makeChecklistData([{ checkListTitle: "V2" }]);
|
|
240
|
+
const { result, rerender } = renderHook(
|
|
241
|
+
({ d }) => useProcessedChecklist(d),
|
|
242
|
+
{ initialProps: { d: data1 } }
|
|
243
|
+
);
|
|
244
|
+
const first = result.current;
|
|
245
|
+
rerender({ d: data2 });
|
|
246
|
+
expect(result.current).not.toBe(first);
|
|
247
|
+
});
|
|
248
|
+
|
|
249
|
+
it("recomputes when titlesOnly changes", () => {
|
|
250
|
+
const data = makeChecklistData([{ checkListTitle: "Toggle" }]);
|
|
251
|
+
const { result, rerender } = renderHook(
|
|
252
|
+
({ t }) => useProcessedChecklist(data, t),
|
|
253
|
+
{ initialProps: { t: false as boolean } }
|
|
254
|
+
);
|
|
255
|
+
const first = result.current;
|
|
256
|
+
rerender({ t: true });
|
|
257
|
+
expect(result.current).not.toBe(first);
|
|
258
|
+
});
|
|
259
|
+
});
|
|
260
|
+
});
|
|
261
|
+
|
|
262
|
+
describe("label1BoldOptions", () => {
|
|
263
|
+
it("exports BOLD mark renderer that wraps in span with label1 class", () => {
|
|
264
|
+
const renderer = label1BoldOptions.renderMark[MARKS.BOLD];
|
|
265
|
+
expect(renderer).toBeDefined();
|
|
266
|
+
});
|
|
267
|
+
|
|
268
|
+
it("renders bold text in a span with label1 className", () => {
|
|
269
|
+
const renderer = label1BoldOptions.renderMark[MARKS.BOLD];
|
|
270
|
+
const result = renderer("Bold text");
|
|
271
|
+
// createElement returns a React element with type 'span' and className 'label1'
|
|
272
|
+
expect(result).toBeDefined();
|
|
273
|
+
expect((result as any).type).toBe("span");
|
|
274
|
+
expect((result as any).props.className).toBe("label1");
|
|
275
|
+
expect((result as any).props.children).toBe("Bold text");
|
|
276
|
+
});
|
|
277
|
+
});
|
|
@@ -0,0 +1,134 @@
|
|
|
1
|
+
import useBodyScrollLock from "./use-body-scroll-lock";
|
|
2
|
+
|
|
3
|
+
import { cleanup, renderHook } from "@testing-library/react";
|
|
4
|
+
|
|
5
|
+
describe("useBodyScrollLock", () => {
|
|
6
|
+
beforeEach(() => {
|
|
7
|
+
cleanup();
|
|
8
|
+
document.body.style.overflow = "";
|
|
9
|
+
document.body.style.marginRight = "";
|
|
10
|
+
});
|
|
11
|
+
|
|
12
|
+
afterEach(() => {
|
|
13
|
+
cleanup();
|
|
14
|
+
document.body.style.overflow = "";
|
|
15
|
+
document.body.style.marginRight = "";
|
|
16
|
+
});
|
|
17
|
+
|
|
18
|
+
describe("Scroll locking (isOpen=true)", () => {
|
|
19
|
+
it("sets body overflow to hidden when isOpen is true", () => {
|
|
20
|
+
renderHook(() => useBodyScrollLock(true, false));
|
|
21
|
+
expect(document.body.style.overflow).toBe("hidden");
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
it("sets marginRight based on scrollbar width when isOpen is true", () => {
|
|
25
|
+
renderHook(() => useBodyScrollLock(true, false));
|
|
26
|
+
const scrollbarWidth =
|
|
27
|
+
window.innerWidth - document.documentElement.clientWidth;
|
|
28
|
+
expect(document.body.style.marginRight).toBe(`${scrollbarWidth}px`);
|
|
29
|
+
});
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
describe("Scroll unlocking (isOpen=false)", () => {
|
|
33
|
+
it("resets overflow to unset when isOpen=false and hideScrollOnIsOpenFalse=true", () => {
|
|
34
|
+
renderHook(() => useBodyScrollLock(false, true));
|
|
35
|
+
expect(document.body.style.overflow).toBe("unset");
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
it("preserves existing overflow when isOpen=false and hideScrollOnIsOpenFalse=false", () => {
|
|
39
|
+
document.body.style.overflow = "auto";
|
|
40
|
+
renderHook(() => useBodyScrollLock(false, false));
|
|
41
|
+
expect(document.body.style.overflow).toBe("auto");
|
|
42
|
+
});
|
|
43
|
+
|
|
44
|
+
it("attempts to set marginRight to unset when isOpen is false", () => {
|
|
45
|
+
const calls: string[] = [];
|
|
46
|
+
const original = Object.getOwnPropertyDescriptor(
|
|
47
|
+
CSSStyleDeclaration.prototype,
|
|
48
|
+
"marginRight"
|
|
49
|
+
);
|
|
50
|
+
Object.defineProperty(document.body.style, "marginRight", {
|
|
51
|
+
set(val: string) {
|
|
52
|
+
calls.push(val);
|
|
53
|
+
},
|
|
54
|
+
get() {
|
|
55
|
+
return "";
|
|
56
|
+
},
|
|
57
|
+
configurable: true,
|
|
58
|
+
});
|
|
59
|
+
renderHook(() => useBodyScrollLock(false, false));
|
|
60
|
+
expect(calls).toContain("unset");
|
|
61
|
+
// Restore
|
|
62
|
+
if (original) {
|
|
63
|
+
Object.defineProperty(
|
|
64
|
+
CSSStyleDeclaration.prototype,
|
|
65
|
+
"marginRight",
|
|
66
|
+
original
|
|
67
|
+
);
|
|
68
|
+
}
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
it("attempts to set marginRight to unset for isOpen false regardless of hideScroll", () => {
|
|
72
|
+
const calls: string[] = [];
|
|
73
|
+
Object.defineProperty(document.body.style, "marginRight", {
|
|
74
|
+
set(val: string) {
|
|
75
|
+
calls.push(val);
|
|
76
|
+
},
|
|
77
|
+
get() {
|
|
78
|
+
return "";
|
|
79
|
+
},
|
|
80
|
+
configurable: true,
|
|
81
|
+
});
|
|
82
|
+
renderHook(() => useBodyScrollLock(false, true));
|
|
83
|
+
expect(calls).toContain("unset");
|
|
84
|
+
});
|
|
85
|
+
});
|
|
86
|
+
|
|
87
|
+
describe("Cleanup on unmount", () => {
|
|
88
|
+
it("resets body overflow to unset on unmount", () => {
|
|
89
|
+
const { unmount } = renderHook(() => useBodyScrollLock(true, false));
|
|
90
|
+
expect(document.body.style.overflow).toBe("hidden");
|
|
91
|
+
unmount();
|
|
92
|
+
expect(document.body.style.overflow).toBe("unset");
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
it("resets overflow on unmount even when isOpen was false", () => {
|
|
96
|
+
const { unmount } = renderHook(() => useBodyScrollLock(false, true));
|
|
97
|
+
unmount();
|
|
98
|
+
expect(document.body.style.overflow).toBe("unset");
|
|
99
|
+
});
|
|
100
|
+
});
|
|
101
|
+
|
|
102
|
+
describe("Dependency changes", () => {
|
|
103
|
+
it("locks scroll when isOpen changes from false to true", () => {
|
|
104
|
+
const { rerender } = renderHook(
|
|
105
|
+
({ isOpen, hide }) => useBodyScrollLock(isOpen, hide),
|
|
106
|
+
{ initialProps: { isOpen: false, hide: false } }
|
|
107
|
+
);
|
|
108
|
+
expect(document.body.style.overflow).not.toBe("hidden");
|
|
109
|
+
rerender({ isOpen: true, hide: false });
|
|
110
|
+
expect(document.body.style.overflow).toBe("hidden");
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
it("unlocks scroll when isOpen changes from true to false with hideScroll=true", () => {
|
|
114
|
+
const { rerender } = renderHook(
|
|
115
|
+
({ isOpen, hide }) => useBodyScrollLock(isOpen, hide),
|
|
116
|
+
{ initialProps: { isOpen: true, hide: true } }
|
|
117
|
+
);
|
|
118
|
+
expect(document.body.style.overflow).toBe("hidden");
|
|
119
|
+
rerender({ isOpen: false, hide: true });
|
|
120
|
+
expect(document.body.style.overflow).toBe("unset");
|
|
121
|
+
});
|
|
122
|
+
|
|
123
|
+
it("runs cleanup then effect when dependencies change", () => {
|
|
124
|
+
const { rerender } = renderHook(
|
|
125
|
+
({ isOpen, hide }) => useBodyScrollLock(isOpen, hide),
|
|
126
|
+
{ initialProps: { isOpen: true, hide: false } }
|
|
127
|
+
);
|
|
128
|
+
expect(document.body.style.overflow).toBe("hidden");
|
|
129
|
+
// Change hideScrollOnIsOpenFalse - triggers cleanup + re-run
|
|
130
|
+
rerender({ isOpen: true, hide: true });
|
|
131
|
+
expect(document.body.style.overflow).toBe("hidden");
|
|
132
|
+
});
|
|
133
|
+
});
|
|
134
|
+
});
|