How to create a React Button

 by Robin Wieruch
 - Edit this Post

A button may be the first interactive element that you are using in a React component. Therefore, this is a short React tutorial by example for beginners about creating a button in React, how to use it, and how to extract it as a reusable component. First of all, a button is just an HTML button element which can be rendered in React's JSX:

import * as React from 'react';
const App = () => {
return (
<div>
<button type="button">Click Me</button>
</div>
);
};
export default App;

By using an , we can react to the button's click event:

import * as React from 'react';
const App = () => {
const handleClick = () => {
// implementation details
};
return (
<div>
<button type="button" onClick={handleClick}>
Click Me
</button>
</div>
);
};
export default App;

This way, a button can be used to trigger various effects when clicking it. For example, it can change a stateful value by using :

import * as React from 'react';
const App = () => {
const [count, setCount] = React.useState(0);
const handleClick = () => {
setCount(count + 1);
};
return (
<div>
<button type="button" onClick={handleClick}>
Click Me
</button>
{count}
</div>
);
};
export default App;

Another example would be toggling a :

import * as React from 'react';
const App = () => {
const [isOpen, setOpen] = React.useState(false);
const handleClick = () => {
setOpen(!isOpen);
};
return (
<div>
<button type="button" onClick={handleClick}>
Click Me
</button>
{isOpen && <div>Content</div>}
</div>
);
};
export default App;

Anyway, whatever this button is performing on its onClick handler after all, you may want to create a reusable component for it. Therefore, we will extract it as a new and to it:

import * as React from 'react';
const App = () => {
const [isOpen, setOpen] = React.useState(false);
const handleClick = () => {
setOpen(!isOpen);
};
return (
<div>
<Button onClick={handleClick}>Toggle</Button>
{isOpen && <div>Content</div>}
</div>
);
};
const Button = ({ onClick, children }) => {
return (
<button type="button" onClick={onClick}>
{children}
</button>
);
};
export default App;

Our Button component is a now. For example, if you would give your input field some , every Button component which is used in your React project would use the same style.

If you would want to create a button group now, you could just use multiple Button components side by side:

import * as React from 'react';
const App = () => {
const [isOpen, setOpen] = React.useState(false);
const handleOpen = () => {
setOpen(true);
};
const handleClose = () => {
setOpen(false);
};
return (
<div>
<Button onClick={handleOpen}>Open</Button>
<Button onClick={handleClose}>Close</Button>
{isOpen && <div>Content</div>}
</div>
);
};
const Button = ({ onClick, children }) => {
return (
<button type="button" onClick={onClick}>
{children}
</button>
);
};
export default App;

From here you can extend your Button component with many features. For example, the type is not always button, but can be submit for handling forms in React. Therefore we could enable developers from the outside to pass an optional type prop for the Button which defaults internally to button if nothing is passed:

const Button = ({ type = 'button', onClick, children }) => {
return (
<button type={type} onClick={onClick}>
{children}
</button>
);
};

Or you can pass a disabled prop to the Button component whenever it should not be clickable. Since it defaults to undefined if it is not provided, the button element will not be disabled:

const Button = ({ type = 'button', disabled, onClick, children }) => {
return (
<button type={type} disabled={disabled} onClick={onClick}>
{children}
</button>
);
};

However, passing all these extra information to the Button component is cumbersome. In a perfect world, all this information should be taken by the Button component the same way as if we would be using a mere button element. We can make this happen by using the JavaScript's rest destructuring for React's props:

const Button = ({ type = 'button', onClick, children, ...rest }) => {
return (
<button type={type} onClick={onClick} {...rest}>
{children}
</button>
);
};

Now, whether we pass a disabled boolean or a style object to the Button component, internally it will pass it to the button element. This way, the Button element behaves similar to the button element. Everything that's more explicit, like the onClick handler or the default value for the type prop, needs to be written explicitly in the Button component's function signature.

Keep reading about 

Here you will learn how to use a form button in React to trigger a server action in a Server Component without any form fields or form data. This can be useful if you want to trigger a server action…

In this React tutorial, we will get to know event handlers in React for HTML elements such as button and input elements. You will learn how to use a button with its onClick event and how to define and…

The Road to React

Learn React by building real world applications. No setup configuration. No tooling. Plain React in 200+ pages of learning material. Learn React like 50.000+ readers.

Get it on Amazon.