'use strict';
let React;
let ReactDOMClient;
let act;
describe('ReactDOMSafariMicrotaskBug-test', () => {
let container;
let flushMicrotasksPrematurely;
beforeEach(() => {
let queue = [];
window.queueMicrotask = function (cb) {
queue.push(cb);
};
flushMicrotasksPrematurely = function () {
while (queue.length > 0) {
const prevQueue = queue;
queue = [];
prevQueue.forEach(cb => cb());
}
};
jest.resetModules();
container = document.createElement('div');
React = require('react');
ReactDOMClient = require('react-dom/client');
act = require('internal-test-utils').act;
document.body.appendChild(container);
});
afterEach(() => {
document.body.removeChild(container);
});
it('should deal with premature microtask in commit phase', async () => {
let ran = false;
function Foo() {
const [state, setState] = React.useState(0);
return (
<div
ref={() => {
if (!ran) {
ran = true;
setState(1);
flushMicrotasksPrematurely();
}
}}>
{state}
</div>
);
}
const root = ReactDOMClient.createRoot(container);
await act(() => {
root.render(<Foo />);
});
expect(container.textContent).toBe('1');
});
it('should deal with premature microtask in event handler', async () => {
function Foo() {
const [state, setState] = React.useState(0);
return (
<button
onClick={() => {
setState(1);
flushMicrotasksPrematurely();
}}>
{state}
</button>
);
}
const root = ReactDOMClient.createRoot(container);
await act(() => {
root.render(<Foo />);
});
expect(container.textContent).toBe('0');
await act(() => {
container.firstChild.dispatchEvent(
new MouseEvent('click', {bubbles: true}),
);
});
expect(container.textContent).toBe('1');
});
});