React Component Event Handlers
what should be put in the onClick={…}?
Update: read the revised version of this article from my blog at https://davidfeng.us/posts/react/event-handler/.
Let’s begin with a simple Toggle button example from React docs:
In the Toggle
class, a handleClick
method is defined as an instance method, which changes the local state. In the render
method, this.handleClick
is passed to the button
as an event handler. Finally, do not forget to bind this
to handleClick
in the constructor.
Note that in the onClick={...}
, only the function name (this.handleClick
) is passed. Do not invoke it, i.e. do not write onClick={this.handleClick()}
, since we do not want to change the state of a component while rendering it based on its state (more on that later). Also, this function has the synthetic event e
as a parameter (the first line of this function is usually e.preventDefault();
).
A more sophiscated example:
In CollisionViz, my recent web app which shows car crashes in NYC, I would like to have two buttons in my header, which open two modals respectively (more on modals in another story). To indicate which modal I want to show, in the state of the header component I created a currentModal
field, whose value could be ""
, "modal1"
, or "modal2"
(I could have used a boolean if I have only one modal). In the skeleton below, what should switchModal
do? How to pass it into the buttons’ onClick
handler and the modals’ closeModal
prop?
Phase 1:
1.1. Define three methods to set currentModal
to ""
, "modal1"
, or "modal2"
respectively. Bind them to this
in the constructor.
closeModal(e) {
e.preventDefault();
this.setState({currentModal: ""});
}showModal1(e) {
e.preventDefault();
this.setState({currentModal: "modal1"});
}showModal2(e) {
e.preventDefault();
this.setState({currentModal: "modal2"});
}
1.2. Pass these methods to the buttons and the modals.
<button onClick={this.showModal1}>
Open modal1
</button>
<button onClick={this.showModal2}>
Open modal2
</button>
<Modal
show={this.state.currentModal === 'modal1'}
closeModal={this.closeModal} >
This is modal1
</Modal>
...
Result: it works. However, obviously the code is not DRY. How can we improve?
Phase 2:
2.1. Define a single switchModal
method that takes a modalName
parameter. Bind it to this
in the constructor.
switchModal(modalName) {
this.setState({currentModal: modalName});
}
2.2. Pass the method with corresponding modal names to the buttons and the modals.
<button onClick={this.switchModal("modal1")}>
Open modal1
</button>
<button onClick={this.switchModal("modal2")}>
Open modal2
</button>
<Modal
show={this.state.currentModal === 'modal1'}
closeModal={this.switchModal("")} >
This is modal1
</Modal>
...
Result: it does not work. Why?

Remember that we should not invoke the function we pass in? That’s exactly what we did here. How can we fix this?
Phase 3:
Change the way we pass in the switchModal
function in Phase 2.2:
<button onClick={ e => this.switchModal("modal1") }>
Open modal1
</button>
<button onClick={ e => this.switchModal("modal2") }>
Open modal2
</button>
<Modal
show={this.state.currentModal === 'modal1'}
closeModal={ e => this.switchModal("") } >
This is modal1
</Modal>
Result: it works. What’s happening here is that we passed a function in the form of e => this.showModal(ModalName)
as event handlers without invoking switchModal
function. Notice the event handler function has an e
parameter, which is in line with the pattern in the previous Toggle
example.
Can we make the code more DRY?
Phase 4:
4.1. Change the switchModal
method so that it returns an event handler function:
switchModal(modalName) {
return ( e => {
e.preventDefault();
this.setState({shownModal: modalName});
});
}
4.2. Invoke switchModal
with different parameters when passing it to the buttons and modals as event handlers, which looks exactly the same as what we did in 2.2. However, this time by invoking switchModal
, we got the event handler back (without invoking it) as the return value.
<button onClick={this.switchModal("modal1")}>
Open modal1
</button>
<button onClick={this.switchModal("modal2")}>
Open modal2
</button>
<Modal
show={this.state.currentModal === 'modal1'}
closeModal={this.switchModal("")} >
This is modal1
</Modal>
...
Conclusions:
An event handler function has a synthetic event parameter e
, and it often has e.preventDefault()
and changes the state of a React component. When writing event handlers for a React component, e.g.
<Component onClick={...} ... />
Either write the event handler function’s name or invoke a function that returns an event handler function between the curly braces.