@openmrs/esm-react-utils 9.0.2-pre.4055 → 9.0.2-pre.4060
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/.turbo/turbo-build.log +1 -1
- package/package.json +12 -12
- package/src/useDebounce.test.tsx +109 -0
package/.turbo/turbo-build.log
CHANGED
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@openmrs/esm-react-utils",
|
|
3
|
-
"version": "9.0.2-pre.
|
|
3
|
+
"version": "9.0.2-pre.4060",
|
|
4
4
|
"license": "MPL-2.0",
|
|
5
5
|
"description": "React utilities for OpenMRS.",
|
|
6
6
|
"type": "module",
|
|
@@ -79,17 +79,17 @@
|
|
|
79
79
|
"swr": "2.x"
|
|
80
80
|
},
|
|
81
81
|
"devDependencies": {
|
|
82
|
-
"@openmrs/esm-api": "9.0.2-pre.
|
|
83
|
-
"@openmrs/esm-config": "9.0.2-pre.
|
|
84
|
-
"@openmrs/esm-context": "9.0.2-pre.
|
|
85
|
-
"@openmrs/esm-emr-api": "9.0.2-pre.
|
|
86
|
-
"@openmrs/esm-error-handling": "9.0.2-pre.
|
|
87
|
-
"@openmrs/esm-extensions": "9.0.2-pre.
|
|
88
|
-
"@openmrs/esm-feature-flags": "9.0.2-pre.
|
|
89
|
-
"@openmrs/esm-globals": "9.0.2-pre.
|
|
90
|
-
"@openmrs/esm-navigation": "9.0.2-pre.
|
|
91
|
-
"@openmrs/esm-state": "9.0.2-pre.
|
|
92
|
-
"@openmrs/esm-utils": "9.0.2-pre.
|
|
82
|
+
"@openmrs/esm-api": "9.0.2-pre.4060",
|
|
83
|
+
"@openmrs/esm-config": "9.0.2-pre.4060",
|
|
84
|
+
"@openmrs/esm-context": "9.0.2-pre.4060",
|
|
85
|
+
"@openmrs/esm-emr-api": "9.0.2-pre.4060",
|
|
86
|
+
"@openmrs/esm-error-handling": "9.0.2-pre.4060",
|
|
87
|
+
"@openmrs/esm-extensions": "9.0.2-pre.4060",
|
|
88
|
+
"@openmrs/esm-feature-flags": "9.0.2-pre.4060",
|
|
89
|
+
"@openmrs/esm-globals": "9.0.2-pre.4060",
|
|
90
|
+
"@openmrs/esm-navigation": "9.0.2-pre.4060",
|
|
91
|
+
"@openmrs/esm-state": "9.0.2-pre.4060",
|
|
92
|
+
"@openmrs/esm-utils": "9.0.2-pre.4060",
|
|
93
93
|
"@swc/cli": "^0.7.7",
|
|
94
94
|
"@swc/core": "^1.11.29",
|
|
95
95
|
"@vitest/coverage-v8": "^4.0.18",
|
|
@@ -0,0 +1,109 @@
|
|
|
1
|
+
import { afterAll, afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
|
|
2
|
+
import { act, renderHook } from '@testing-library/react';
|
|
3
|
+
import { useDebounce } from './useDebounce';
|
|
4
|
+
|
|
5
|
+
describe('useDebounce', () => {
|
|
6
|
+
beforeEach(() => {
|
|
7
|
+
vi.useFakeTimers();
|
|
8
|
+
});
|
|
9
|
+
|
|
10
|
+
afterEach(() => {
|
|
11
|
+
vi.clearAllTimers();
|
|
12
|
+
vi.restoreAllMocks();
|
|
13
|
+
});
|
|
14
|
+
|
|
15
|
+
afterAll(() => {
|
|
16
|
+
vi.useRealTimers();
|
|
17
|
+
});
|
|
18
|
+
|
|
19
|
+
it('updates value after default delay', async () => {
|
|
20
|
+
const { result, rerender } = renderHook(
|
|
21
|
+
({ value }) => useDebounce(value),
|
|
22
|
+
{ initialProps: { value: 'initial' } },
|
|
23
|
+
);
|
|
24
|
+
|
|
25
|
+
rerender({ value: 'updated' });
|
|
26
|
+
|
|
27
|
+
expect(result.current).toBe('initial');
|
|
28
|
+
|
|
29
|
+
await act(async () => {
|
|
30
|
+
await vi.advanceTimersByTimeAsync(300);
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
expect(result.current).toBe('updated');
|
|
34
|
+
});
|
|
35
|
+
|
|
36
|
+
it('emits only final value on rapid updates', async () => {
|
|
37
|
+
const { result, rerender } = renderHook(
|
|
38
|
+
({ value }) => useDebounce(value),
|
|
39
|
+
{ initialProps: { value: 1 } },
|
|
40
|
+
);
|
|
41
|
+
|
|
42
|
+
rerender({ value: 2 });
|
|
43
|
+
rerender({ value: 3 });
|
|
44
|
+
rerender({ value: 4 });
|
|
45
|
+
|
|
46
|
+
await act(async () => {
|
|
47
|
+
await vi.advanceTimersByTimeAsync(300);
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
expect(result.current).toBe(4);
|
|
51
|
+
});
|
|
52
|
+
|
|
53
|
+
it('clears timeout on unmount', () => {
|
|
54
|
+
const clearTimeoutSpy = vi.spyOn(globalThis, 'clearTimeout');
|
|
55
|
+
|
|
56
|
+
const { rerender, unmount } = renderHook(
|
|
57
|
+
({ value }) => useDebounce(value),
|
|
58
|
+
{ initialProps: { value: 'start' } },
|
|
59
|
+
);
|
|
60
|
+
|
|
61
|
+
// Trigger a value change to schedule a pending timeout,
|
|
62
|
+
// so there is actually something to clean up on unmount
|
|
63
|
+
rerender({ value: 'changed' });
|
|
64
|
+
|
|
65
|
+
const callsBefore = clearTimeoutSpy.mock.calls.length;
|
|
66
|
+
|
|
67
|
+
unmount();
|
|
68
|
+
|
|
69
|
+
expect(clearTimeoutSpy.mock.calls.length).toBeGreaterThan(callsBefore);
|
|
70
|
+
});
|
|
71
|
+
|
|
72
|
+
it('preserves reference identity', async () => {
|
|
73
|
+
const updatedObj = { id: 1 };
|
|
74
|
+
|
|
75
|
+
const { result, rerender } = renderHook(
|
|
76
|
+
({ value }) => useDebounce(value),
|
|
77
|
+
{ initialProps: { value: { id: 0 } } },
|
|
78
|
+
);
|
|
79
|
+
|
|
80
|
+
rerender({ value: updatedObj });
|
|
81
|
+
|
|
82
|
+
await act(async () => {
|
|
83
|
+
await vi.advanceTimersByTimeAsync(300);
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
expect(result.current).toBe(updatedObj);
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
it('respects custom delay', async () => {
|
|
90
|
+
const { result, rerender } = renderHook(
|
|
91
|
+
({ value }) => useDebounce(value, 500),
|
|
92
|
+
{ initialProps: { value: 'initial' } },
|
|
93
|
+
);
|
|
94
|
+
|
|
95
|
+
rerender({ value: 'updated' });
|
|
96
|
+
|
|
97
|
+
await act(async () => {
|
|
98
|
+
await vi.advanceTimersByTimeAsync(300);
|
|
99
|
+
});
|
|
100
|
+
|
|
101
|
+
expect(result.current).toBe('initial');
|
|
102
|
+
|
|
103
|
+
await act(async () => {
|
|
104
|
+
await vi.advanceTimersByTimeAsync(200);
|
|
105
|
+
});
|
|
106
|
+
|
|
107
|
+
expect(result.current).toBe('updated');
|
|
108
|
+
});
|
|
109
|
+
});
|