-
Notifications
You must be signed in to change notification settings - Fork 47.4k
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
Changes from 7 commits
035eb3d
856cb2b
a269747
a13a3d6
11b931d
ae88abd
87bf19f
120b399
cfa6552
ffa5565
f242683
de20f22
194ff74
2141b8c
b3b0788
0a58682
ba0e995
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -338,4 +338,45 @@ describe('ReactDOM', () => { | |
]; | ||
expect(actual).toEqual(expected); | ||
}); | ||
|
||
it('should call the setState callback even if shouldComponentUpdate = false', () => { | ||
const mockFn = jest.fn().mockReturnValue(false); | ||
let setState, getState; | ||
|
||
const div = document.createElement('div'); | ||
document.body.appendChild(div); | ||
|
||
class Component extends React.Component { | ||
constructor(props, context) { | ||
super(props, context); | ||
this.state = { | ||
hasUpdatedState: false, | ||
}; | ||
} | ||
|
||
componentWillMount() { | ||
setState = (newState, callback) => this.setState(newState, callback); | ||
getState = () => this.state; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. These helpers make it a bit hard to read what's really going on in the test. Could you please just store the instance instead, and then read |
||
} | ||
|
||
shouldComponentUpdate() { | ||
return mockFn(); | ||
} | ||
|
||
render() { | ||
return <div>{this.state.hasUpdatedState}</div>; | ||
} | ||
} | ||
|
||
ReactDOM.render(<Component />, div); | ||
|
||
expect(setState).toBeDefined(); | ||
expect(getState).toBeDefined(); | ||
expect(mockFn).not.toBeCalled(); | ||
|
||
setState({hasUpdatedState: true}, () => { | ||
expect(mockFn).toBeCalled(); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This doesn't verify that the callback itself has been called. |
||
expect(getState().hasUpdatedState).toBe(true); | ||
}); | ||
}); | ||
}); |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -103,6 +103,7 @@ class ReactShallowRenderer { | |
} | ||
|
||
this._rendering = false; | ||
this._updater._invokeCallback(); | ||
|
||
return this.getRenderOutput(); | ||
} | ||
|
@@ -192,31 +193,44 @@ class ReactShallowRenderer { | |
class Updater { | ||
constructor(renderer) { | ||
this._renderer = renderer; | ||
this._callback = null; | ||
this._publicInstance = null; | ||
} | ||
|
||
_enqueueCallback(callback, publicInstance) { | ||
if (typeof callback === 'function') { | ||
this._callback = callback; | ||
this._publicInstance = publicInstance; | ||
} | ||
} | ||
|
||
_invokeCallback() { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Maybe |
||
if (typeof this._callback === 'function' && this._publicInstance) { | ||
const {_callback, _publicInstance} = this; | ||
this._callback = null; | ||
this._publicInstance = null; | ||
_callback.call(_publicInstance); | ||
} | ||
} | ||
|
||
isMounted(publicInstance) { | ||
return !!this._renderer._element; | ||
} | ||
|
||
enqueueForceUpdate(publicInstance, callback, callerName) { | ||
this._enqueueCallback(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._enqueueCallback(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._enqueueCallback(callback, publicInstance); | ||
const currentState = this._renderer._newState || publicInstance.state; | ||
|
||
if (typeof partialState === 'function') { | ||
|
@@ -229,10 +243,6 @@ class Updater { | |
}; | ||
|
||
this._renderer.render(this._renderer._element, this._renderer._context); | ||
|
||
if (typeof callback === 'function') { | ||
callback.call(publicInstance); | ||
} | ||
} | ||
} | ||
|
||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think we need to add it to the body for this test.