Skip to content

Handler Function in JSX

We have learned a lot about React components, but there are no interactions yet. If you happen to develop an application with React, there will come a time where you have to implement a user interaction. The best place to get started in our project is the Search component -- which already comes with an input field element.

In native HTML, we can add event handlers on elements by using the addEventListener() method programmatically on a DOM node. In React, we are going to discover how to add handlers in JSX the declarative way. First, refactor the Search component's function from a concise body to a block body, so that we can add implementation details prior the return statement:

src/App.jsx

jsx
const Search = () => {
  // perform a task in between

  return (
    <div>
      <label htmlFor="search">Search: </label>
      <input id="search" type="text" />
    </div>
  );
};

Next, define a function, which can be either a function declaration or arrow function expression, for the change event of the input field. In React, this function is called an (event) handler. Afterward, the function can be passed to the onChange attribute (JSX named attribute) of the HTML input field:

src/App.jsx

jsx
const Search = () => {
  const handleChange = (event) => {
    // synthetic event
    console.log(event);
    // value of target (here: input HTML element)
    console.log(event.target.value);
  };

  return (
    <div>
      <label htmlFor="search">Search: </label>
      <input id="search" type="text" onChange={handleChange} />
    </div>
  );
};

After opening your application in a web browser, open the browser's developer tools "Console"-tab to see the logging occur after you type into the input field. What you see is called a synthetic event as a JavaScript object and the input field's internal value.

React's synthetic event is essentially a wrapper around the browser's native event. Since React started as a library for single-page applications, there was the need for enhanced functionalities on the event to prevent the native browser behavior. For example, in native HTML submitting a form triggers a page refresh. However, in React this page refresh should be prevented, because the developer should take care about what happens next. Anyway, if you happen to need access to the native HTML event, you could do so by using event.nativeEvent, but after several years of React development I never ran into this case myself.

After all, this is how we pass HTML elements in JSX handler functions to add listeners for user interactions. Always pass functions to these handlers, not the return value of the function, except when the return value is a function again. Knowing this is crucial because it's a well-known source for bugs in a React beginner's application:

Code Playground

js
// if handleChange is a function
// which does not return a function
// don't do this
<input onChange={handleChange()} />

// do this instead
<input onChange={handleChange} />

As you can see, HTML and JavaScript work well together in JSX. JavaScript in HTML can display JavaScript variables (e.g. title string in <span>{title}</span>), can pass JavaScript primitives to HTML attributes (e.g. url string to <a href={url}> HTML element), and can pass functions to an HTML element's attributes for handling user interactions (e.g. handleChange function to <input onChange={handleChange} />). When developing React applications, mixing HTML and JavaScript in JSX will become your bread and butter.

Exercises:

Interview Questions:

  • Question: How do you define an event handler in React?
    • Answer: Create a function that handles the event, like function handleClick() {...}.
  • Question: How do you attach an event handler in JSX?
    • Answer: Use the appropriate attribute, like onClick={handleClick}.
  • Question: What is the common pattern for naming event handler functions?
    • Answer: Prefix the function name with "handle" followed by the event name, like handleClick for a click event.
  • Question: Can you use arrow functions directly in the JSX for event handlers?
    • Answer: Yes, using arrow functions directly in JSX is a common pattern for concise event handlers.
  • Question: How do you pass arguments to an event handler in JSX?
    • Answer: Use an arrow function to call the handler with arguments, like onClick={() => handleClick(arg)}.
  • Question: Can you reuse event handlers across multiple elements?
    • Answer: Yes, event handlers can be reused for multiple elements with the same event type.
  • Question: What is the purpose of the e.target property in an event handler?
    • Answer: It refers to the DOM element that triggered the event, allowing you to access its properties or manipulate it.
  • Question: How do you access the event object in an event handler?
    • Answer: Include (event) as a parameter in the handler function, like function handleClick(event) {...}.
  • Question: What does event.preventDefault() do in an event handler?
    • Answer: It prevents the default behavior of the event, such as submitting a form or following a link.
  • Question: What is the purpose of the e.stopPropagation() method in an event handler?
    • Answer: It stops the event from propagating up or down the DOM tree, preventing parent or child elements from handling the same event.