-
Notifications
You must be signed in to change notification settings - Fork 186
/
test-utils.ts
232 lines (206 loc) · 6.46 KB
/
test-utils.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
import * as React from "react";
import * as DeprecatedReactTestUtils from "react-dom/test-utils";
declare global {
var IS_REACT_ACT_ENVIRONMENT: boolean;
}
const act =
// @ts-ignore - Older versions of React don't have the `act` method, so TypeScript will complain about it
typeof React.act === "function" ? React.act : DeprecatedReactTestUtils.act;
type Item = {
callback: IntersectionObserverCallback;
elements: Set<Element>;
created: number;
};
let isMocking = false;
const observers = new Map<IntersectionObserver, Item>();
// If we are running in a valid testing environment, we can mock the IntersectionObserver.
if (typeof beforeEach !== "undefined" && typeof afterEach !== "undefined") {
beforeEach(() => {
// Use the exposed mock function. Currently, only supports Jest (`jest.fn`) and Vitest with globals (`vi.fn`).
if (typeof jest !== "undefined") setupIntersectionMocking(jest.fn);
else if (typeof vi !== "undefined") {
// Cast the `vi.fn` to `jest.fn` - The returned `Mock` type has a different signature than `jest.fn`
setupIntersectionMocking(vi.fn as unknown as typeof jest.fn);
}
});
afterEach(() => {
resetIntersectionMocking();
});
}
function warnOnMissingSetup() {
if (isMocking) return;
console.error(
`React Intersection Observer was not configured to handle mocking.
Outside Jest and Vitest, you might need to manually configure it by calling setupIntersectionMocking() and resetIntersectionMocking() in your test setup file.
// test-setup.js
import { resetIntersectionMocking, setupIntersectionMocking } from 'react-intersection-observer/test-utils';
beforeEach(() => {
setupIntersectionMocking(vi.fn);
});
afterEach(() => {
resetIntersectionMocking();
});`,
);
}
/**
* Create a custom IntersectionObserver mock, allowing us to intercept the `observe` and `unobserve` calls.
* We keep track of the elements being observed, so when `mockAllIsIntersecting` is triggered it will
* know which elements to trigger the event on.
* @param mockFn The mock function to use. Defaults to `jest.fn`.
*/
export function setupIntersectionMocking(mockFn: typeof jest.fn) {
global.IntersectionObserver = mockFn((cb, options = {}) => {
const item = {
callback: cb,
elements: new Set<Element>(),
created: Date.now(),
};
const instance: IntersectionObserver = {
thresholds: Array.isArray(options.threshold)
? options.threshold
: [options.threshold ?? 0],
root: options.root ?? null,
rootMargin: options.rootMargin ?? "",
observe: mockFn((element: Element) => {
item.elements.add(element);
}),
unobserve: mockFn((element: Element) => {
item.elements.delete(element);
}),
disconnect: mockFn(() => {
observers.delete(instance);
}),
takeRecords: mockFn(),
};
observers.set(instance, item);
return instance;
});
isMocking = true;
}
/**
* Reset the IntersectionObserver mock to its initial state, and clear all the elements being observed.
*/
export function resetIntersectionMocking() {
if (
global.IntersectionObserver &&
"mockClear" in global.IntersectionObserver &&
typeof global.IntersectionObserver.mockClear === "function"
) {
global.IntersectionObserver.mockClear();
}
observers.clear();
}
function getIsReactActEnvironment() {
return Boolean(global.IS_REACT_ACT_ENVIRONMENT);
}
function triggerIntersection(
elements: Element[],
trigger: boolean | number,
observer: IntersectionObserver,
item: Item,
) {
const entries: IntersectionObserverEntry[] = [];
const isIntersecting =
typeof trigger === "number"
? observer.thresholds.some((threshold) => trigger >= threshold)
: trigger;
let ratio: number;
if (typeof trigger === "number") {
const intersectedThresholds = observer.thresholds.filter(
(threshold) => trigger >= threshold,
);
ratio =
intersectedThresholds.length > 0
? intersectedThresholds[intersectedThresholds.length - 1]
: 0;
} else {
ratio = trigger ? 1 : 0;
}
for (const element of elements) {
entries.push(<IntersectionObserverEntry>{
boundingClientRect: element.getBoundingClientRect(),
intersectionRatio: ratio,
intersectionRect: isIntersecting
? element.getBoundingClientRect()
: {
bottom: 0,
height: 0,
left: 0,
right: 0,
top: 0,
width: 0,
x: 0,
y: 0,
toJSON() {},
},
isIntersecting,
rootBounds:
observer.root instanceof Element
? observer.root?.getBoundingClientRect()
: null,
target: element,
time: Date.now() - item.created,
});
}
// Trigger the IntersectionObserver callback with all the entries
if (act && getIsReactActEnvironment())
act(() => item.callback(entries, observer));
else item.callback(entries, observer);
}
/**
* Set the `isIntersecting` on all current IntersectionObserver instances
* @param isIntersecting {boolean | number}
*/
export function mockAllIsIntersecting(isIntersecting: boolean | number) {
warnOnMissingSetup();
for (const [observer, item] of observers) {
triggerIntersection(
Array.from(item.elements),
isIntersecting,
observer,
item,
);
}
}
/**
* Set the `isIntersecting` for the IntersectionObserver of a specific element.
*
* @param element {Element}
* @param isIntersecting {boolean | number}
*/
export function mockIsIntersecting(
element: Element,
isIntersecting: boolean | number,
) {
warnOnMissingSetup();
const observer = intersectionMockInstance(element);
if (!observer) {
throw new Error(
"No IntersectionObserver instance found for element. Is it still mounted in the DOM?",
);
}
const item = observers.get(observer);
if (item) {
triggerIntersection([element], isIntersecting, observer, item);
}
}
/**
* Call the `intersectionMockInstance` method with an element, to get the (mocked)
* `IntersectionObserver` instance. You can use this to spy on the `observe` and
* `unobserve` methods.
* @param element {Element}
* @return IntersectionObserver
*/
export function intersectionMockInstance(
element: Element,
): IntersectionObserver {
warnOnMissingSetup();
for (const [observer, item] of observers) {
if (item.elements.has(element)) {
return observer;
}
}
throw new Error(
"Failed to find IntersectionObserver for element. Is it being observed?",
);
}