mirror of
https://github.com/facebook/react.git
synced 2025-11-01 09:12:30 +00:00
6b30832666
The old version of prettier we were using didn't support the Flow syntax to access properties in a type using `SomeType['prop']`. This updates `prettier` and `rollup-plugin-prettier` to the latest versions. I added the prettier config `arrowParens: "avoid"` to reduce the diff size as the default has changed in Prettier 2.0. The largest amount of changes comes from function expressions now having a space. This doesn't have an option to preserve the old behavior, so we have to update this.
219 lines
6.5 KiB
JavaScript
219 lines
6.5 KiB
JavaScript
/**
|
|
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*
|
|
* @emails react-core
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
describe('ReactDOMComponentTree', () => {
|
|
let React;
|
|
let ReactDOM;
|
|
let container;
|
|
|
|
beforeEach(() => {
|
|
React = require('react');
|
|
ReactDOM = require('react-dom');
|
|
container = document.createElement('div');
|
|
document.body.appendChild(container);
|
|
});
|
|
|
|
afterEach(() => {
|
|
document.body.removeChild(container);
|
|
container = null;
|
|
});
|
|
|
|
it('finds nodes for instances on events', () => {
|
|
const mouseOverID = 'mouseOverID';
|
|
const clickID = 'clickID';
|
|
let currentTargetID = null;
|
|
// the current target of an event is set to result of getNodeFromInstance
|
|
// when an event is dispatched so we can test behavior by invoking
|
|
// events on elements in the tree and confirming the expected node is
|
|
// set as the current target
|
|
class Component extends React.Component {
|
|
handler = e => {
|
|
currentTargetID = e.currentTarget.id;
|
|
};
|
|
render() {
|
|
return (
|
|
<div id={mouseOverID} onMouseOver={this.handler}>
|
|
<div id={clickID} onClick={this.handler} />
|
|
</div>
|
|
);
|
|
}
|
|
}
|
|
|
|
function simulateMouseEvent(elem, type) {
|
|
const event = new MouseEvent(type, {
|
|
bubbles: true,
|
|
});
|
|
elem.dispatchEvent(event);
|
|
}
|
|
|
|
const component = <Component />;
|
|
ReactDOM.render(component, container);
|
|
expect(currentTargetID).toBe(null);
|
|
simulateMouseEvent(document.getElementById(mouseOverID), 'mouseover');
|
|
expect(currentTargetID).toBe(mouseOverID);
|
|
simulateMouseEvent(document.getElementById(clickID), 'click');
|
|
expect(currentTargetID).toBe(clickID);
|
|
});
|
|
|
|
it('finds closest instance for node when an event happens', () => {
|
|
const nonReactElemID = 'aID';
|
|
const innerHTML = {__html: `<div id="${nonReactElemID}"></div>`};
|
|
const closestInstanceID = 'closestInstance';
|
|
let currentTargetID = null;
|
|
|
|
class ClosestInstance extends React.Component {
|
|
_onClick = e => {
|
|
currentTargetID = e.currentTarget.id;
|
|
};
|
|
render() {
|
|
return (
|
|
<div
|
|
id={closestInstanceID}
|
|
onClick={this._onClick}
|
|
dangerouslySetInnerHTML={innerHTML}
|
|
/>
|
|
);
|
|
}
|
|
}
|
|
|
|
function simulateClick(elem) {
|
|
const event = new MouseEvent('click', {
|
|
bubbles: true,
|
|
});
|
|
elem.dispatchEvent(event);
|
|
}
|
|
|
|
const component = <ClosestInstance />;
|
|
ReactDOM.render(<section>{component}</section>, container);
|
|
expect(currentTargetID).toBe(null);
|
|
simulateClick(document.getElementById(nonReactElemID));
|
|
expect(currentTargetID).toBe(closestInstanceID);
|
|
});
|
|
|
|
it('updates event handlers from fiber props', () => {
|
|
let action = '';
|
|
let instance;
|
|
const handlerA = () => (action = 'A');
|
|
const handlerB = () => (action = 'B');
|
|
|
|
function simulateMouseOver(target) {
|
|
const event = new MouseEvent('mouseover', {
|
|
bubbles: true,
|
|
});
|
|
target.dispatchEvent(event);
|
|
}
|
|
|
|
class HandlerFlipper extends React.Component {
|
|
state = {flip: false};
|
|
flip() {
|
|
this.setState({flip: true});
|
|
}
|
|
render() {
|
|
return (
|
|
<div
|
|
id="update"
|
|
onMouseOver={this.state.flip ? handlerB : handlerA}
|
|
/>
|
|
);
|
|
}
|
|
}
|
|
|
|
ReactDOM.render(
|
|
<HandlerFlipper key="1" ref={n => (instance = n)} />,
|
|
container,
|
|
);
|
|
const node = container.firstChild;
|
|
simulateMouseOver(node);
|
|
expect(action).toEqual('A');
|
|
action = '';
|
|
// Render with the other event handler.
|
|
instance.flip();
|
|
simulateMouseOver(node);
|
|
expect(action).toEqual('B');
|
|
});
|
|
|
|
it('finds a controlled instance from node and gets its current fiber props', () => {
|
|
const inputID = 'inputID';
|
|
const startValue = undefined;
|
|
const finishValue = 'finish';
|
|
|
|
class Controlled extends React.Component {
|
|
state = {value: startValue};
|
|
a = null;
|
|
_onChange = e => this.setState({value: e.currentTarget.value});
|
|
render() {
|
|
return (
|
|
<input
|
|
id={inputID}
|
|
type="text"
|
|
ref={n => (this.a = n)}
|
|
value={this.state.value}
|
|
onChange={this._onChange}
|
|
/>
|
|
);
|
|
}
|
|
}
|
|
|
|
const setUntrackedInputValue = Object.getOwnPropertyDescriptor(
|
|
HTMLInputElement.prototype,
|
|
'value',
|
|
).set;
|
|
|
|
function simulateInput(elem, value) {
|
|
const inputEvent = new Event('input', {
|
|
bubbles: true,
|
|
});
|
|
setUntrackedInputValue.call(elem, value);
|
|
elem.dispatchEvent(inputEvent);
|
|
}
|
|
|
|
const component = <Controlled />;
|
|
const instance = ReactDOM.render(component, container);
|
|
expect(() => simulateInput(instance.a, finishValue)).toErrorDev(
|
|
'Warning: A component is changing an uncontrolled input to be controlled. ' +
|
|
'This is likely caused by the value changing from undefined to ' +
|
|
'a defined value, which should not happen. ' +
|
|
'Decide between using a controlled or uncontrolled input ' +
|
|
'element for the lifetime of the component. More info: ' +
|
|
'https://reactjs.org/link/controlled-components',
|
|
);
|
|
});
|
|
|
|
it('finds instance of node that is attempted to be unmounted', () => {
|
|
const component = <div />;
|
|
const node = ReactDOM.render(<div>{component}</div>, container);
|
|
expect(() => ReactDOM.unmountComponentAtNode(node)).toErrorDev(
|
|
"unmountComponentAtNode(): The node you're attempting to unmount " +
|
|
'was rendered by React and is not a top-level container. You may ' +
|
|
'have accidentally passed in a React root node instead of its ' +
|
|
'container.',
|
|
{withoutStack: true},
|
|
);
|
|
});
|
|
|
|
it('finds instance from node to stop rendering over other react rendered components', () => {
|
|
const component = (
|
|
<div>
|
|
<span>Hello</span>
|
|
</div>
|
|
);
|
|
const anotherComponent = <div />;
|
|
const instance = ReactDOM.render(component, container);
|
|
expect(() => ReactDOM.render(anotherComponent, instance)).toErrorDev(
|
|
'render(...): Replacing React-rendered children with a new root ' +
|
|
'component. If you intended to update the children of this node, ' +
|
|
'you should instead have the existing children update their state ' +
|
|
'and render the new components instead of calling ReactDOM.render.',
|
|
{withoutStack: true},
|
|
);
|
|
});
|
|
});
|