Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Bug fix - SetState callback called before component state is updated in ReactShallowRenderer #11507

Merged
merged 17 commits into from
Nov 22, 2017
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
Show all changes
17 commits
Select commit Hold shift + click to select a range
035eb3d
Create test to verify ReactShallowRenderer bug (#11496)
accordeiro Nov 9, 2017
856cb2b
Fix ReactShallowRenderer callback bug on componentWillMount (#11496)
accordeiro Nov 9, 2017
a269747
Improve fnction naming and clean up queued callback before call
accordeiro Nov 9, 2017
a13a3d6
Merge branch 'master' into shallow-renderer-bug
accordeiro Nov 9, 2017
11b931d
Run prettier on ReactShallowRenderer.js
accordeiro Nov 10, 2017
ae88abd
Consolidate callback call on ReactShallowRenderer.js
accordeiro Nov 10, 2017
87bf19f
Ensure callback behavior is similar between ReactDOM and ReactShallow…
accordeiro Nov 13, 2017
120b399
Fix Code Review requests (#11507)
accordeiro Nov 13, 2017
cfa6552
Move test to ReactCompositeComponent
gaearon Nov 13, 2017
ffa5565
Verify the callback gets called
gaearon Nov 13, 2017
f242683
Ensure multiple callbacks are correctly handled on ReactShallowRenderer
accordeiro Nov 19, 2017
de20f22
Merge branch 'shallow-renderer-bug' of github.com:accordeiro/react in…
accordeiro Nov 19, 2017
194ff74
Ensure the setState callback is called inside componentWillMount (Rea…
accordeiro Nov 19, 2017
2141b8c
Clear ReactShallowRenderer callback queue before actually calling the…
accordeiro Nov 22, 2017
b3b0788
Add test for multiple callbacks on ReactShallowRenderer
accordeiro Nov 22, 2017
0a58682
Ensure the ReactShallowRenderer callback queue is cleared after invok…
accordeiro Nov 22, 2017
ba0e995
Remove references to internal fields on ReactShallowRenderer test
accordeiro Nov 22, 2017
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
34 changes: 21 additions & 13 deletions packages/react-test-renderer/src/ReactShallowRenderer.js
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,7 @@ class ReactShallowRenderer {
'ReactShallowRenderer render(): Invalid component element.%s',
typeof element === 'function'
? ' Instead of passing a component class, make sure to instantiate ' +
'it by passing it to React.createElement.'
'it by passing it to React.createElement.'
: '',
);
// Show a special message for host elements since it's a common case.
Expand Down Expand Up @@ -139,6 +139,7 @@ class ReactShallowRenderer {
}

this._rendered = this._instance.render();
this._updater._invokeCallback();
// Intentionally do not call componentDidMount()
// because DOM refs are not available.
}
Expand Down Expand Up @@ -183,6 +184,7 @@ class ReactShallowRenderer {

if (shouldUpdate) {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does ReactDOM call the callback if shouldComponentUpdate skips an update? I would expect so, but this PR doesn't. Can you verify this?

Maybe it's better to move the callback call here. Then you don't need to duplicate it in two branches.

this._rendered = this._instance.render();
this._updater._invokeCallback();
}
// Intentionally do not call componentDidUpdate()
// because DOM refs are not available.
Expand All @@ -192,31 +194,41 @@ class ReactShallowRenderer {
class Updater {
constructor(renderer) {
this._renderer = renderer;
this._callback = null;
this._publicInstance = null;
}

_updateCallback(callback, publicInstance) {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe _enqueueCallback

if (typeof callback === 'function') {
this._callback = callback;
this._publicInstance = publicInstance;
}
}

_invokeCallback() {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe _invokeCallbackIfNecessary? It doesn't always exist.

if (typeof this._callback === 'function' && this._publicInstance) {
this._callback.call(this._publicInstance);
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You'll probably want to immediately set it to null before calling. Since now it is outdated.

}
}

isMounted(publicInstance) {
return !!this._renderer._element;
}

enqueueForceUpdate(publicInstance, callback, callerName) {
this._updateCallback(callback, publicInstance);
this._renderer._forcedUpdate = true;
this._renderer.render(this._renderer._element, this._renderer._context);

if (typeof callback === 'function') {
callback.call(publicInstance);
}
}

enqueueReplaceState(publicInstance, completeState, callback, callerName) {
this._updateCallback(callback, publicInstance);
this._renderer._newState = completeState;
this._renderer.render(this._renderer._element, this._renderer._context);

if (typeof callback === 'function') {
callback.call(publicInstance);
}
}

enqueueSetState(publicInstance, partialState, callback, callerName) {
this._updateCallback(callback, publicInstance);
const currentState = this._renderer._newState || publicInstance.state;

if (typeof partialState === 'function') {
Expand All @@ -229,10 +241,6 @@ class Updater {
};

this._renderer.render(this._renderer._element, this._renderer._context);

if (typeof callback === 'function') {
callback.call(publicInstance);
}
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -845,6 +845,34 @@ describe('ReactShallowRenderer', () => {
expect(result).toEqual(<div>baz:bar</div>);
});

it('this.state should be updated on setState callback inside componentWillMount', function() {
let stateSuccessfullyUpdated = false;

class Component extends React.Component {
constructor(props, context) {
super(props, context);
this.state = {
hasUpdatedState: false,
};
}

componentWillMount() {
this.setState(
{hasUpdatedState: true},
() => (stateSuccessfullyUpdated = this.state.hasUpdatedState),
);
}

render() {
return <div>{this.props.children}</div>;
}
}

const shallowRenderer = createRenderer();
shallowRenderer.render(<Component />);
expect(stateSuccessfullyUpdated).toBe(true);
});

it('throws usefully when rendering badly-typed elements', () => {
spyOn(console, 'error');
const shallowRenderer = createRenderer();
Expand Down