Skip to content

Commit 5ca4927

Browse files
committed
Use dedicated entrypoint instead
Less renaming to-do
1 parent 571fbc8 commit 5ca4927

14 files changed

+799
-309
lines changed

async.d.ts

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export * from './types/pure-async'

async.js

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
// makes it so people can import from '@testing-library/react/async'
2+
module.exports = require('./dist/async')

pure-async.d.ts

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export * from './types/pure-async'

pure-async.js

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
// makes it so people can import from '@testing-library/react/pure-async'
2+
module.exports = require('./dist/pure-async')

src/__tests__/async.js

+73
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,73 @@
1+
// TODO: Upstream that the rule should check import source
2+
/* eslint-disable testing-library/no-await-sync-events */
3+
import * as React from 'react'
4+
import {act, render, fireEvent} from '../async'
5+
6+
const isReact19 = React.version.startsWith('19.')
7+
8+
const testGateReact19 = isReact19 ? test : test.skip
9+
10+
testGateReact19('async data requires async APIs', async () => {
11+
let resolve
12+
const promise = new Promise(_resolve => {
13+
resolve = _resolve
14+
})
15+
16+
function Component() {
17+
const value = React.use(promise)
18+
return <div>{value}</div>
19+
}
20+
21+
const {container} = await render(
22+
<React.Suspense fallback="loading...">
23+
<Component />
24+
</React.Suspense>,
25+
)
26+
27+
expect(container).toHaveTextContent('loading...')
28+
29+
await act(async () => {
30+
resolve('Hello, Dave!')
31+
})
32+
33+
expect(container).toHaveTextContent('Hello, Dave!')
34+
})
35+
36+
testGateReact19('async fireEvent', async () => {
37+
let resolve
38+
function Component() {
39+
const [promise, setPromise] = React.useState('initial')
40+
const value = typeof promise === 'string' ? promise : React.use(promise)
41+
return (
42+
<button
43+
onClick={() =>
44+
setPromise(
45+
new Promise(_resolve => {
46+
resolve = _resolve
47+
}),
48+
)
49+
}
50+
>
51+
Value: {value}
52+
</button>
53+
)
54+
}
55+
56+
const {container} = await render(
57+
<React.Suspense fallback="loading...">
58+
<Component />
59+
</React.Suspense>,
60+
)
61+
62+
expect(container).toHaveTextContent('Value: initial')
63+
64+
await fireEvent.click(container.querySelector('button'))
65+
66+
expect(container).toHaveTextContent('loading...')
67+
68+
await act(() => {
69+
resolve('Hello, Dave!')
70+
})
71+
72+
expect(container).toHaveTextContent('Hello, Dave!')
73+
})

src/__tests__/renderAsync.js

-25
This file was deleted.

src/act-compat.js

-14
Original file line numberDiff line numberDiff line change
@@ -82,22 +82,8 @@ function withGlobalActEnvironment(actImplementation) {
8282

8383
const act = withGlobalActEnvironment(reactAct)
8484

85-
async function actAsync(scope) {
86-
const previousActEnvironment = getIsReactActEnvironment()
87-
setIsReactActEnvironment(true)
88-
try {
89-
// React.act isn't async yet so we need to force it.
90-
return await reactAct(async () => {
91-
scope()
92-
})
93-
} finally {
94-
setIsReactActEnvironment(previousActEnvironment)
95-
}
96-
}
97-
9885
export default act
9986
export {
100-
actAsync,
10187
setIsReactActEnvironment as setReactActEnvironment,
10288
getIsReactActEnvironment,
10389
}

src/async.js

+42
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
/* istanbul ignore file */
2+
import {getIsReactActEnvironment, setReactActEnvironment} from './act-compat'
3+
import {cleanup} from './pure-async'
4+
5+
// if we're running in a test runner that supports afterEach
6+
// or teardown then we'll automatically run cleanup afterEach test
7+
// this ensures that tests run in isolation from each other
8+
// if you don't like this then either import the `pure` module
9+
// or set the RTL_SKIP_AUTO_CLEANUP env variable to 'true'.
10+
if (typeof process === 'undefined' || !process.env?.RTL_SKIP_AUTO_CLEANUP) {
11+
// ignore teardown() in code coverage because Jest does not support it
12+
/* istanbul ignore else */
13+
if (typeof afterEach === 'function') {
14+
afterEach(async () => {
15+
await cleanup()
16+
})
17+
} else if (typeof teardown === 'function') {
18+
// Block is guarded by `typeof` check.
19+
// eslint does not support `typeof` guards.
20+
// eslint-disable-next-line no-undef
21+
teardown(async () => {
22+
await cleanup()
23+
})
24+
}
25+
26+
// No test setup with other test runners available
27+
/* istanbul ignore else */
28+
if (typeof beforeAll === 'function' && typeof afterAll === 'function') {
29+
// This matches the behavior of React < 18.
30+
let previousIsReactActEnvironment = getIsReactActEnvironment()
31+
beforeAll(() => {
32+
previousIsReactActEnvironment = getIsReactActEnvironment()
33+
setReactActEnvironment(true)
34+
})
35+
36+
afterAll(() => {
37+
setReactActEnvironment(previousIsReactActEnvironment)
38+
})
39+
}
40+
}
41+
42+
export * from './pure-async'

src/fire-event-async.js

+70
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,70 @@
1+
/* istanbul ignore file */
2+
import {fireEvent as dtlFireEvent} from '@testing-library/dom'
3+
4+
// react-testing-library's version of fireEvent will call
5+
// dom-testing-library's version of fireEvent. The reason
6+
// we make this distinction however is because we have
7+
// a few extra events that work a bit differently
8+
const fireEvent = (...args) => dtlFireEvent(...args)
9+
10+
Object.keys(dtlFireEvent).forEach(key => {
11+
fireEvent[key] = (...args) => dtlFireEvent[key](...args)
12+
})
13+
14+
// React event system tracks native mouseOver/mouseOut events for
15+
// running onMouseEnter/onMouseLeave handlers
16+
// @link https://github.com/facebook/react/blob/b87aabdfe1b7461e7331abb3601d9e6bb27544bc/packages/react-dom/src/events/EnterLeaveEventPlugin.js#L24-L31
17+
const mouseEnter = fireEvent.mouseEnter
18+
const mouseLeave = fireEvent.mouseLeave
19+
fireEvent.mouseEnter = async (...args) => {
20+
await mouseEnter(...args)
21+
return fireEvent.mouseOver(...args)
22+
}
23+
fireEvent.mouseLeave = async (...args) => {
24+
await mouseLeave(...args)
25+
return fireEvent.mouseOut(...args)
26+
}
27+
28+
const pointerEnter = fireEvent.pointerEnter
29+
const pointerLeave = fireEvent.pointerLeave
30+
fireEvent.pointerEnter = async (...args) => {
31+
await pointerEnter(...args)
32+
return fireEvent.pointerOver(...args)
33+
}
34+
fireEvent.pointerLeave = async (...args) => {
35+
await pointerLeave(...args)
36+
return fireEvent.pointerOut(...args)
37+
}
38+
39+
const select = fireEvent.select
40+
fireEvent.select = async (node, init) => {
41+
await select(node, init)
42+
// React tracks this event only on focused inputs
43+
node.focus()
44+
45+
// React creates this event when one of the following native events happens
46+
// - contextMenu
47+
// - mouseUp
48+
// - dragEnd
49+
// - keyUp
50+
// - keyDown
51+
// so we can use any here
52+
// @link https://github.com/facebook/react/blob/b87aabdfe1b7461e7331abb3601d9e6bb27544bc/packages/react-dom/src/events/SelectEventPlugin.js#L203-L224
53+
await fireEvent.keyUp(node, init)
54+
}
55+
56+
// React event system tracks native focusout/focusin events for
57+
// running blur/focus handlers
58+
// @link https://github.com/facebook/react/pull/19186
59+
const blur = fireEvent.blur
60+
const focus = fireEvent.focus
61+
fireEvent.blur = async (...args) => {
62+
await fireEvent.focusOut(...args)
63+
return blur(...args)
64+
}
65+
fireEvent.focus = async (...args) => {
66+
await fireEvent.focusIn(...args)
67+
return focus(...args)
68+
}
69+
70+
export {fireEvent}

0 commit comments

Comments
 (0)