Browse Source
Co-Authored-By: Dan Abramov <dan.abramov@gmail.com> Co-Authored-By: Sophie Alpert <git@sophiebits.com>main
Sophie Alpert
6 years ago
10 changed files with 2410 additions and 0 deletions
@ -0,0 +1,252 @@ |
|||
--- |
|||
id: hooks-custom |
|||
title: Writing Custom Hooks |
|||
permalink: docs/hooks-custom.html |
|||
next: hooks-reference.html |
|||
prev: hooks-rules.html |
|||
--- |
|||
|
|||
*Hooks* are a new feature proposal that lets you use state and other React features without writing a class. They're currently in React v16.7.0-alpha and being discussed in [an open RFC](https://github.com/reactjs/rfcs/pulls). |
|||
|
|||
Building your own Hooks lets you extract component logic into reusable functions. |
|||
|
|||
When we were learning about [using the Effect Hook](/docs/hooks-effect.html#example-using-hooks-1), we saw this component from a chat application that displays a message indicating whether a friend is online or offline: |
|||
|
|||
```js{4-15} |
|||
import { useState, useEffect } from 'react'; |
|||
|
|||
function FriendStatus(props) { |
|||
const [isOnline, setIsOnline] = useState(null); |
|||
|
|||
function handleStatusChange(status) { |
|||
setIsOnline(status.isOnline); |
|||
} |
|||
|
|||
useEffect(() => { |
|||
ChatAPI.subscribeToFriendStatus(props.friend.id, handleStatusChange); |
|||
return () => { |
|||
ChatAPI.unsubscribeFromFriendStatus(props.friend.id, handleStatusChange); |
|||
}; |
|||
}); |
|||
|
|||
if (isOnline === null) { |
|||
return 'Loading...'; |
|||
} |
|||
return isOnline ? 'Online' : 'Offline'; |
|||
} |
|||
``` |
|||
|
|||
Now let's say that our chat application also has a contact list, and we want to render names of online users with a green color. We could copy and paste similar logic above into our `FriendListItem` component but it wouldn't be ideal: |
|||
|
|||
```js{4-15} |
|||
import { useState, useEffect } from 'react'; |
|||
|
|||
function FriendListItem(props) { |
|||
const [isOnline, setIsOnline] = useState(null); |
|||
|
|||
function handleStatusChange(status) { |
|||
setIsOnline(status.isOnline); |
|||
} |
|||
|
|||
useEffect(() => { |
|||
ChatAPI.subscribeToFriendStatus(props.friend.id, handleStatusChange); |
|||
return () => { |
|||
ChatAPI.unsubscribeFromFriendStatus(props.friend.id, handleStatusChange); |
|||
}; |
|||
}); |
|||
|
|||
return ( |
|||
<li style={{ color: isOnline ? 'green' : 'black' }}> |
|||
{props.friend.name} |
|||
</li> |
|||
); |
|||
} |
|||
``` |
|||
|
|||
Instead, we'd like to share this logic between `FriendStatus` and `FriendListItem`. |
|||
|
|||
Traditionally in React, we've had two popular ways to share stateful logic between components: [render props](/docs/render-props.html) and [higher-order components](/docs/higher-order-components.html). We will now look at how Hooks solve many of the same problems without forcing you to add more components to the tree. |
|||
|
|||
## Extracting a Custom Hook |
|||
|
|||
When we want to share logic between two JavaScript functions, we extract it to a third function. Both components and Hooks are functions, so this works for them too! |
|||
|
|||
**A custom Hook is a JavaScript function whose name starts with "`use`" and that may call other Hooks.** For example, `useFriendStatus` below is our first custom Hook: |
|||
|
|||
```js{3} |
|||
import { useState, useEffect } from 'react'; |
|||
|
|||
function useFriendStatus(friendID) { |
|||
const [isOnline, setIsOnline] = useState(null); |
|||
|
|||
function handleStatusChange(status) { |
|||
setIsOnline(status.isOnline); |
|||
} |
|||
|
|||
useEffect(() => { |
|||
ChatAPI.subscribeToFriendStatus(friendID, handleStatusChange); |
|||
return () => { |
|||
ChatAPI.unsubscribeFromFriendStatus(friendID, handleStatusChange); |
|||
}; |
|||
}); |
|||
|
|||
return isOnline; |
|||
} |
|||
``` |
|||
|
|||
There's nothing new inside of it -- the logic is copied from the components above. Just like in a component, make sure to only call other Hooks unconditionally at the top level of your custom Hook. |
|||
|
|||
Unlike a React component, a custom Hook doesn't need to have a specific signature. We can decide what it takes as arguments, and what, if anything, it should return. In other words, it's just like a normal function. Its name should always start with `use` so that you can tell at a glance that the [rules of Hooks](/docs/hooks-rules.html) apply to it. |
|||
|
|||
The purpose of our `useFriendStatus` Hook is to subscribe us to a friend's status. This is why it takes `friendID` as an argument, and returns whether this friend is online: |
|||
|
|||
```js |
|||
function useFriendStatus(friendID) { |
|||
const [isOnline, setIsOnline] = useState(null); |
|||
|
|||
// ... |
|||
|
|||
return isOnline; |
|||
} |
|||
``` |
|||
|
|||
Now let's see how we can use our custom Hook. |
|||
|
|||
## Using a Custom Hook |
|||
|
|||
In the beginning, our stated goal was to remove the duplicated logic from the `FriendStatus` and `FriendListItem` components. Both of them want to know whether a friend is online. |
|||
|
|||
Now that we've extracted this logic to a `useFriendStatus` hook, we can *just use it:* |
|||
|
|||
```js{2} |
|||
function FriendStatus(props) { |
|||
const isOnline = useFriendStatus(props.friend.id); |
|||
|
|||
if (isOnline === null) { |
|||
return 'Loading...'; |
|||
} |
|||
return isOnline ? 'Online' : 'Offline'; |
|||
} |
|||
``` |
|||
|
|||
```js{2} |
|||
function FriendListItem(props) { |
|||
const isOnline = useFriendStatus(props.friend.id); |
|||
|
|||
return ( |
|||
<li style={{ color: isOnline ? 'green' : 'black' }}> |
|||
{props.friend.name} |
|||
</li> |
|||
); |
|||
} |
|||
``` |
|||
|
|||
**Is this code equivalent to the original examples?** Yes, it works in exactly the same way. If you look closely, you'll notice we didn't make any changes to the behavior. All we did was to extract some common code between two functions into a separate function. **Custom Hooks are a convention that naturally follows from the design of Hooks, rather than a React feature.** |
|||
|
|||
**Do I have to name my custom Hooks starting with “`use`”?** Please do. This convention is very important. Without it, we wouldn't be able to automatically check for violations of [rules of Hooks](/docs/hooks-rules.html) because we couldn't tell if a certain function contains calls to Hooks inside of it. |
|||
|
|||
**Do two components using the same Hook share state?** No. Custom Hooks are a mechanism to reuse *stateful logic* (such as setting up a subscription and remembering the current value), but every time you use a custom Hook, all state and effects inside of it are fully isolated. |
|||
|
|||
**How does a custom Hook get isolated state?** Each *call* to a Hook gets isolated state. Because we call `useFriendStatus` directly, from React's point of view our component just calls `useState` and `useEffect`. And as we [learned](/docs/hooks-state.html#tip-using-multiple-state-variables) [earlier](/docs/hooks-effect.html#tip-use-multiple-effects-to-separate-concerns), we can call `useState` and `useEffect` many times in one component, and they will be completely independent. |
|||
|
|||
### Tip: Pass Information Between Hooks |
|||
|
|||
Since Hooks are functions, we can pass information between them. |
|||
|
|||
To illustrate this, we'll use another component from our hypothetical chat example. This is a chat message recipient picker that displays whether the currently selected friend is online: |
|||
|
|||
```js{8-9,13} |
|||
const friendList = [ |
|||
{ id: 1, name: 'Phoebe' }, |
|||
{ id: 2, name: 'Rachel' }, |
|||
{ id: 3, name: 'Ross' }, |
|||
]; |
|||
|
|||
function ChatRecipientPicker() { |
|||
const [recipientID, setRecipientID] = useState(1); |
|||
const isRecipientOnline = useFriendStatus(recipientID); |
|||
|
|||
return ( |
|||
<> |
|||
<Circle color={isRecipientOnline ? 'green' : 'red'} /> |
|||
<select |
|||
value={recipientID} |
|||
onChange={e => setRecipientID(Number(e.target.value))} |
|||
> |
|||
{friendList.map(friend => ( |
|||
<option key={friend.id} value={friend.id}> |
|||
{friend.name} |
|||
</option> |
|||
))} |
|||
</select> |
|||
</> |
|||
); |
|||
} |
|||
``` |
|||
|
|||
We keep the currently chosen friend ID in the `recipientID` state variable, and update it if the user chooses a different friend in the `<select>` picker. |
|||
|
|||
Because the `useState` Hook call gives us the latest value of the `recipientID` state variable, we can pass it to our custom `useFriendStatus` Hook as an argument: |
|||
|
|||
```js |
|||
const [recipientID, setRecipientID] = useState(1); |
|||
const isRecipientOnline = useFriendStatus(recipientID); |
|||
``` |
|||
|
|||
This lets us know whether the *currently selected* friend is online. If we pick a different friend and update the `recipientID` state variable, our `useFriendStatus` Hook will unsubscribe from the previously selected friend, and subscribe to the status of the newly selected one. |
|||
|
|||
## `useYourImagination()` |
|||
|
|||
Custom Hooks offer the flexibility of sharing logic that wasn't possible in React components before. You can write custom Hooks that cover a wide range of use cases like form handling, animation, declarative subscriptions, timers, and probably many more we haven't considered. What's more, you can build Hooks that are just as easy to use as React's built-in features. |
|||
|
|||
Try to resist adding abstraction too early. Now that function components can do more, it's likely that the average function component in your codebase will become longer. This is normal -- don't feel like you *have to* immediately split it into Hooks. But we also encourage you to start spotting cases where a custom Hook could hide complex logic behind a simple interface, or help untangle a messy component. |
|||
|
|||
For example, maybe you have a complex component that contains a lot of local state that is managed in an ad-hoc way. `useState` doesn't make centralizing the update logic any easier so might you prefer to write it as a [Redux](https://redux.js.org/) reducer: |
|||
|
|||
```js |
|||
function todosReducer(state, action) { |
|||
switch (action.type) { |
|||
case 'add': |
|||
return [...state, { |
|||
text: action.text, |
|||
completed: false |
|||
}]; |
|||
// ... other actions ... |
|||
default: |
|||
return state; |
|||
} |
|||
} |
|||
``` |
|||
|
|||
Reducers are very convenient to test in isolation, and scale to express complex update logic. You can further break them apart into smaller reducers if necessary. However, you might also enjoy the benefits of using React local state, or might not want to install another library. |
|||
|
|||
So what if we could write a `useReducer` Hook that lets us manage the *local* state of our component with a reducer? A simplified version of it might look like this: |
|||
|
|||
```js |
|||
function useReducer(reducer, initialState) { |
|||
const [state, setState] = useState(initialState); |
|||
|
|||
function dispatch(action) { |
|||
const nextState = reducer(state, action); |
|||
setState(nextState); |
|||
} |
|||
|
|||
return [state, dispatch]; |
|||
} |
|||
``` |
|||
|
|||
Now we could use it in our component, and let the reducer drive its state management: |
|||
|
|||
```js{2} |
|||
function Todos() { |
|||
const [todos, dispatch] = useReducer(todosReducer, []); |
|||
|
|||
function handleAddClick(text) { |
|||
dispatch({ type: 'add', text }); |
|||
} |
|||
|
|||
// ... |
|||
} |
|||
``` |
|||
|
|||
The need to manage local state with a reducer in a complex component is common enough that we've built the `useReducer` Hook right into React. You'll find it together with other built-in Hooks in the [Hooks API reference](/docs/hooks-reference.html). |
@ -0,0 +1,475 @@ |
|||
--- |
|||
id: hooks-state |
|||
title: Using the Effect Hook |
|||
permalink: docs/hooks-effect.html |
|||
next: hooks-rules.html |
|||
prev: hooks-intro.html |
|||
--- |
|||
|
|||
*Hooks* are a new feature proposal that lets you use state and other React features without writing a class. They're currently in React v16.7.0-alpha and being discussed in [an open RFC](https://github.com/reactjs/rfcs/pulls). |
|||
|
|||
The *Effect Hook* lets you perform side effects in function components: |
|||
|
|||
```js{1,6-10} |
|||
import { useState, useEffect } from 'react'; |
|||
|
|||
function Example() { |
|||
const [count, setCount] = useState(0); |
|||
|
|||
// Similar to componentDidMount and componentDidUpdate: |
|||
useEffect(() => { |
|||
// Update the document title using the browser API |
|||
document.title = `You clicked ${count} times`; |
|||
}); |
|||
|
|||
return ( |
|||
<div> |
|||
<p>You clicked {count} times</p> |
|||
<button onClick={() => setCount(count + 1)}> |
|||
Click me |
|||
</button> |
|||
</div> |
|||
); |
|||
} |
|||
``` |
|||
|
|||
This snippet is based on the [counter example from the previous page](/docs/hooks-state.html), but we added a new feature to it: we set the document title to a custom message including the number of clicks. |
|||
|
|||
Data fetching, setting up a subscription, and manually changing the DOM in React components are all examples of side effects. Whether or not you're used to calling these operations "side effects" (or just "effects"), you've likely performed them in your components before. |
|||
|
|||
>Tip |
|||
> |
|||
>If you're familiar with React class lifecycle methods, you can think of `useEffect` Hook as `componentDidMount`, `componentDidUpdate`, and `componentWillUnmount` combined. |
|||
|
|||
There are two common kinds of side effects in React components: those that don't require cleanup, and those that do. Let's look at this distinction in more detail. |
|||
|
|||
## Effects Without Cleanup |
|||
|
|||
Sometimes, we want to **run some additional code after React has updated the DOM.** Network requests, manual DOM mutations, and logging are common examples of effects that don't require a cleanup. We say that because we can run them and immediately forget about them. Let's compare how classes and Hooks let us express such side effects. |
|||
|
|||
### Example Using Classes |
|||
|
|||
In React class components, the `render` method itself shouldn't cause side effects. It would be too early -- we typically want to perform our effects *after* React has updated the DOM. |
|||
|
|||
This is why in React classes, we put side effects into `componentDidMount` and `componentDidUpdate`. Coming back to our example, here is a React counter class component that updates the document title right after React makes changes to the DOM: |
|||
|
|||
```js{9-15} |
|||
class Example extends React.Component { |
|||
constructor(props) { |
|||
super(props); |
|||
this.state = { |
|||
count: 0 |
|||
}; |
|||
} |
|||
|
|||
componentDidMount() { |
|||
document.title = `You clicked ${this.state.count} times`; |
|||
} |
|||
|
|||
componentDidUpdate() { |
|||
document.title = `You clicked ${this.state.count} times`; |
|||
} |
|||
|
|||
render() { |
|||
return ( |
|||
<div> |
|||
<p>You clicked {this.state.count} times</p> |
|||
<button onClick={() => this.setState({ count: this.state.count + 1 })}> |
|||
Click me |
|||
</button> |
|||
</div> |
|||
); |
|||
} |
|||
} |
|||
``` |
|||
|
|||
Note how **we have to duplicate the code between these two lifecycle methods in class.** |
|||
|
|||
This is because in many cases we want to perform the same side effect regardless of whether the component just mounted, or if it has been updated. Conceptually, we want it to happen after every render -- but React class components don't have a method like this. We could extract a separate method but we would still have to call it in two places. |
|||
|
|||
Now let's see how we can do the same with the `useEffect` Hook. |
|||
|
|||
### Example Using Hooks |
|||
|
|||
We've already seen this example at the top of this page, but let's take a closer look at it: |
|||
|
|||
```js{1,6-8} |
|||
import { useState, useEffect } from 'react'; |
|||
|
|||
function Example() { |
|||
const [count, setCount] = useState(0); |
|||
|
|||
useEffect(() => { |
|||
document.title = `You clicked ${count} times`; |
|||
}); |
|||
|
|||
return ( |
|||
<div> |
|||
<p>You clicked {count} times</p> |
|||
<button onClick={() => setCount(count + 1)}> |
|||
Click me |
|||
</button> |
|||
</div> |
|||
); |
|||
} |
|||
``` |
|||
|
|||
**What does `useEffect` do?** By using this Hook, you tell React that your component needs to do something after render. React will remember the function you passed (we'll refer to it as our "effect"), and call it later after performing the DOM updates. In this effect, we set the document title, but we could also perform data fetching or call some other imperative API. |
|||
|
|||
**Why is `useEffect` called inside a component?** Placing `useEffect` inside the component lets us access the `count` state variable (or any props) right from the effect. We don't need a special API to read it -- it's already in the function scope. Hooks embrace JavaScript closures and avoid introducing React-specific APIs where JavaScript already provides a solution. |
|||
|
|||
**Does `useEffect` run after every render?** Yes! By default, it runs both after the first render *and* after every update. (We will later talk about [how to customize this](#tip-optimizing-performance-by-skipping-effects).) Instead of thinking in terms of "mounting" and "updating", you might find it easier to think that effects happen "after render". React guarantees the DOM has been updated by the time it runs the effects. |
|||
|
|||
### Detailed Explanation |
|||
|
|||
Now that we know more about effects, these lines should make sense: |
|||
|
|||
```js |
|||
function Example() { |
|||
const [count, setCount] = useState(0); |
|||
|
|||
useEffect(() => { |
|||
document.title = `You clicked ${count} times`; |
|||
}); |
|||
``` |
|||
|
|||
We declare the `count` state variable, and then we tell React we need to use an effect. We pass a function to the `useEffect` Hook. This function we pass *is* our effect. Inside our effect, we set the document title using the `document.title` browser API. We can read the latest `count` inside the effect because it's in the scope of our function. When React renders our component, it will remember the effect we used, and then run our effect after updating the DOM. This happens for every render, including the first one. |
|||
|
|||
Experienced JavaScript developers might notice that the function passed to `useEffect` is going to be different on every render. This is intentional. In fact, this is what lets us read the `count` value from inside the effect without worrying about it getting stale. Every time we re-render, we schedule a _different_ effect, replacing the previous one. In a way, this makes the effects behave more like a part of the render result -- each effect "belongs" to a particular render. We will see more clearly why this is useful [later on this page](#explanation-why-effects-run-on-each-update). |
|||
|
|||
>Tip |
|||
> |
|||
>Unlike `componentDidMount` or `componentDidUpdate`, effects scheduled with `useEffect` don't block the browser from updating the screen. This makes your app feel more responsive. The majority of effects don't need to happen synchronously. In the uncommon cases where they do (such as measuring the layout), there is a separate [`useLayoutEffect`](/docs/hooks-reference.html#uselayouteffect) Hook with an API identical to `useEffect`. |
|||
|
|||
## Effects with Cleanup |
|||
|
|||
Earlier, we looked at how to express side effects that don't require any cleanup. However, some effects do. For example, **we might want to set up a subscription** to some external data source. In that case, it is important to clean up so that we don't introduce a memory leak! Let's compare how we can do it with classes and with Hooks. |
|||
|
|||
### Example Using Classes |
|||
|
|||
In a React class, you would typically set up a subscription in `componentDidMount`, and clean it up in `componentWillUnmount`. For example, let's say we have a `ChatAPI` module that lets us subscribe to a friend's online status. Here's how we might subscribe and display that status using a class: |
|||
|
|||
```js{8-26} |
|||
class FriendStatus extends React.Component { |
|||
constructor(props) { |
|||
super(props); |
|||
this.state = { isOnline: null }; |
|||
this.handleStatusChange = this.handleStatusChange.bind(this); |
|||
} |
|||
|
|||
componentDidMount() { |
|||
ChatAPI.subscribeToFriendStatus( |
|||
this.props.friend.id, |
|||
this.handleStatusChange |
|||
); |
|||
} |
|||
|
|||
componentWillUnmount() { |
|||
ChatAPI.unsubscribeFromFriendStatus( |
|||
this.props.friend.id, |
|||
this.handleStatusChange |
|||
); |
|||
} |
|||
|
|||
handleStatusChange(status) { |
|||
this.setState({ |
|||
isOnline: status.isOnline |
|||
}); |
|||
} |
|||
|
|||
render() { |
|||
if (this.state.isOnline === null) { |
|||
return 'Loading...'; |
|||
} |
|||
return this.state.isOnline ? 'Online' : 'Offline'; |
|||
} |
|||
} |
|||
``` |
|||
|
|||
Notice how `componentDidMount` and `componentWillUnmount` need to mirror each other. Lifecycle methods force us to split this logic even though conceptually code in both of them is related to the same effect. |
|||
|
|||
>Note |
|||
> |
|||
>Eagle-eyed readers may notice that this example also needs a `componentDidUpdate` method to be fully correct. We'll ignore this for now but will come back to it in a [later section](#explanation-why-effects-run-on-each-update) of this page. |
|||
|
|||
### Example Using Hooks |
|||
|
|||
Let's see how we could write this component with Hooks. |
|||
|
|||
You might be thinking that we'd need a separate effect to perform the cleanup. But code for adding and removing a subscription is so tightly related that `useEffect` is designed to keep it together. If your effect returns a function, React will run it when it is time to clean up: |
|||
|
|||
```js{10-16} |
|||
import { useState, useEffect } from 'react'; |
|||
|
|||
function FriendStatus(props) { |
|||
const [isOnline, setIsOnline] = useState(null); |
|||
|
|||
function handleStatusChange(status) { |
|||
setIsOnline(status.isOnline); |
|||
} |
|||
|
|||
useEffect(() => { |
|||
ChatAPI.subscribeToFriendStatus(props.friend.id, handleStatusChange); |
|||
// Specify how to clean up after this effect: |
|||
return function cleanup() { |
|||
ChatAPI.unsubscribeFromFriendStatus(props.friend.id, handleStatusChange); |
|||
}; |
|||
}); |
|||
|
|||
if (isOnline === null) { |
|||
return 'Loading...'; |
|||
} |
|||
return isOnline ? 'Online' : 'Offline'; |
|||
} |
|||
``` |
|||
|
|||
**Why did we return a function from our effect?** This is the optional cleanup mechanism for effects. Every effect may return a function that cleans up after it. This lets us keep the logic for adding and removing subscriptions close to each other. They're part of the same effect! |
|||
|
|||
**When exactly does React clean up an effect?** React performs the cleanup when the component unmounts. However, as we learned earlier, effects run for every render and not just once. This is why React *also* cleans up effects from the previous render before running the effects next time. We'll discuss [why this helps avoid bugs](#explanation-why-effects-run-on-each-update) and [how to opt out of this behavior in case it creates performance issues](#tip-optimizing-performance-by-skipping-effects) later below. |
|||
|
|||
>Note |
|||
> |
|||
>We don't have to return a named function from the effect. We called it `cleanup` here to clarify its purpose, but you could return an arrow function or call it something different. |
|||
|
|||
## Recap |
|||
|
|||
We've learned that `useEffect` lets us express different kinds of side effects after a component renders. Some effects might require cleanup so they return a function: |
|||
|
|||
```js |
|||
useEffect(() => { |
|||
ChatAPI.subscribeToFriendStatus(props.friend.id, handleStatusChange); |
|||
return () => { |
|||
ChatAPI.unsubscribeFromFriendStatus(props.friend.id, handleStatusChange); |
|||
}; |
|||
}); |
|||
``` |
|||
|
|||
Other effects might not have a cleanup phase, and don't return anything. |
|||
|
|||
```js |
|||
useEffect(() => { |
|||
document.title = `You clicked ${count} times`; |
|||
}); |
|||
``` |
|||
|
|||
The Effect Hook unifies both use cases with a single API. |
|||
|
|||
------------- |
|||
|
|||
**If you feel like you have a decent grasp on how the Effect Hook works, or if you feel overwhelmed, you can jump to the [next page about Rules of Hooks](/docs/hooks-rules.html) now.** |
|||
|
|||
------------- |
|||
|
|||
## Tips for Using Effects |
|||
|
|||
We'll continue this page with an in-depth look at some aspects of `useEffect` that experienced React users will likely be curious about. Don't feel obligated to dig into them now. You can always come back to this page to learn more details about the Effect Hook. |
|||
|
|||
### Tip: Use Multiple Effects to Separate Concerns |
|||
|
|||
One of the problems we outlined in the [Motivation](/docs/hooks-intro.html#complex-components-become-hard-to-understand) for Hooks is that class lifecycle methods often contain unrelated logic, but related logic gets broken up into several methods. Here is a component that combines the counter and the friend status indicator logic from the previous examples: |
|||
|
|||
```js |
|||
class FriendStatusWithCounter extends React.Component { |
|||
constructor(props) { |
|||
super(props); |
|||
this.state = { count: 0, isOnline: null }; |
|||
this.handleStatusChange = this.handleStatusChange.bind(this); |
|||
} |
|||
|
|||
componentDidMount() { |
|||
document.title = `You clicked ${this.state.count} times`; |
|||
ChatAPI.subscribeToFriendStatus( |
|||
this.props.friend.id, |
|||
this.handleStatusChange |
|||
); |
|||
} |
|||
|
|||
componentDidUpdate() { |
|||
document.title = `You clicked ${this.state.count} times`; |
|||
} |
|||
|
|||
componentWillUnmount() { |
|||
ChatAPI.unsubscribeFromFriendStatus( |
|||
this.props.friend.id, |
|||
this.handleStatusChange |
|||
); |
|||
} |
|||
|
|||
handleStatusChange(status) { |
|||
this.setState({ |
|||
isOnline: status.isOnline |
|||
}); |
|||
} |
|||
// ... |
|||
``` |
|||
|
|||
Note how the logic that sets `document.title` is split between `componentDidMount` and `componentDidUpdate`. The subscription logic is also spread between `componentDidMount` and `componentWillUnmount`. And `componentDidMount` contains code for both tasks. |
|||
|
|||
So, how can Hooks solve this problem? Just like [you can use the *State* Hook more than once](/docs/hooks-state.html#tip-using-multiple-state-variables), you can also use several effects. This lets us separate unrelated logic into different effects: |
|||
|
|||
```js{3,8} |
|||
function FriendStatusWithCounter(props) { |
|||
const [count, setCount] = useState(0); |
|||
useEffect(() => { |
|||
document.title = `You clicked ${count} times`; |
|||
}); |
|||
|
|||
const [isOnline, setIsOnline] = useState(null); |
|||
useEffect(() => { |
|||
ChatAPI.subscribeToFriendStatus(props.friend.id, handleStatusChange); |
|||
return () => { |
|||
ChatAPI.unsubscribeFromFriendStatus(props.friend.id, handleStatusChange); |
|||
}; |
|||
}); |
|||
|
|||
function handleStatusChange(status) { |
|||
setIsOnline(status.isOnline); |
|||
} |
|||
// ... |
|||
} |
|||
``` |
|||
|
|||
**Hooks lets us split the code based on what it is doing** rather than a lifecycle method name. React will apply *every* effect used by the component, in the order they were specified. |
|||
|
|||
### Explanation: Why Effects Run on Each Update |
|||
|
|||
If you're used to classes, you might be wondering why the effect cleanup phase happens after every re-render, and not just once during unmounting. Let's look at a practical example to see why this design helps us create components with fewer bugs. |
|||
|
|||
[Earlier on this page](#example-using-classes-1), we introduced an example `FriendStatus` component that displays whether a friend is online or not. Our class reads `friend.id` from `this.props`, subscribes to the friend status after the component mounts, and unsubscribes during unmounting: |
|||
|
|||
```js |
|||
componentDidMount() { |
|||
ChatAPI.subscribeToFriendStatus( |
|||
this.props.friend.id, |
|||
this.handleStatusChange |
|||
); |
|||
} |
|||
|
|||
componentWillUnmount() { |
|||
ChatAPI.unsubscribeFromFriendStatus( |
|||
this.props.friend.id, |
|||
this.handleStatusChange |
|||
); |
|||
} |
|||
``` |
|||
|
|||
**But what happens if the `friend` prop changes** while the component is on the screen? Our component would continue displaying the online status of a different friend. This is a bug. We would also cause a memory leak or crash when unmounting since the unsubscribe call would use the wrong friend ID. |
|||
|
|||
In a class component, we would need to add `componentDidUpdate` to handle this case: |
|||
|
|||
```js{8-19} |
|||
componentDidMount() { |
|||
ChatAPI.subscribeToFriendStatus( |
|||
this.props.friend.id, |
|||
this.handleStatusChange |
|||
); |
|||
} |
|||
|
|||
componentDidUpdate(prevProps) { |
|||
// Unsubscribe from the previous friend.id |
|||
ChatAPI.unsubscribeFromFriendStatus( |
|||
prevProps.friend.id, |
|||
this.handleStatusChange |
|||
); |
|||
// Subscribe to the next friend.id |
|||
ChatAPI.subscribeToFriendStatus( |
|||
this.props.friend.id, |
|||
this.handleStatusChange |
|||
); |
|||
} |
|||
|
|||
componentWillUnmount() { |
|||
ChatAPI.unsubscribeFromFriendStatus( |
|||
this.props.friend.id, |
|||
this.handleStatusChange |
|||
); |
|||
} |
|||
``` |
|||
|
|||
Forgetting to handle `componentDidUpdate` properly is a common source of bugs in React applications. |
|||
|
|||
Now consider the version of this component that uses Hooks: |
|||
|
|||
```js |
|||
function FriendStatus(props) { |
|||
// ... |
|||
useEffect(() => { |
|||
ChatAPI.subscribeToFriendStatus(props.friend.id, handleStatusChange); |
|||
return () => { |
|||
ChatAPI.unsubscribeFromFriendStatus(props.friend.id, handleStatusChange); |
|||
}; |
|||
}); |
|||
``` |
|||
|
|||
It doesn't suffer from this bug. (But we also didn't make any changes to it.) |
|||
|
|||
There is no special code for handling updates because `useEffect` handles them *by default*. It cleans up the previous effects before applying the next effects. To illustrate this, here is a sequence of subscribe and unsubscribe calls that this component could produce over time: |
|||
|
|||
```js |
|||
// Mount with { friend: { id: 100 } } props |
|||
ChatAPI.subscribeToFriendStatus(100, handleStatusChange); // Run first effect |
|||
|
|||
// Update with { friend: { id: 200 } } props |
|||
ChatAPI.unsubscribeFromFriendStatus(100, handleStatusChange); // Clean up previous effect |
|||
ChatAPI.subscribeToFriendStatus(200, handleStatusChange); // Run next effect |
|||
|
|||
// Update with { friend: { id: 300 } } props |
|||
ChatAPI.unsubscribeFromFriendStatus(200, handleStatusChange); // Clean up previous effect |
|||
ChatAPI.subscribeToFriendStatus(300, handleStatusChange); // Run next effect |
|||
|
|||
// Unmount |
|||
ChatAPI.unsubscribeFromFriendStatus(300, handleStatusChange); // Clean up last effect |
|||
``` |
|||
|
|||
This behavior ensures consistency by default and prevents bugs that are common in class components due to missing update logic. |
|||
|
|||
### Tip: Optimizing Performance by Skipping Effects |
|||
|
|||
In some cases, cleaning up or applying the effect after every render might create a performance problem. In class components, we can solve this by writing an extra comparison with `prevProps` or `prevState` inside `componentDidUpdate`: |
|||
|
|||
```js |
|||
componentDidUpdate(prevProps, prevState) { |
|||
if (prevState.count !== this.state.count) { |
|||
document.title = `You clicked ${this.state.count} times`; |
|||
} |
|||
} |
|||
``` |
|||
|
|||
This requirement is common enough that it is built into the `useEffect` Hook API. You can tell React to *skip* applying an effect if certain values haven't changed between re-renders. To do so, pass an array as an optional second argument to `useEffect`: |
|||
|
|||
```js{3} |
|||
useEffect(() => { |
|||
document.title = `You clicked ${count} times`; |
|||
}, [count]); // Only re-run the effect if count changes |
|||
``` |
|||
|
|||
In the example above, we pass `[count]` as the second argument. What does this mean? If the `count` is `5`, and then our component re-renders with `count` still equal to `5`, React will compare `[5]` from the previous render and `[5]` from the next render. Because all items in the array are the same (`5 === 5`), React would skip the effect. That's our optimization. |
|||
|
|||
When we render with `count` updated to `6`, React will compare the items in the `[5]` array from the previous render to items in the `[6]` array from the next render. This time, React will re-apply the effect because `5 !== 6`. If there are multiple items in the array, React will re-run the effect even if just one of them is different. |
|||
|
|||
This also works for effects that have a cleanup phase: |
|||
|
|||
```js{6} |
|||
useEffect(() => { |
|||
ChatAPI.subscribeToFriendStatus(props.friend.id, handleStatusChange); |
|||
return () => { |
|||
ChatAPI.unsubscribeFromFriendStatus(props.friend.id, handleStatusChange); |
|||
}; |
|||
}, [props.friend.id]); // Only re-subscribe if props.friend.id changes |
|||
``` |
|||
|
|||
In the future, the second argument might get added automatically by a build-time transformation. |
|||
|
|||
>Note |
|||
> |
|||
>If you use this optimization, make sure the array includes **any values from the outer scope that change over time and that are used by the effect**. Otherwise, your code will reference stale values from previous renders. We'll also discuss other optimization options in the [Hooks API reference](/docs/hooks-reference.html). |
|||
> |
|||
>If you want to run an effect and clean it up only once (on mount and unmount), you can pass an empty array (`[]`) as a second argument. This tells React that your effect doesn't depend on *any* values from props or state, so it never needs to re-run. This isn't handled as a special case -- it follows directly from how the inputs array always works. While passing `[]` is closer to the familiar `componentDidMount` and `componentWillUnmount` mental model, we suggest not making it a habit because it often leads to bugs, [as discussed above](#explanation-why-effects-run-on-each-update). Don't forget that React defers running `useEffect` until after the browser has painted, so doing extra work is less of a problem. |
|||
|
|||
## Next Steps |
|||
|
|||
Congratulations! This was a long page, but hopefully by the end most of your questions about effects were answered. You've learned both the State Hook and the Effect Hook, and there is a *lot* you can do with both of them combined. They cover most of the use cases for classes -- and where they don't, you might find the [additional Hooks](/docs/hooks-reference.html) helpful. |
|||
|
|||
We're also starting to see how Hooks solve problems outlined in [Motivation](/docs/hooks-intro.html#motivation). We've seen how effect cleanup avoids duplication in `componentDidUpdate` and `componentWillUnmount`, brings related code closer together, and helps us avoid bugs. We've also seen how we can separate effects by their purpose, which is something we couldn't do in classes at all. |
|||
|
|||
At this point you might be questioning how Hooks work. How can React know which `useState` call corresponds to which state variable between re-renders? How does React "match up" previous and next effects on every update? **On the next page we will learn about the [Rules of Hooks](/docs/hooks-rules.html) -- they're essential to making Hooks work.** |
@ -0,0 +1,518 @@ |
|||
--- |
|||
id: hooks-faq |
|||
title: Hooks FAQ |
|||
permalink: docs/hooks-faq.html |
|||
prev: hooks-reference.html |
|||
--- |
|||
|
|||
*Hooks* are a new feature proposal that lets you use state and other React features without writing a class. They're currently in React v16.7.0-alpha and being discussed in [an open RFC](https://github.com/reactjs/rfcs/pulls). |
|||
|
|||
This page answers some of the frequently asked questions about [Hooks](/docs/hooks-overview.html). |
|||
|
|||
<!-- |
|||
if you ever need to regenerate this, this snippet in the devtools console might help: |
|||
|
|||
$$('.anchor').map(a => |
|||
`${' '.repeat(2 * +a.parentNode.nodeName.slice(1))}` + |
|||
`[${a.parentNode.textContent}](${a.getAttribute('href')})` |
|||
).join('\n') |
|||
--> |
|||
|
|||
* **[Adoption Strategy](#adoption-strategy)** |
|||
* [Do I need to rewrite all my class components?](#do-i-need-to-rewrite-all-my-class-components) |
|||
* [How much of my React knowledge stays relevant?](#how-much-of-my-react-knowledge-stays-relevant) |
|||
* [Should I use Hooks, classes, or a mix of both?](#should-i-use-hooks-classes-or-a-mix-of-both) |
|||
* [Do Hooks cover all use cases for classes?](#do-hooks-cover-all-use-cases-for-classes) |
|||
* [Do Hooks replace render props and higher-order components?](#do-hooks-replace-render-props-and-higher-order-components) |
|||
* [What do Hooks mean for popular APIs like Redux connect() and React Router?](#what-do-hooks-mean-for-popular-apis-like-redux-connect-and-react-router) |
|||
* [Do Hooks work with static typing?](#do-hooks-work-with-static-typing) |
|||
* [How to test components that use Hooks?](#how-to-test-components-that-use-hooks) |
|||
* [What exactly do the lint rules enforce?](#what-exactly-do-the-lint-rules-enforce) |
|||
* **[From Classes to Hooks](#from-classes-to-hooks)** |
|||
* [How do lifecycle methods correspond to Hooks?](#how-do-lifecycle-methods-correspond-to-hooks) |
|||
* [Is there something like instance variables?](#is-there-something-like-instance-variables) |
|||
* [Should I use one or many state variables?](#should-i-use-one-or-many-state-variables) |
|||
* [Can I run an effect only on updates?](#can-i-run-an-effect-only-on-updates) |
|||
* [How to get the previous props or state?](#how-to-get-the-previous-props-or-state) |
|||
* [How do I implement getDerivedStateFromProps?](#how-do-i-implement-getderivedstatefromprops) |
|||
* [Can I make a ref to a function component?](#can-i-make-a-ref-to-a-function-component) |
|||
* [What does const [thing, setThing] = useState() mean?](#what-does-const-thing-setthing--usestate-mean) |
|||
* **[Performance Optimizations](#performance-optimizations)** |
|||
* [Can I skip an effect on updates?](#can-i-skip-an-effect-on-updates) |
|||
* [How do I implement shouldComponentUpdate?](#how-do-i-implement-shouldcomponentupdate) |
|||
* [How to memoize calculations?](#how-to-memoize-calculations) |
|||
* [Are Hooks slow because of creating functions in render?](#are-hooks-slow-because-of-creating-functions-in-render) |
|||
* [How to avoid passing callbacks down?](#how-to-avoid-passing-callbacks-down) |
|||
* [How to read an often-changing value from useCallback?](#how-to-read-an-often-changing-value-from-usecallback) |
|||
* **[Under the Hood](#under-the-hood)** |
|||
* [How does React associate Hook calls with components?](#how-does-react-associate-hook-calls-with-components) |
|||
* [What is the prior art for Hooks?](#what-is-the-prior-art-for-hooks) |
|||
|
|||
## Adoption Strategy |
|||
|
|||
### Do I need to rewrite all my class components? |
|||
|
|||
No. There are [no plans](/docs/hooks-intro.html#gradual-adoption-strategy) to remove classes from React -- we all need to keep shipping products and can't afford rewrites. We recommend trying Hooks in new code. |
|||
|
|||
### How much of my React knowledge stays relevant? |
|||
|
|||
Hooks are a more direct way to use the React features you already know -- such as state, lifecycle, context, and refs. They don't fundamentally change how React works, and your knowledge of components, props, and top-down data flow is just as relevant. |
|||
|
|||
Hooks do have a learning curve of their own. If there's something missing in this documentation, [raise an issue](https://github.com/reactjs/reactjs.org/issues/new) and we'll try to help. |
|||
|
|||
### Should I use Hooks, classes, or a mix of both? |
|||
|
|||
When you're ready, we'd encourage you to start trying Hooks in new components you write. Make sure everyone on your team is on board with using them and familiar with this documentation. We don't recommend rewriting your existing classes to Hooks unless you planned to rewrite them anyway (e.g. to fix bugs). |
|||
|
|||
You can't use Hooks *inside* of a class component, but you can definitely mix classes and function components with Hooks in a single tree. Whether a component is a class or a function that uses Hooks is an implementation detail of that component. In the longer term, we expect Hooks to be the primary way people write React components. |
|||
|
|||
### Do Hooks cover all use cases for classes? |
|||
|
|||
Our goal is for Hooks to cover all use cases for classes as soon as possible. There are no Hook equivalents to the uncommon `getSnapshotBeforeUpdate` and `componentDidCatch` lifecycles yet, but we plan to add them soon. |
|||
|
|||
It is a very early time for Hooks, so some integrations like DevTools support or Flow/TypeScript typings may not be ready yet. Some third-party libraries might also not be compatible with Hooks at the moment. |
|||
|
|||
### Do Hooks replace render props and higher-order components? |
|||
|
|||
Often, render props and higher-order components render only a single child. We think Hooks are a simpler way to serve this use case. There is still a place for both patterns (for example, a virtual scroller component might have a `renderItem` prop, or a visual container component might have its own DOM structure). But in most cases, Hooks will be sufficient and can help reduce nesting in your tree. |
|||
|
|||
### What do Hooks mean for popular APIs like Redux `connect()` and React Router? |
|||
|
|||
You can continue to use the exact same APIs as you always have; they'll continue to work. |
|||
|
|||
In the future, new versions of these libraries might also export custom Hooks such as `useRedux()` or `useRouter()` that let you use the same features without needing wrapper components. |
|||
|
|||
### Do Hooks work with static typing? |
|||
|
|||
Hooks were designed with static typing in mind. Because they're functions, they are easier to type correctly than patterns like higher-order components. We have reached out both to Flow and TypeScript teams in advance, and they plan to include definitions for React Hooks in the future. |
|||
|
|||
Importantly, custom Hooks give you the power to constrain React API if you'd like to type them more strictly in some way. React gives you the primitives, but you can combine them in different ways than what we provide out of the box. |
|||
|
|||
### How to test components that use Hooks? |
|||
|
|||
From React's point of view, a component using Hooks is just a regular component. If your testing solution doesn't rely on React internals, testing components with Hooks shouldn't be different from how you normally test components. |
|||
|
|||
If you need to test a custom Hook, you can do so by creating a component in your test, and using your Hook from it. Then you can test the component you wrote. |
|||
|
|||
### What exactly do the [lint rules](https://www.npmjs.com/package/eslint-plugin-react-hooks) enforce? |
|||
|
|||
We provide an [ESLint plugin](https://www.npmjs.com/package/eslint-plugin-react-hooks) that enforces [rules of Hooks](/docs/hooks-rules.html) to avoid bugs. It assumes that any function starting with "`use`" and a capital letter right after it is a Hook. We recognize this heuristic isn't perfect and there may be some false positives, but without an ecosystem-wide convention there is just no way to make Hooks work well -- and longer names will discourage people from either adopting Hooks or following the convention. |
|||
|
|||
In particular, the rule enforces that: |
|||
|
|||
* Calls to Hooks are either inside a `PascalCase` function (assumed to be a component) or another `useSomething` function (assumed to be a custom Hook). |
|||
* Hooks are called in the same order on every render. |
|||
|
|||
There are a few more heuristics, and they might change over time as we fine-tune the rule to balance finding bugs with avoiding false positives. |
|||
|
|||
## From Classes to Hooks |
|||
|
|||
### How do lifecycle methods correspond to Hooks? |
|||
|
|||
* `constructor`: Function components don't need a constructor. You can initialize the state in the [`useState`](/docs/hooks-reference.html#usestate) call. If computing it is expensive, you can pass a function to `useState`. |
|||
|
|||
* `getDerivedStateFromProps`: Schedule an update [while rendering](#how-do-i-implement-getderivedstatefromprops) instead. |
|||
|
|||
* `shouldComponentUpdate`: See `React.pure` [below](#how-do-i-implement-shouldcomponentupdate). |
|||
|
|||
* `render`: This is the function component body itself. |
|||
|
|||
* `componentDidMount`, `componentDidUpdate`, `componentWillUnmount`: The [`useEffect` Hook](/docs/hooks-reference.html#useeffect) can express all combinations of these (including [less](#can-i-skip-an-effect-on-updates) [common](#can-i-run-an-effect-only-on-updates) cases). |
|||
|
|||
* `componentDidCatch` and `getDerivedStateFromError`: There are no Hook equivalents for these methods yet, but they will be added soon. |
|||
|
|||
### Is there something like instance variables? |
|||
|
|||
Yes! The [`useRef()`](/docs/hooks-reference.html#useref) Hook isn't just for DOM refs. The "ref" object is a generic container whose `current` property is mutable and can hold any value, similar to an instance property on a class. |
|||
|
|||
You can write to it from inside `useEffect`: |
|||
|
|||
```js{2,8,15} |
|||
function Timer() { |
|||
const intervalRef = useRef(); |
|||
|
|||
useEffect(() => { |
|||
const id = setInterval(() => { |
|||
// ... |
|||
}); |
|||
intervalRef.current = id; |
|||
return () => { |
|||
clearInterval(intervalRef.current); |
|||
}; |
|||
}); |
|||
|
|||
// ... |
|||
} |
|||
``` |
|||
|
|||
If we just wanted to set an interval, we wouldn't need the ref (`id` could be local to the effect), but it's useful if we want to clear the interval from an event handler: |
|||
|
|||
```js{3} |
|||
// ... |
|||
function handleCancelClick() { |
|||
clearInterval(intervalRef.current); |
|||
} |
|||
// ... |
|||
``` |
|||
|
|||
Conceptually, you can think of refs as similar to instance variables in a class. Avoid setting refs during rendering -- this can lead to surprising behavior. Instead, only modify refs in event handlers and effects. |
|||
|
|||
### Should I use one or many state variables? |
|||
|
|||
If you're coming from classes, you might be tempted to always call `useState()` once and put all state into a single object. You can do it if you'd like. Here is an example of a component that follows the mouse movement. We keep its position and size in the local state: |
|||
|
|||
```js |
|||
function Box() { |
|||
const [state, setState] = useState({ left: 0, top: 0, width: 100, height: 100 }); |
|||
// ... |
|||
} |
|||
``` |
|||
|
|||
Now let's say we want to write some logic that changes `left` and `top` when the user moves their mouse. Note how we have to merge these fields into the previous state object manually: |
|||
|
|||
```js{4,5} |
|||
// ... |
|||
useEffect(() => { |
|||
function handleWindowMouseMove(e) { |
|||
// Spreading "...state" ensures we don't "lose" width and height |
|||
setState(state => ({ ...state, left: e.pageX, top: e.pageY })); |
|||
} |
|||
// Note: this implementation is a bit simplified |
|||
window.addEventListener('mousemove', handleWindowMouseMove); |
|||
return () => window.removeEventListener('mousemove', handleWindowMouseMove); |
|||
}, []); |
|||
// ... |
|||
``` |
|||
|
|||
This is because when we update a state variable, we *replace* its value. This is different from `this.setState` in a class, which *merges* the updated fields into the object. |
|||
|
|||
If you miss automatic merging, you can write a custom `useLegacyState` Hook that merges object state updates. However, instead **we recommend to split state into multiple state variables based on which values tend to change together.** |
|||
|
|||
For example, we could split our component state into `position` and `size` objects, and always replace the `position` with no need for merging: |
|||
|
|||
```js{2,7} |
|||
function Box() { |
|||
const [position, setPosition] = useState({ left: 0, top: 0 }); |
|||
const [size, setSize] = useState({ width: 100, height: 100 }); |
|||
|
|||
useEffect(() => { |
|||
function handleWindowMouseMove(e) { |
|||
setPosition({ left: e.pageX, top: e.pageY }); |
|||
} |
|||
// ... |
|||
``` |
|||
|
|||
Separating independent state variables also has another benefit. It makes it easy to later extract some related logic into a custom Hook, for example: |
|||
|
|||
```js{2,7} |
|||
function Box() { |
|||
const position = useWindowPosition(); |
|||
const [size, setSize] = useState({ width: 100, height: 100 }); |
|||
// ... |
|||
} |
|||
|
|||
function useWindowPosition() { |
|||
const [position, setPosition] = useState({ left: 0, top: 0 }); |
|||
useEffect(() => { |
|||
// ... |
|||
}, []); |
|||
return position; |
|||
} |
|||
``` |
|||
|
|||
Note how we were able to move the `useState` call for the `position` state variable and the related effect into a custom Hook without changing their code. If all state was in a single object, extracting it would be more difficult. |
|||
|
|||
Both putting all state in a single `useState` call, and having a `useState` call per each field can work. Components tend to be most readable when you find a balance between these two extremes, and group related state into a few independent state variables. If the state logic becomes complex, we recommend [managing it with a reducer](/docs/hooks-reference.html#usereducer) or a custom Hook. |
|||
|
|||
### Can I run an effect only on updates? |
|||
|
|||
This is a rare use case. If you need it, you can [use a mutable ref](#is-there-something-like-instance-variables) to manually store a boolean value corresponding to whether you are on the first or a subsequent render, then check that flag in your effect. (If you find yourself doing this often, you could create a custom Hook for it.) |
|||
|
|||
### How to get the previous props or state? |
|||
|
|||
Currently, you can do it manually [with a ref](#is-there-something-like-instance-variables): |
|||
|
|||
```js{6,8} |
|||
function Counter() { |
|||
const [count, setCount] = useState(0); |
|||
|
|||
const prevCountRef = useRef(); |
|||
useEffect(() => { |
|||
prevCountRef.current = count; |
|||
}); |
|||
const prevCount = prevCountRef.current; |
|||
|
|||
return <h1>Now: {count}, before: {prevCount}</h1>; |
|||
} |
|||
``` |
|||
|
|||
This might be a bit convoluted but you can extract it into a custom Hook: |
|||
|
|||
```js{3,7} |
|||
function Counter() { |
|||
const [count, setCount] = useState(0); |
|||
const prevCount = usePrevious(count); |
|||
return <h1>Now: {count}, before: {prevCount}</h1>; |
|||
} |
|||
|
|||
function usePrevious(value) { |
|||
const ref = useRef(); |
|||
useEffect(() => { |
|||
ref.current = value; |
|||
}); |
|||
return ref.current; |
|||
} |
|||
``` |
|||
|
|||
Note how this would work for props, state, or any other calculated value. |
|||
|
|||
```js{5} |
|||
function Counter() { |
|||
const [count, setCount] = useState(0); |
|||
|
|||
const calculation = count * 100; |
|||
const prevCalculation = usePrevious(calculation); |
|||
// ... |
|||
``` |
|||
|
|||
It's possible that in the future React will provide a `usePrevious` Hook out of the box since it's a relatively common use case. |
|||
|
|||
See also [the recommended pattern for derived state](#how-do-i-implement-getderivedstatefromprops). |
|||
|
|||
### How do I implement `getDerivedStateFromProps`? |
|||
|
|||
While you probably [don't need it](/blog/2018/06/07/you-probably-dont-need-derived-state.html), for the rare cases that you do (such as implementing a `<Transition>` component), you can update the state right during rendering. React will re-run the component with updated state immediately after exiting the first render so it wouldn't be expensive. |
|||
|
|||
Here, we store the previous value of the `row` prop in a state variable so that we can compare: |
|||
|
|||
```js |
|||
function ScrollView({row}) { |
|||
let [isScrollingDown, setIsScrollingDown] = useState(false); |
|||
let [prevRow, setPrevRow] = useState(null); |
|||
|
|||
if (row !== prevRow) { |
|||
// Row changed since last render. Update isScrollingDown. |
|||
setIsScrollingDown(prevRow !== null && row > prevRow); |
|||
setPrevRow(row); |
|||
} |
|||
|
|||
return `Scrolling down: ${isScrollingDown}`; |
|||
} |
|||
``` |
|||
|
|||
This might look strange at first, but an update during rendering is exactly what `getDerivedStateFromProps` has always been like conceptually. |
|||
|
|||
### Can I make a ref to a function component? |
|||
|
|||
While you shouldn't need this often, you may expose some imperative methods to a parent component with the [`useImperativeMethods`](/docs/hooks-reference.html#useimperativemethods) Hook. |
|||
|
|||
### What does `const [thing, setThing] = useState()` mean? |
|||
|
|||
If you're not familiar with this syntax, check out the [explanation](/docs/hooks-state.html#tip-what-do-square-brackets-mean) in the State Hook documentation. |
|||
|
|||
|
|||
## Performance Optimizations |
|||
|
|||
### Can I skip an effect on updates? |
|||
|
|||
Yes. See [conditionally firing an effect](/docs/hooks-reference.html#conditionally-firing-an-effect). Note that forgetting to handle updates often [introduces bugs](/docs/hooks-effect.html#explanation-why-effects-run-on-each-update), which is why this isn't the default behavior. |
|||
|
|||
### How do I implement `shouldComponentUpdate`? |
|||
|
|||
You can wrap a function component with `React.pure` to shallowly compare its props: |
|||
|
|||
```js |
|||
const Button = React.pure((props) => { |
|||
// your component |
|||
}); |
|||
``` |
|||
|
|||
It's not a Hook because it doesn't compose like Hooks do. `React.pure` is equivalent to `PureComponent`, but it only compares props. (You can also add a second argument to specify a custom comparison function that takes the old and new props. If it returns true, the update is skipped.) |
|||
|
|||
`React.pure` doesn't compare state because there is no single state object to compare. But you can make children pure too, or even [optimize individual children with `useMemo`](/docs/hooks-faq.html#how-to-memoize-calculations). |
|||
|
|||
|
|||
### How to memoize calculations? |
|||
|
|||
The [`useMemo`](/docs/hooks-reference.html#usememo) Hook lets you cache calculations between multiple renders by "remembering" the previous computation: |
|||
|
|||
```js |
|||
const memoizedValue = useMemo(() => computeExpensiveValue(a, b), [a, b]); |
|||
``` |
|||
|
|||
This code calls `computeExpensiveValue(a, b)`. But if the inputs `[a, b]` haven't changed since the last value, `useMemo` skips calling it a second time and simply reuses the last value it returned. |
|||
|
|||
Conveniently, this also lets you skip an expensive re-render of a child: |
|||
|
|||
```js |
|||
function Parent({ a, b }) { |
|||
// Only re-rendered if `a` changes: |
|||
const child1 = useMemo(() => <Child1 a={a} />, [a]); |
|||
// Only re-rendered if `b` changes: |
|||
const child2 = useMemo(() => <Child2 b={b} />, [b]); |
|||
return ( |
|||
<> |
|||
{child1} |
|||
{child2} |
|||
</> |
|||
) |
|||
} |
|||
``` |
|||
|
|||
Note that this approach won't work in a loop because Hook calls [can't](/docs/hooks-rules.html) be placed inside loops. But you can extract a separate component for the list item, and call `useMemo` there. |
|||
|
|||
### Are Hooks slow because of creating functions in render? |
|||
|
|||
No. In modern browsers, the raw performance of closures compared to classes doesn't differ significantly except in extreme scenarios. |
|||
|
|||
In addition, consider that the design of Hooks is more efficient in a couple ways: |
|||
|
|||
* Hooks avoid a lot of the overhead that classes require, like the cost of creating class instances and binding event handlers in the constructor. |
|||
|
|||
* **Idiomatic code using Hooks doesn't need the deep component tree nesting** that is prevalent in codebases that use higher-order components, render props, and context. With smaller component trees, React has less work to do. |
|||
|
|||
Traditionally, performance concerns around inline functions in React have been related to how passing new callbacks on each render breaks `shouldComponentUpdate` optimizations in child components. Hooks approach this problem from three sides. |
|||
|
|||
* The [`useCallback`](/docs/hooks-reference.html#usecallback) Hook lets you keep the same callback reference between re-renders so that `shouldComponentUpdate` continues to work: |
|||
|
|||
```js{2} |
|||
// Will not change unless `a` or `b` changes |
|||
const memoizedCallback = useCallback(() => { |
|||
doSomething(a, b); |
|||
}, [a, b]); |
|||
``` |
|||
|
|||
* The [`useMemo` Hook](/docs/hooks-faq.html#how-to-memoize-calculations) makes it easier to control when individual children update, reducing the need for pure components. |
|||
|
|||
* Finally, the `useReducer` Hook reduces the need to pass callbacks deeply, as explained below. |
|||
|
|||
### How to avoid passing callbacks down? |
|||
|
|||
We've found that most people don't enjoy manually passing callbacks through every level of a component tree. Even though it is more explicit, it can feel like a lot of "plumbing". |
|||
|
|||
In large component trees, an alternative we recommend is to pass down a `dispatch` function from [`useReducer`](/docs/hooks-reference.html#usereducer) via context: |
|||
|
|||
```js{4,5} |
|||
const TodosDispatch = React.createContext(null); |
|||
|
|||
function TodosApp() { |
|||
// Tip: `dispatch` won't change between re-renders |
|||
const [todos, dispatch] = useReducer(todosReducer); |
|||
|
|||
return ( |
|||
<TodosDispatch.Provider value={dispatch}> |
|||
<DeepTree todos={todos} /> |
|||
</TodosDispatch.Provider> |
|||
); |
|||
} |
|||
``` |
|||
|
|||
Any child in the tree inside `TodosApp` can read use the `dispatch` function to pass actions up to `TodosApp`: |
|||
|
|||
```js{2,3} |
|||
function DeepChild(props) { |
|||
// If we want to perform an action, we can get dispatch from context. |
|||
const dispatch = useContext(TodosDispatch); |
|||
|
|||
function handleClick() { |
|||
dispatch({ type: 'add', text: 'hello' }); |
|||
} |
|||
|
|||
return ( |
|||
<button onClick={handleClick}>Add todo</button> |
|||
); |
|||
} |
|||
``` |
|||
|
|||
This is both more convenient from the maintenance perspective (no need to keep forwarding callbacks), and avoids the callback problem altogether. Passing `dispatch` down like this is the recommended pattern for deep updates. |
|||
|
|||
Note that you can still choose whether to pass the application *state* down as props (more explicit) or as context (more convenient for very deep updates). If you use context to pass down the state too, use two different context types -- the `dispatch` context never changes, so components that read it don't need to rerender unless they also need the application state. |
|||
|
|||
### How to read an often-changing value from `useCallback`? |
|||
|
|||
>Note |
|||
> |
|||
>We recommend to [pass `dispatch` down in context](#how-to-avoid-passing-callbacks-down) rather than individual callbacks in props. The approach below is only mentioned here for completeness and as an escape hatch. |
|||
|
|||
In some rare cases you might need to memoize a callback with [`useCallback`](/docs/hooks-reference.html#usecallback) but the memoization doesn't work very well because the inner function has to be re-created too often. If the function you're memoizing is an event handler and isn't used during rendering, you can use [ref as an instance variable](#is-there-something-like-instance-variables), and save the last committed value into it manually: |
|||
|
|||
```js{6,10} |
|||
function Form() { |
|||
const [text, updateText] = useState(''); |
|||
const textRef = useRef(); |
|||
|
|||
useEffect(() => { |
|||
textRef.current = text; // Write it to the ref |
|||
}); |
|||
|
|||
const handleSubmit = useCallback(() => { |
|||
const currentText = textRef.current; // Read it from the ref |
|||
alert(currentText); |
|||
}, [textRef]); // Don't recreate handleSubmit like [text] would do |
|||
|
|||
return ( |
|||
<> |
|||
<input value={text} onChange={e => updateText(e.target.value)} /> |
|||
<ExpensiveTree onSubmit={handleSubmit} /> |
|||
</> |
|||
); |
|||
} |
|||
``` |
|||
|
|||
This is a rather convoluted pattern but it shows that you can do this escape hatch optimization if you need it. It's more bearable if you extract it to a custom Hook: |
|||
|
|||
```js{4,16} |
|||
function Form() { |
|||
const [text, updateText] = useState(''); |
|||
// Will be memoized even if `text` changes: |
|||
const handleSubmit = useEventCallback(() => { |
|||
alert(text); |
|||
}, [text]); |
|||
|
|||
return ( |
|||
<> |
|||
<input value={text} onChange={e => updateText(e.target.value)} /> |
|||
<ExpensiveTree onSubmit={handleSubmit} /> |
|||
</> |
|||
); |
|||
} |
|||
|
|||
function useEventCallback(fn, dependencies) { |
|||
const ref = useRef(() => { |
|||
throw new Error('Cannot call an event handler while rendering.'); |
|||
}); |
|||
|
|||
useEffect(() => { |
|||
ref.current = fn; |
|||
}, [fn, ...dependencies]); |
|||
|
|||
return useCallback(() => { |
|||
const fn = ref.current; |
|||
return fn(); |
|||
}, [ref]); |
|||
} |
|||
``` |
|||
|
|||
In either case, we **don't recommend this pattern** and only show it here for completeness. Instead, it is preferable to [avoid passing callbacks deep down](#how-to-avoid-passing-callbacks-down). |
|||
|
|||
## Under the Hood |
|||
|
|||
### How does React associate Hook calls with components? |
|||
|
|||
React keeps track of the currently rendering component. Thanks to the [Rules of Hooks](/docs/hooks-rules.html), we know that Hooks are only called from React components (or custom Hooks -- which are also only called from React components). |
|||
|
|||
There is an internal list of "memory cells" associated with each component. They're just JavaScript objects where we can put some data. When you call a Hook like `useState()`, it reads the current cell (or initializes it during the first render), and then moves the pointer to the next one. This is how multiple `useState()` calls each get independent local state. |
|||
|
|||
### What is the prior art for Hooks? |
|||
|
|||
Hooks synthesize ideas from several different sources: |
|||
|
|||
* Our old experiments with functional APIs in the [react-future](https://github.com/reactjs/react-future/tree/master/07%20-%20Returning%20State) repository. |
|||
* React community's experiments with render prop APIs, including [Ryan Florence](https://github.com/ryanflorence)'s [Reactions Component](https://github.com/reactions/component). |
|||
* [Dominic Gannaway](https://github.com/trueadm)'s [`adopt` keyword](https://gist.github.com/trueadm/17beb64288e30192f3aa29cad0218067) proposal as a sugar syntax for render props. |
|||
* State variables and state cells in [DisplayScript](http://displayscript.org/introduction.html). |
|||
* [Reducer components](https://reasonml.github.io/reason-react/docs/en/state-actions-reducer.html) in ReasonReact. |
|||
* [Subscriptions](http://reactivex.io/rxjs/class/es6/Subscription.js~Subscription.html) in Rx. |
|||
* [Algebraic effects](https://github.com/ocamllabs/ocaml-effects-tutorial#2-effectful-computations-in-a-pure-setting) in Multicore OCaml. |
|||
|
|||
[Sebastian Markbåge](https://github.com/sebmarkbage) came up with the original design for Hooks, later refined by [Andrew Clark](https://github.com/acdlite), [Sophie Alpert](https://github.com/sophiebits), [Dominic Gannaway](https://github.com/trueadm), and other members of the React team. |
@ -0,0 +1,96 @@ |
|||
--- |
|||
id: hooks-intro |
|||
title: Introducing Hooks |
|||
permalink: docs/hooks-intro.html |
|||
next: hooks-overview.html |
|||
--- |
|||
|
|||
*Hooks* are a new feature proposal that lets you use state and other React features without writing a class. They're currently in React v16.7.0-alpha and being discussed in [an open RFC](https://github.com/reactjs/rfcs/pulls). |
|||
|
|||
```js{4,5} |
|||
import { useState } from 'react'; |
|||
|
|||
function Example() { |
|||
// Declare a new state variable, which we'll call "count" |
|||
const [count, setCount] = useState(0); |
|||
|
|||
return ( |
|||
<div> |
|||
<p>You clicked {count} times</p> |
|||
<button onClick={() => setCount(count + 1)}> |
|||
Click me |
|||
</button> |
|||
</div> |
|||
); |
|||
} |
|||
``` |
|||
|
|||
This new function `useState` is the first "Hook" we'll learn about, but this example is just a teaser. Don't worry if it doesn't make sense yet! |
|||
|
|||
**You can start learning Hooks [on the next page](/docs/hooks-overview.html).** On this page, we'll continue by explaining why we're adding Hooks to React and how they can help you write great applications. |
|||
|
|||
## No Breaking Changes |
|||
|
|||
Before we continue, note that Hooks are: |
|||
|
|||
* **Completely opt-in.** You can try Hooks in a few components without rewriting any existing code. But you don't have to learn or use Hooks right now if you don't want to. |
|||
* **100% backwards-compatible.** Hooks don't contain any breaking changes. |
|||
* **Available now.** Hooks are currently in an alpha release, and we hope to include them in React 16.7 after receiving community feedback. |
|||
|
|||
**There are no plans to remove classes from React.** You can read more about the gradual adoption strategy for Hooks in the [bottom section](#gradual-adoption-strategy) of this page. |
|||
|
|||
**Hooks don't replace your knowledge of React concepts.** Instead, Hooks provide a more direct API to the React concepts you already know: props, state, context, refs, and lifecycle. As we will show later, Hooks also offer a new powerful way to combine them. |
|||
|
|||
**If you just want to start learning Hooks, feel free to [jump directly to the next page!](/docs/hooks-overview.html)** You can also keep reading this page to learn more about why we're adding Hooks, and how we're going to start using them without rewriting our applications. |
|||
|
|||
## Motivation |
|||
|
|||
Hooks solve a wide variety of seemingly unconnected problems in React that we've encountered over five years of writing and maintaining tens of thousands of components. Whether you're learning React, use it daily, or even prefer a different library with a similar component model, you might recognize some of these problems. |
|||
|
|||
### It's hard to reuse stateful logic between components |
|||
|
|||
React doesn't offer a way to "attach" reusable behavior to a component (for example, connecting it to a store). If you've worked with React for a while, you may be familiar with patterns like [render props](/docs/render-props.html) and [higher-order components](/docs/higher-order-components.html) that try to solve this. But these patterns require you to restructure your components when you use them, which can be cumbersome and make code harder to follow. If you look at a typical React application in React DevTools, you will likely find a "wrapper hell" of components surrounded by layers of providers, consumers, higher-order components, render props, and other abstractions. While we could [filter them out in DevTools](https://github.com/facebook/react-devtools/pull/503), this points to a deeper underlying problem: React needs a better primitive for sharing stateful logic. |
|||
|
|||
With Hooks, you can extract stateful logic from a component so it can be tested independently and reused. **Hooks allow you to reuse stateful logic without changing your component hierarchy.** This makes it easy to share Hooks among many components or with the community. |
|||
|
|||
We'll discuss this more in [Writing Custom Hooks](/docs/hooks-custom.html#using-a-custom-hook). |
|||
|
|||
### Complex components become hard to understand |
|||
|
|||
We've often had to maintain components that started out simple but grew into an unmanageable mess of stateful logic and side effects. Each lifecycle method often contains a mix of unrelated logic. For example, components might perform some data fetching in `componentDidMount` and `componentDidUpdate`. However, the same `componentDidMount` method might also contain some unrelated logic that sets up event listeners, with cleanup performed in `componentWillUnmount`. Mutually related code that changes together gets split apart, but completely unrelated code ends up combined in a single method. This makes it too easy to introduce bugs and inconsistencies. |
|||
|
|||
In many cases it's not possible to break these components into smaller ones because the stateful logic is all over the place. It's also difficult to test them. This is one of the reasons many people prefer to combine React with a separate state management library. However, that often introduces too much abstraction, requires you to jump between different files, and makes reusing components more difficult. |
|||
|
|||
To solve this, **Hooks let you split one component into smaller functions based on what pieces are related (such as setting up a subscription or fetching data)**, rather than forcing a split based on lifecycle methods. You may also opt into managing the component's local state with a reducer to make it more predictable. |
|||
|
|||
We'll discuss this more in [Using the Effect Hook](/docs/hooks-effect.html#tip-use-multiple-effects-to-separate-concerns). |
|||
|
|||
### Classes confuse both people and machines |
|||
|
|||
In our observation, classes are the biggest barrier to learning React. You have to understand how `this` works in JavaScript, which is very different from how it works in most languages. You have to remember to bind the event handlers. Without unstable [syntax proposals](https://babeljs.io/docs/en/babel-plugin-transform-class-properties/), the code is very verbose. People can understand props, state, and top-down data flow perfectly well but still struggle with classes. The distinction between function and class components in React and when to use each one leads to disagreements even between experienced React developers. |
|||
|
|||
Additionally, React has been out for about five years, and we want to make sure it stays relevant in the next five years. As [Svelte](https://svelte.technology/), [Angular](https://angular.io/), [Glimmer](https://glimmerjs.com/), and others show, [ahead-of-time compilation](https://en.wikipedia.org/wiki/Ahead-of-time_compilation) of components has a lot of future potential. Especially if it's not limited to templates. Recently, we've been experimenting with [component folding](https://github.com/facebook/react/issues/7323) using [Prepack](https://prepack.io/), and we've seen promising early results. However, we found that class components can encourage unintentional patterns that make these optimizations fall back to a slower path. Classes present issues for today's tools, too. For example, classes don't minify very well, and they make hot reloading flaky and unreliable. We want to present an API that makes it more likely for code to stay on the optimizable path. |
|||
|
|||
To solve these problems, **Hooks let you use more of React's features without classes.** Conceptually, React components have always been closer to functions. Hooks embrace functions, but without sacrificing the practical spirit of React. Hooks provide access to imperative escape hatches and don't require you to learn complex functional or reactive programming techniques. |
|||
|
|||
>Examples |
|||
> |
|||
>[Hooks at a Glance](/docs/hooks-overview.html) is a good place to start learning Hooks. |
|||
|
|||
## Gradual Adoption Strategy |
|||
|
|||
>**TLDR: There are no plans to remove classes from React.** |
|||
|
|||
We know that React developers are focused on shipping products and don't have time to look into every new API that's being released. Hooks are very new, and it might be better to wait for more examples and tutorials before considering learning or adopting them. |
|||
|
|||
We also understand that the bar for adding a new primitive to React is extremely high. For curious readers, we have prepared a [detailed RFC](https://TODO) that dives into motivation with more details, and provides extra perspective on the specific design decisions and related prior art. |
|||
|
|||
**Crucially, Hooks work side-by-side with existing code so you can adopt them gradually.** We are sharing this experimental API to get early feedback from those in the community who are interested in shaping the future of React — and we will iterate on Hooks in the open. |
|||
|
|||
Finally, there is no rush to migrate to Hooks. We recommend avoiding any "big rewrites", especially for existing, complex class components. It takes a bit of a mindshift to start "thinking in Hooks". In our experience, it's best to practice using Hooks in new and non-critical components first, and ensure that everybody on your team feels comfortable with them. After you give Hooks a try, please feel free to [send us feedback](https://github.com/facebook/react/issues/new), positive or negative. |
|||
|
|||
We intend for Hooks to cover all existing use cases for classes, but **we will keep supporting class components for the foreseeable future.** At Facebook, we have tens of thousands of components written as classes, and we have absolutely no plans to rewrite them. Instead, we are starting to use Hooks in the new code side by side with classes. |
|||
|
|||
## Next Steps |
|||
|
|||
By the end of this page, you should have a rough idea of what problems Hooks are solving, but many details are probably unclear. Don't worry! **Let's now go to [the next page](/docs/hooks-overview.html) where we start learning about Hooks by example.** |
@ -0,0 +1,274 @@ |
|||
--- |
|||
id: hooks-overview |
|||
title: Hooks at a Glance |
|||
permalink: docs/hooks-overview.html |
|||
next: hooks-state.html |
|||
prev: hooks-intro.html |
|||
--- |
|||
|
|||
*Hooks* are a new feature proposal that lets you use state and other React features without writing a class. They're currently in React v16.7.0-alpha and being discussed in [an open RFC](https://github.com/reactjs/rfcs/pulls). |
|||
|
|||
Hooks are [backwards-compatible](/docs/hooks-intro.html#no-breaking-changes). This page provides an overview of Hooks for experienced React users. |
|||
|
|||
This is a fast-paced overview. If you get confused, look for a yellow box like this: |
|||
|
|||
>Detailed Explanation |
|||
> |
|||
>Read the [Motivation](/docs/hooks-intro.html#motivation) to learn why we're introducing Hooks to React. |
|||
|
|||
**↑↑↑ Each section ends with a yellow box like this.** They link to detailed explanations. |
|||
|
|||
## 📌 State Hook |
|||
|
|||
This example renders a counter. When you click the button, it increments the value: |
|||
|
|||
```js{1,4,5} |
|||
import { useState } from 'react'; |
|||
|
|||
function Example() { |
|||
// Declare a new state variable, which we'll call "count" |
|||
const [count, setCount] = useState(0); |
|||
|
|||
return ( |
|||
<div> |
|||
<p>You clicked {count} times</p> |
|||
<button onClick={() => setCount(count + 1)}> |
|||
Click me |
|||
</button> |
|||
</div> |
|||
); |
|||
} |
|||
``` |
|||
|
|||
Here, `useState` is a *Hook* (we'll talk about what this means in a moment). We call it inside a function component to add some local state to it. React will preserve this state between re-renders. `useState` returns a pair: the *current* state value and a function that lets you update it. You can call this function from an event handler or somewhere else. It's similar to `this.setState` in a class, except it doesn't merge the old and new state together. (We'll show an example comparing `useState` to `this.state` in [Using the State Hook](/docs/hooks-state.html).) |
|||
|
|||
The only argument to `useState` is the initial state. In the example above, it is `0` because our counter starts from zero. Note that unlike `this.state`, the state here doesn't have to be an object -- although it can be if you want. The initial state argument is only used during the first render. |
|||
|
|||
#### Declaring multiple state variables |
|||
|
|||
You can use the State Hook more than once in a single component: |
|||
|
|||
```js |
|||
function ExampleWithManyStates() { |
|||
// Declare multiple state variables! |
|||
const [age, setAge] = useState(42); |
|||
const [fruit, setFruit] = useState('banana'); |
|||
const [todos, setTodos] = useState([{ text: 'Learn Hooks' }]); |
|||
// ... |
|||
} |
|||
``` |
|||
|
|||
The [array destructuring](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Destructuring_assignment#Array_destructuring) syntax let us give different names to the state variables we declared by calling `useState`. These names aren't a part of the `useState` API. Instead, React assumes that if you call `useState` many times, you do it in the same order during every render. We'll come back to why this works and when this is useful later. |
|||
|
|||
#### But what is a Hook? |
|||
|
|||
Hooks are functions that let you “hook into” React state and lifecycle features from function components. Hooks don't work inside classes -- they let you use React without classes. (We [don't recommend](/docs/hooks-intro.html#gradual-adoption-strategy) rewriting your existing components overnight but you can start using Hooks in the new ones if you'd like.) |
|||
|
|||
React provides a few built-in Hooks like `useState`. You can also create custom Hooks to reuse stateful behavior between different components. We'll look at the built-in Hooks first. |
|||
|
|||
>Detailed Explanation |
|||
> |
|||
>You can learn more about the State Hook on a dedicated page: [Using the State Hook](/docs/hooks-state.html). |
|||
|
|||
## ⚡️ Effect Hook |
|||
|
|||
You've likely performed data fetching, subscriptions, or manually changing the DOM from React components before. We call these operations "side effects" (or "effects" for short) because they can affect other components and can't be done during rendering. |
|||
|
|||
The Effect Hook, `useEffect`, adds the ability to perform side effects from a function component. It serves the same purpose as `componentDidMount`, `componentDidUpdate`, and `componentWillUnmount` in React classes, but unified into a single API. (We'll show examples comparing `useEffect` to these methods in [Using the Effect Hook](/docs/hooks-effect.html).) |
|||
|
|||
For example, this component sets the document title after React updates the DOM: |
|||
|
|||
```js{1,6-10} |
|||
import { useState, useEffect } from 'react'; |
|||
|
|||
function Example() { |
|||
const [count, setCount] = useState(0); |
|||
|
|||
// Similar to componentDidMount and componentDidUpdate: |
|||
useEffect(() => { |
|||
// Update the document title using the browser API |
|||
document.title = `You clicked ${count} times`; |
|||
}); |
|||
|
|||
return ( |
|||
<div> |
|||
<p>You clicked {count} times</p> |
|||
<button onClick={() => setCount(count + 1)}> |
|||
Click me |
|||
</button> |
|||
</div> |
|||
); |
|||
} |
|||
``` |
|||
|
|||
When you call `useEffect`, you're telling React to run your "effect" function after flushing changes to the DOM. Effects are declared inside the component so they have access to its props and state. By default, React runs the effects after every render -- *including* the first render. (We'll talk more about how this compares to class lifecycles in [Using the Effect Hook](/docs/hooks-effect.html).) |
|||
|
|||
Effects may also optionally specify how to "clean up" after them by returning a function. For example, this component uses an effect to subscribe to a friend's online status, and cleans up by unsubscribing from it: |
|||
|
|||
```js{10-16} |
|||
import { useState, useEffect } from 'react'; |
|||
|
|||
function FriendStatus(props) { |
|||
const [isOnline, setIsOnline] = useState(null); |
|||
|
|||
function handleStatusChange(status) { |
|||
setIsOnline(status.isOnline); |
|||
} |
|||
|
|||
useEffect(() => { |
|||
ChatAPI.subscribeToFriendStatus(props.friend.id, handleStatusChange); |
|||
|
|||
return () => { |
|||
ChatAPI.unsubscribeFromFriendStatus(props.friend.id, handleStatusChange); |
|||
}; |
|||
}); |
|||
|
|||
if (isOnline === null) { |
|||
return 'Loading...'; |
|||
} |
|||
return isOnline ? 'Online' : 'Offline'; |
|||
} |
|||
``` |
|||
|
|||
In this example, React would unsubscribe from our `ChatAPI` when the component unmounts, as well as before re-running the effect due to a subsequent render. (If you want, there's a way to [tell React to skip re-subscribing](/docs/hooks-effect.html#tip-optimizing-performance-by-skipping-effects) if the `props.friend.id` we passed to `ChatAPI` didn’t change.) |
|||
|
|||
Just like with `useState`, you can use more than a single effect in a component: |
|||
|
|||
```js{3,8} |
|||
function FriendStatusWithCounter(props) { |
|||
const [count, setCount] = useState(0); |
|||
useEffect(() => { |
|||
document.title = `You clicked ${count} times`; |
|||
}); |
|||
|
|||
const [isOnline, setIsOnline] = useState(null); |
|||
useEffect(() => { |
|||
ChatAPI.subscribeToFriendStatus(props.friend.id, handleStatusChange); |
|||
return () => { |
|||
ChatAPI.unsubscribeFromFriendStatus(props.friend.id, handleStatusChange); |
|||
}; |
|||
}); |
|||
|
|||
function handleStatusChange(status) { |
|||
setIsOnline(status.isOnline); |
|||
} |
|||
// ... |
|||
``` |
|||
|
|||
Hooks let you organize side effects in a component by what pieces are related (such as adding and removing a subscription), rather than forcing a split based on lifecycle methods. |
|||
|
|||
>Detailed Explanation |
|||
> |
|||
>You can learn more about `useEffect` on a dedicated page: [Using the Effect Hook](/docs/hooks-effect.html). |
|||
|
|||
## ✌️ Rules of Hooks |
|||
|
|||
Hooks are JavaScript functions, but they impose two additional rules: |
|||
|
|||
* Only call Hooks **at the top level**. Don’t call Hooks inside loops, conditions, or nested functions. |
|||
* Only call Hooks **from React function components**. Don’t call Hooks from regular JavaScript functions. (There is just one other valid place to call Hooks -- your own custom Hooks. We'll learn about them in a moment.) |
|||
|
|||
We provide a [linter plugin](https://www.npmjs.com/package/eslint-plugin-react-hooks) to enforce these rules automatically. We understand these rules might seem limiting or confusing at first, but they are essential to making Hooks work well. |
|||
|
|||
>Detailed Explanation |
|||
> |
|||
>You can learn more about these rules on a dedicated page: [Rules of Hooks](/docs/hooks-rules.html). |
|||
|
|||
## 💡 Custom Hooks |
|||
|
|||
Sometimes, we want to reuse some stateful logic between components. Traditionally, there were two popular solutions to this problem: [higher-order components](/docs/higher-order-components.html) and [render props](/docs/render-props.html). Custom Hooks let you do this, but without adding more components to your tree. |
|||
|
|||
Earlier on this page, we introduced a `FriendStatus` component that calls the `useState` and `useEffect` Hooks to subscribe to a friend's online status. Let's say we also want to reuse this subscription logic in another component. |
|||
|
|||
First, we'll extract this logic into a custom Hook called `useFriendStatus`: |
|||
|
|||
```js{3} |
|||
import { useState, useEffect } from 'react'; |
|||
|
|||
function useFriendStatus(friendID) { |
|||
const [isOnline, setIsOnline] = useState(null); |
|||
|
|||
function handleStatusChange(status) { |
|||
setIsOnline(status.isOnline); |
|||
} |
|||
|
|||
useEffect(() => { |
|||
ChatAPI.subscribeToFriendStatus(friendID, handleStatusChange); |
|||
return () => { |
|||
ChatAPI.unsubscribeFromFriendStatus(friendID, handleStatusChange); |
|||
}; |
|||
}); |
|||
|
|||
return isOnline; |
|||
} |
|||
``` |
|||
|
|||
It takes `friendID` as an argument, and returns whether our friend is online. |
|||
|
|||
Now we can use it from both components: |
|||
|
|||
|
|||
```js{2} |
|||
function FriendStatus(props) { |
|||
const isOnline = useFriendStatus(props.friend.id); |
|||
|
|||
if (isOnline === null) { |
|||
return 'Loading...'; |
|||
} |
|||
return isOnline ? 'Online' : 'Offline'; |
|||
} |
|||
``` |
|||
|
|||
```js{2} |
|||
function FriendListItem(props) { |
|||
const isOnline = useFriendStatus(props.friend.id); |
|||
|
|||
return ( |
|||
<li style={{ color: isOnline ? 'green' : 'black' }}> |
|||
{props.friend.name} |
|||
</li> |
|||
); |
|||
} |
|||
``` |
|||
|
|||
The state of these components is completely independent. Hooks are a way to reuse *stateful logic*, not state itself. In fact, each *call* to a Hook has a competely isolated state -- so you can even use the same custom Hook twice in one component. |
|||
|
|||
Custom Hooks are more of a convention than a feature. If a function's name starts with "`use`" and it calls other Hooks, we say it is a custom Hook. The `useSomething` naming convention is how our linter plugin is able to find bugs in the code using Hooks. |
|||
|
|||
You can write custom Hooks that cover a wide range of use cases like form handling, animation, declarative subscriptions, timers, and probably many more we haven't considered. We are excited to see what custom Hooks the React community will come up with. |
|||
|
|||
>Detailed Explanation |
|||
> |
|||
>You can learn more about custom Hooks on a dedicated page: [Writing Custom Hooks](/docs/hooks-custom.html). |
|||
|
|||
## 🔌 Other Hooks |
|||
|
|||
There are a few less commonly used built-in Hooks that you might find useful. For example, [`useContext`](/docs/hooks-reference.html#usecontext) lets you subscribe to React context without introducing nesting: |
|||
|
|||
```js{2,3} |
|||
function Example() { |
|||
const locale = useContext(LocaleContext); |
|||
const theme = useContext(ThemeContext); |
|||
// ... |
|||
} |
|||
``` |
|||
|
|||
And [`useReducer`](/docs/hooks-reference.html#usereducer) lets you manage local state of complex components with a reducer: |
|||
|
|||
```js{2} |
|||
function Todos() { |
|||
const [todos, dispatch] = useReducer(todosReducer); |
|||
// ... |
|||
``` |
|||
|
|||
>Detailed Explanation |
|||
> |
|||
>You can learn more about all the built-in Hooks on a dedicated page: [Hooks API Reference](/docs/hooks-reference.html). |
|||
|
|||
## Next Steps |
|||
|
|||
Phew, that was fast! If some things didn't quite make sense or you'd like to learn more in detail, you can read the next pages, starting with the [State Hook](/docs/hooks-state.html) documentation. |
|||
|
|||
You can also check out the [Hooks API reference](/docs/hooks-reference.html) and the [Hooks FAQ](/docs/hooks-faq.html). |
|||
|
|||
Finally, don't miss the [introduction page](/docs/hooks-intro.html) which explains *why* we're adding Hooks and how we'll start using them side by side with classes -- without rewriting our apps. |
@ -0,0 +1,358 @@ |
|||
--- |
|||
id: hooks-reference |
|||
title: Hooks API Reference |
|||
permalink: docs/hooks-reference.html |
|||
prev: hooks-custom.html |
|||
next: hooks-faq.html |
|||
--- |
|||
|
|||
*Hooks* are a new feature proposal that lets you use state and other React features without writing a class. They're currently in React v16.7.0-alpha and being discussed in [an open RFC](https://github.com/reactjs/rfcs/pulls). |
|||
|
|||
This page describes the APIs for the built-in Hooks in React. |
|||
|
|||
If you're new to Hooks, you might want to check out [the overview](/docs/hooks-overview.html) first. You may also find useful information in the [frequently asked questions](/docs/hooks-faq.html) section. |
|||
|
|||
- [Basic Hooks](#basic-hooks) |
|||
- [`useState`](#usestate) |
|||
- [`useEffect`](#useeffect) |
|||
- [`useContext`](#usecontext) |
|||
- [Additional Hooks](#additional-hooks) |
|||
- [`useReducer`](#usereducer) |
|||
- [`useCallback`](#usecallback) |
|||
- [`useMemo`](#usememo) |
|||
- [`useRef`](#useref) |
|||
- [`useImperativeMethods`](#useimperativemethods) |
|||
- [`useMutationEffect`](#usemutationeffect) |
|||
- [`useLayoutEffect`](#uselayouteffect) |
|||
|
|||
## Basic Hooks |
|||
|
|||
### `useState` |
|||
|
|||
```js |
|||
const [state, setState] = useState(initialState); |
|||
``` |
|||
|
|||
Returns a stateful value, and a function to update it. |
|||
|
|||
During the initial render, the returned state (`state`) is the same as the value passed as the first argument (`initialState`). |
|||
|
|||
The `setState` function is used to update the state. It accepts a new state value and enqueues a re-render of the component. |
|||
|
|||
```js |
|||
setState(newState); |
|||
``` |
|||
|
|||
During subsequent re-renders, the first value returned by `useState` will always be the most recent state after applying updates. |
|||
|
|||
#### Functional updates |
|||
|
|||
If the new state is computed using the previous state, you can pass a function to `setState`. The function will receive the previous value, and return an updated value. Here's an example of a counter component that uses both forms of `setState`: |
|||
|
|||
```js |
|||
function Counter({initialCount}) { |
|||
const [count, setCount] = useState(initialCount); |
|||
return ( |
|||
<> |
|||
Count: {count} |
|||
<button onClick={() => setCount(0)}>Reset</button> |
|||
<button onClick={() => setCount(prevCount => prevCount + 1)}>+</button> |
|||
<button onClick={() => setCount(prevCount => prevCount - 1)}>-</button> |
|||
</> |
|||
); |
|||
} |
|||
``` |
|||
|
|||
The "+" and "-" buttons use the functional form, because the updated value is based on the previous value. But the "Reset" button uses the normal form, because it always sets the count back to 0. |
|||
|
|||
> Note |
|||
> |
|||
> Unlike the `setState` method found in class components, `useState` does not automatically merge update objects. You can replicate this behavior by combining the function updater form with object spread syntax: |
|||
> |
|||
> ```js |
|||
> setState(prevState => { |
|||
> // Object.assign would also work |
|||
> return {...prevState, ...updatedValues}; |
|||
> }); |
|||
> ``` |
|||
> |
|||
> Another option is `useReducer`, which is more suited for managing state objects that contain multiple sub-values. |
|||
|
|||
#### Lazy initialization |
|||
|
|||
The `initialState` argument is the state used during the initial render. In subsequent renders, it is disregarded. If the initial state is the result of an expensive computation, you may provide a function instead, which will be executed only on the initial render: |
|||
|
|||
```js |
|||
const [state, setState] = useState(() => { |
|||
const initialState = someExpensiveComputation(props); |
|||
return initialState; |
|||
}); |
|||
``` |
|||
|
|||
### `useEffect` |
|||
|
|||
```js |
|||
useEffect(didUpdate); |
|||
``` |
|||
|
|||
Accepts a function that contains imperative, possibly effectful code. |
|||
|
|||
Mutations, subscriptions, timers, logging, and other side effects are not allowed inside the main body of a function component (referred to as React's _render phase_). Doing so will lead to confusing bugs and inconsistencies in the UI. |
|||
|
|||
Instead, use `useEffect`. The function passed to `useEffect` will run after the render is committed to the screen. Think of effects as an escape hatch from React's purely functional world into the imperative world. |
|||
|
|||
By default, effects run after every completed render, but you can choose to fire it [only when certain values have changed](#conditionally-firing-an-effect). |
|||
|
|||
#### Cleaning up an effect |
|||
|
|||
Often, effects create resources that need to be cleaned up before the component leaves the screen, such as a subscription or timer ID. To do this, the function passed to `useEffect` may return a clean-up function. For example, to create a subscription |
|||
|
|||
```js |
|||
useEffect(() => { |
|||
const subscription = props.source.subscribe(); |
|||
return () => { |
|||
// Clean up the subscription |
|||
subscription.unsubscribe(); |
|||
}; |
|||
}); |
|||
``` |
|||
|
|||
The clean-up function runs before the component is removed from the UI to prevent memory leaks. Additionally, if a component renders multiple times (as they typically do), the **previous effect is cleaned up before executing the next effect**. In our example, this means a new subscription is created on every update. To avoid firing an effect on every update, refer to the next section. |
|||
|
|||
#### Timing of effects |
|||
|
|||
Unlike `componentDidMount` and `componentDidUpdate`, the function passed to `useEffect` fires **after** layout and paint, during a deferred event. This makes it suitable for the many common side effects, like setting up subscriptions and event handlers, because most types of work shouldn't block the browser from updating the screen. |
|||
|
|||
However, not all effects can be deferred. For example, a DOM mutation that is visible to the user must fire synchronously before the next paint so that the user does not perceive a visual inconsistency. (The distinction is conceptually similar to passive versus active event listeners.) For these types of effects, React provides two additional Hooks: [`useMutationEffect`](#usemutationeffect) and [`useLayoutEffect`](#uselayouteffect). These Hooks have the same signature as `useEffect`, and only differ in when they are fired. |
|||
|
|||
Although `useEffect` is deferred until after the browser has painted, it's guaranteed to fire before any new renders. React will always flush a previous render's effects before starting a new update. |
|||
|
|||
#### Conditionally firing an effect |
|||
|
|||
The default behavior for effects is to fire the effect after every completed render. That way an effect is always recreated if one of its inputs changes. |
|||
|
|||
However, this may be overkill in some cases, like the subscription example from the previous section. We don't need to create a new subscription on every update, only if the `source` props has changed. |
|||
|
|||
To implement this, pass a second argument to `useEffect` that is the array of values that the effect depends on. Our updated example now looks like this: |
|||
|
|||
```js |
|||
useEffect( |
|||
() => { |
|||
const subscription = props.source.subscribe(); |
|||
return () => { |
|||
subscription.unsubscribe(); |
|||
}; |
|||
}, |
|||
[props.source], |
|||
); |
|||
``` |
|||
|
|||
Now the subscription will only be recreated when `props.source` changes. |
|||
|
|||
Passing in an empty array `[]` of inputs tells React that your effect doesn't depend on any values from the component, so that effect would run only on mount and unmount, never on updates. |
|||
|
|||
> Note |
|||
> |
|||
> The array of inputs is not passed as arguments to the effect function. Conceptually, though, that's what they represent: every value referenced inside the effect function should also appear in the inputs array. In the future, a sufficiently advanced compiler could create this array automatically. |
|||
|
|||
### `useContext` |
|||
|
|||
```js |
|||
const context = useContext(Context); |
|||
``` |
|||
|
|||
Accepts a context object (the value returned from `React.createContext`) and returns the current context value, as given by the nearest context provider for the given context. |
|||
|
|||
When the provider updates, this Hook will trigger a rerender with the latest context value. |
|||
|
|||
## Additional Hooks |
|||
|
|||
The following Hooks are either variants of the basic ones from the previous section, or only needed for specific edge cases. Don't stress about learning them up front. |
|||
|
|||
### `useReducer` |
|||
|
|||
```js |
|||
const [state, dispatch] = useReducer(reducer, initialState); |
|||
``` |
|||
|
|||
An alternative to [`useState`](https://our.intern.facebook.com/intern/wiki/React_Hooks/#usestate). Accepts a reducer of type `(state, action) => newState`, and returns the current state paired with a `dispatch` method. (If you're familiar with Redux, you already know how this works.) |
|||
|
|||
Here's the counter example from the [`useState`](https://our.intern.facebook.com/intern/wiki/React_Hooks/#usestate) section, rewritten to use a reducer: |
|||
|
|||
```js |
|||
const initialState = {count: 0}; |
|||
|
|||
function reducer(state, action) { |
|||
switch (action.type) { |
|||
case 'reset': |
|||
return initialState; |
|||
case 'increment': |
|||
return {count: state.count + 1}; |
|||
case 'decrement': |
|||
return {count: state.count - 1}; |
|||
} |
|||
} |
|||
|
|||
function Counter({initialCount}) { |
|||
const [state, dispatch] = useReducer(reducer, initialState); |
|||
return ( |
|||
<> |
|||
Count: {state.count} |
|||
<button onClick={() => dispatch({type: 'reset'})}> |
|||
Reset |
|||
</button> |
|||
<button onClick={() => dispatch({type: 'increment'})}>+</button> |
|||
<button onClick={() => dispatch({type: 'decrement'})}>-</button> |
|||
</> |
|||
); |
|||
} |
|||
``` |
|||
|
|||
#### Lazy initialization |
|||
|
|||
`useReducer` accepts an optional third argument, `initialAction`. If provided, the initial action is applied during the initial render. This is useful for computing an initial state that includes values passed via props: |
|||
|
|||
```js |
|||
const initialState = {count: 0}; |
|||
|
|||
function reducer(state, action) { |
|||
switch (action.type) { |
|||
case 'reset': |
|||
return {count: action.payload}; |
|||
case 'increment': |
|||
return {count: state.count + 1}; |
|||
case 'decrement': |
|||
return {count: state.count - 1}; |
|||
} |
|||
} |
|||
|
|||
function Counter({initialCount}) { |
|||
const [state, dispatch] = useReducer( |
|||
reducer, |
|||
initialState, |
|||
{type: 'reset', payload: initialCount}, |
|||
); |
|||
|
|||
return ( |
|||
<> |
|||
Count: {state.count} |
|||
<button |
|||
onClick={() => dispatch({type: 'reset', payload: initialCount})}> |
|||
Reset |
|||
</button> |
|||
<button onClick={() => dispatch({type: 'increment'})}>+</button> |
|||
<button onClick={() => dispatch({type: 'decrement'})}>-</button> |
|||
</> |
|||
); |
|||
} |
|||
``` |
|||
|
|||
`useReducer` is usually preferable to `useState` when you have complex state logic that involves multiple sub-values. It also lets you optimize performance for components that trigger deep updates because [you can pass `dispatch` down instead of callbacks](/docs/hooks-faq.html#how-to-avoid-passing-callbacks-down). |
|||
|
|||
### `useCallback` |
|||
|
|||
```js |
|||
const memoizedCallback = useCallback( |
|||
() => { |
|||
doSomething(a, b); |
|||
}, |
|||
[a, b], |
|||
); |
|||
``` |
|||
|
|||
Returns a [memoized](https://en.wikipedia.org/wiki/Memoization) callback. |
|||
|
|||
Pass an inline callback and an array of inputs. `useCallback` will return a memoized version of the callback that only changes if one of the inputs has changed. This is useful when passing callbacks to optimized child components that rely on reference equality to prevent unnecessary renders (e.g. `shouldComponentUpdate`). |
|||
|
|||
`useCallback(fn, inputs)` is equivalent to `useMemo(() => fn, inputs)`. |
|||
|
|||
> Note |
|||
> |
|||
> The array of inputs is not passed as arguments to the callback. Conceptually, though, that's what they represent: every value referenced inside the callback should also appear in the inputs array. In the future, a sufficiently advanced compiler could create this array automatically. |
|||
|
|||
### `useMemo` |
|||
|
|||
```js |
|||
const memoizedValue = useMemo(() => computeExpensiveValue(a, b), [a, b]); |
|||
``` |
|||
|
|||
Returns a [memoized](https://en.wikipedia.org/wiki/Memoization) value. |
|||
|
|||
Pass a "create" function and an array of inputs. `useMemo` will only recompute the memoized value when one of the inputs has changed. This optimization helps to avoid expensive calculations on every render. |
|||
|
|||
If no array is provided, a new value will be computed whenever a new function instance is passed as the first argument. (With an inline function, on every render.) |
|||
|
|||
> Note |
|||
> |
|||
> The array of inputs is not passed as arguments to the function. Conceptually, though, that's what they represent: every value referenced inside the function should also appear in the inputs array. In the future, a sufficiently advanced compiler could create this array automatically. |
|||
|
|||
### `useRef` |
|||
|
|||
```js |
|||
const refContainer = useRef(initialValue); |
|||
``` |
|||
|
|||
`useRef` returns a mutable ref object whose `.current` property is initialized to the passed argument (`initialValue`). The returned object will persist for the full lifetime of the component. |
|||
|
|||
A common use case is to access a child imperatively: |
|||
|
|||
```js |
|||
function TextInputWithFocusButton() { |
|||
const inputEl = useRef(null); |
|||
const onButtonClick = () => { |
|||
// `current` points to the mounted text input element |
|||
inputEl.current.focus(); |
|||
}; |
|||
return ( |
|||
<> |
|||
<input ref={inputEl} type="text" /> |
|||
<button onClick={onButtonClick}>Focus the input</button> |
|||
</> |
|||
); |
|||
} |
|||
``` |
|||
|
|||
Note that `useRef()` is useful for more than the `ref` attribute. It's [handy for keeping any mutable value around](/docs/hooks-faq.html#is-there-something-like-instance-variables) similar to how you'd use instance fields in classes. |
|||
|
|||
### `useImperativeMethods` |
|||
|
|||
```js |
|||
useImperativeMethods(ref, createInstance, [inputs]) |
|||
``` |
|||
|
|||
`useImperativeMethods` customizes the instance value that is exposed to parent components when using `ref`. As always, imperative code using refs should be avoided in most cases. `useImperativeMethods` should be used with `forwardRef`: |
|||
|
|||
```js |
|||
function FancyInput(props, ref) { |
|||
const inputRef = useRef(); |
|||
useImperativeMethods(ref, () => ({ |
|||
focus: () => { |
|||
inputRef.current.focus(); |
|||
} |
|||
})); |
|||
return <input ref={inputRef} ... />; |
|||
} |
|||
FancyInput = forwardRef(FancyInput); |
|||
``` |
|||
|
|||
In this example, a parent component that renders `<FancyInput ref={fancyInputRef} />` would be able to call `fancyInputRef.current.focus()`. |
|||
|
|||
### `useMutationEffect` |
|||
|
|||
The signature is identical to `useEffect`, but it fires synchronously during the same phase that React performs its DOM mutations, before sibling components have been updated. Use this to perform custom DOM mutations. |
|||
|
|||
Prefer the standard `useEffect` when possible to avoid blocking visual updates. |
|||
|
|||
>Note |
|||
> |
|||
>Avoid reading from the DOM in `useMutationEffect`. If you do, you can cause performance problems by introducing [layout thrash](https://developers.google.com/web/fundamentals/performance/rendering/avoid-large-complex-layouts-and-layout-thrashing). When reading computed styles or layout information, `useLayoutEffect` is more appropriate. |
|||
|
|||
### `useLayoutEffect` |
|||
|
|||
The signature is identical to `useEffect`, but it fires synchronously *after* all DOM mutations. Use this to read layout from the DOM and synchronously re-render. Updates scheduled inside `useLayoutEffect` will be flushed synchronously, before the browser has a chance to paint. |
|||
|
|||
Prefer the standard `useEffect` when possible to avoid blocking visual updates. |
|||
|
|||
> Tip |
|||
> |
|||
> If you're migrating code from a class component, `useLayoutEffect` fires in the same phase as `componentDidMount` and `componentDidUpdate`, so if you're unsure of which effect Hook to use, it's probably the least risky. |
@ -0,0 +1,137 @@ |
|||
--- |
|||
id: hooks-rules |
|||
title: Rules of Hooks |
|||
permalink: docs/hooks-rules.html |
|||
next: hooks-custom.html |
|||
prev: hooks-effect.html |
|||
--- |
|||
|
|||
*Hooks* are a new feature proposal that lets you use state and other React features without writing a class. They're currently in React v16.7.0-alpha and being discussed in [an open RFC](https://github.com/reactjs/rfcs/pulls). |
|||
|
|||
Hooks are JavaScript functions, but you need to follow two rules when using them. We provide a [linter plugin](https://www.npmjs.com/package/eslint-plugin-react-hooks) to enforce these rules automatically: |
|||
|
|||
### Only Call Hooks at the Top Level |
|||
|
|||
**Don't call Hooks inside loops, conditions, or nested functions.** Instead, always use Hooks at the top level of your React function. By following this rule, you ensure that Hooks are called in the same order each time a component renders. That's what allows React to correctly preserve the state of Hooks between multiple `useState` and `useEffect` calls. (If you're curious, we'll explain this in depth [below](#explanation).) |
|||
|
|||
### Only Call Hooks from React Functions |
|||
|
|||
**Don't call Hooks from regular JavaScript functions.** Instead, you can: |
|||
|
|||
* ✅ Call Hooks from React function components. |
|||
* ✅ Call Hooks from custom Hooks (we'll learn about them [on the next page](/docs/hooks-custom.html)). |
|||
|
|||
By following this rule, you ensure that all stateful logic in a component is clearly visible from its source code. |
|||
|
|||
## ESLint Plugin |
|||
|
|||
We released an ESLint plugin called [`eslint-plugin-react-hooks`](https://www.npmjs.com/package/eslint-plugin-react-hooks) that enforces these two rules. You can add this plugin to your project if you'd like to try it: |
|||
|
|||
```bash |
|||
npm install eslint-plugin-react-hooks@next |
|||
``` |
|||
|
|||
```js |
|||
// Your ESLint configuration |
|||
{ |
|||
"plugins": [ |
|||
// ... |
|||
"react-hooks" |
|||
], |
|||
"rules": { |
|||
// ... |
|||
"react-hooks/rules-of-hooks": "error" |
|||
} |
|||
} |
|||
``` |
|||
|
|||
In the future, we intend to include this plugin by default into Create React App and similar toolkits. |
|||
|
|||
**You can skip to the next page explaining how to write [custom Hooks](/docs/hooks-custom.html) now.** On this page, we'll continue by explaining the reasoning behind these rules. |
|||
|
|||
## Explanation |
|||
|
|||
As we [learned earlier](/docs/hooks-state.html#tip-using-multiple-state-variables), we can use multiple State or Effect Hooks in a single component: |
|||
|
|||
```js |
|||
function Form() { |
|||
// 1. Use the name state variable |
|||
const [name, setName] = useState('Mary'); |
|||
|
|||
// 2. Use an effect for persisting the form |
|||
useEffect(function persistForm() { |
|||
localStorage.setItem('formData', name); |
|||
}); |
|||
|
|||
// 3. Use the surname state variable |
|||
const [surname, setSurname] = useState('Poppins'); |
|||
|
|||
// 4. Use an effect for updating the title |
|||
useEffect(function updateTitle() { |
|||
document.title = name + ' ' + surname; |
|||
}); |
|||
|
|||
// ... |
|||
} |
|||
``` |
|||
|
|||
So how does React know which state corresponds to which `useState` call? The answer is that **React relies on the order in which Hooks are called**. Our example works because the order of the Hook calls is the same on every render: |
|||
|
|||
```js |
|||
// ------------ |
|||
// First render |
|||
// ------------ |
|||
useState('Mary') // 1. Initialize the name state variable with 'Mary' |
|||
useEffect(persistForm) // 2. Add an effect for persisting the form |
|||
useState('Poppins') // 3. Initialize the surname state variable with 'Poppins' |
|||
useEffect(updateTitle) // 4. Add an effect for updating the title |
|||
|
|||
// ------------- |
|||
// Second render |
|||
// ------------- |
|||
useState('Mary') // 1. Read the name state variable (argument is ignored) |
|||
useEffect(persistForm) // 2. Replace the effect for persisting the form |
|||
useState('Poppins') // 3. Read the surname state variable (argument is ignored) |
|||
useEffect(updateTitle) // 4. Replace the effect for updating the title |
|||
|
|||
// ... |
|||
``` |
|||
|
|||
As long as the order of the Hook calls is the same between renders, React can associate some local state with each of them. But what happens if we put a Hook call (for example, the `persistForm` effect) inside a condition? |
|||
|
|||
```js |
|||
// 🔴 We're breaking the first rule by using a Hook in a condition |
|||
if (name !== '') { |
|||
useEffect(function persistForm() { |
|||
localStorage.setItem('formData', name); |
|||
}); |
|||
} |
|||
``` |
|||
|
|||
The `name !== ''` condition is `true` on the first render, so we run this Hook. However, on the next render the user might clear the form, making the condition `false`. Now that we skip this Hook during rendering, the order of the Hook calls becomes different: |
|||
|
|||
```js |
|||
useState('Mary') // 1. Read the name state variable (argument is ignored) |
|||
// useEffect(persistForm) // 🔴 This Hook was skipped! |
|||
useState('Poppins') // 🔴 2 (but was 3). Fail to read the surname state variable |
|||
useEffect(updateTitle) // 🔴 3 (but was 4). Fail to replace the effect |
|||
``` |
|||
|
|||
React wouldn't know what to return for the second `useState` Hook call. React expected that the second Hook call in this component corresponds to the `persistForm` effect, just like during the previous render, but it doesn't anymore. From that point, every next Hook call after the one we skipped would also shift by one, leading to bugs. |
|||
|
|||
**This is why Hooks must be called on the top level of our components.** If we want to run an effect conditionally, we can put that condition *inside* our Hook: |
|||
|
|||
```js |
|||
useEffect(function persistForm() { |
|||
// 👍 We're not breaking the first rule anymore |
|||
if (name !== '') { |
|||
localStorage.setItem('formData', name); |
|||
} |
|||
}); |
|||
``` |
|||
|
|||
**Note that you don't need to worry about this problem if you use the [provided lint rule](https://www.npmjs.com/package/eslint-plugin-react-hooks).** But now you also know *why* Hooks work this way, and which issues the rule is preventing. |
|||
|
|||
## Next Steps |
|||
|
|||
Finally, we're ready to learn about [writing custom Hooks](/docs/hooks-custom.html)! Custom Hooks let you combine Hooks provided by React into your own abstractions, and reuse common stateful logic between different components. |
@ -0,0 +1,280 @@ |
|||
--- |
|||
id: hooks-state |
|||
title: Using the State Hook |
|||
permalink: docs/hooks-state.html |
|||
next: hooks-effect.html |
|||
prev: hooks-overview.html |
|||
--- |
|||
|
|||
*Hooks* are a new feature proposal that lets you use state and other React features without writing a class. They're currently in React v16.7.0-alpha and being discussed in [an open RFC](https://github.com/reactjs/rfcs/pulls). |
|||
|
|||
The [previous page](/docs/hooks-intro.html) introduced Hooks with this example: |
|||
|
|||
```js{4-5} |
|||
import { useState } from 'react'; |
|||
|
|||
function Example() { |
|||
// Declare a new state variable, which we'll call "count" |
|||
const [count, setCount] = useState(0); |
|||
|
|||
return ( |
|||
<div> |
|||
<p>You clicked {count} times</p> |
|||
<button onClick={() => setCount(count + 1)}> |
|||
Click me |
|||
</button> |
|||
</div> |
|||
); |
|||
} |
|||
``` |
|||
|
|||
We'll start learning about Hooks by comparing this code to an equivalent class example. |
|||
|
|||
## Equivalent Class Example |
|||
|
|||
If you used classes in React before, this code should look familiar: |
|||
|
|||
```js |
|||
class Example extends React.Component { |
|||
constructor(props) { |
|||
super(props); |
|||
this.state = { |
|||
count: 0 |
|||
}; |
|||
} |
|||
|
|||
render() { |
|||
return ( |
|||
<div> |
|||
<p>You clicked {this.state.count} times</p> |
|||
<button onClick={() => this.setState({ count: this.state.count + 1 })}> |
|||
Click me |
|||
</button> |
|||
</div> |
|||
); |
|||
} |
|||
} |
|||
``` |
|||
|
|||
The state starts as `{ count: 0 }`, and we increment `state.count` when the user clicks a button by calling `this.setState()`. We'll use snippets from this class throughout the page. |
|||
|
|||
>Note |
|||
> |
|||
>You might be wondering why we're using a counter here instead of a more realistic example. This is to help us focus on the API while we're still making our first steps with Hooks. |
|||
|
|||
## Hooks and Function Components |
|||
|
|||
As a reminder, function components in React look like this: |
|||
|
|||
```js |
|||
const Example = (props) => { |
|||
// You can use Hooks here! |
|||
return <div />; |
|||
} |
|||
``` |
|||
|
|||
or this: |
|||
|
|||
```js |
|||
function Example(props) { |
|||
// You can use Hooks here! |
|||
return <div />; |
|||
} |
|||
``` |
|||
|
|||
You might have previously known these as "stateless components". We're now introducing the ability to use React state from these, so we prefer the name "function components". |
|||
|
|||
Hooks **don't** work inside classes. But you can use them instead of writing classes. |
|||
|
|||
## What's a Hook? |
|||
|
|||
Our new example starts by importing the `useState` Hook from React: |
|||
|
|||
```js{1} |
|||
import { useState } from 'react'; |
|||
|
|||
function Example() { |
|||
// ... |
|||
} |
|||
``` |
|||
|
|||
**What is a Hook?** A Hook is a special function that lets you "hook into" React features. For example, `useState` is a Hook that lets you add React state to function components. We'll learn other Hooks later. |
|||
|
|||
**When would I use a Hook?** If you write a function component and realize you need to add some state to it, previously you had to convert it to a class. Now you can use a Hook inside the existing function component. We're going to do that right now! |
|||
|
|||
>Note: |
|||
> |
|||
>There are some special rules about where you can and can't use Hooks within a component. We'll learn them in [Rules of Hooks](/docs/hooks-rules.html). |
|||
|
|||
## Declaring a State Variable |
|||
|
|||
In a class, we initialize the `count` state to `0` by setting `this.state` to `{ count: 0 }` in the constructor: |
|||
|
|||
```js{4-6} |
|||
class Example extends React.Component { |
|||
constructor(props) { |
|||
super(props); |
|||
this.state = { |
|||
count: 0 |
|||
}; |
|||
} |
|||
``` |
|||
|
|||
In a function component, we have no `this`, so we can't assign or read `this.state`. Instead, we call the `useState` Hook directly inside our component: |
|||
|
|||
```js{4,5} |
|||
import { useState } from 'react'; |
|||
|
|||
function Example() { |
|||
// Declare a new state variable, which we'll call "count" |
|||
const [count, setCount] = useState(0); |
|||
``` |
|||
|
|||
**What does calling `useState` do?** It declares a "state variable". Our variable is called `count` but we could call it anything else, like `banana`. This is a way to "preserve" some values between the function calls — `useState` is a new way to use the exact same capabilities that `this.state` provides in a class. Normally, variables "disappear" when the function exits but state variables are preserved by React. |
|||
|
|||
**What do we pass to `useState` as an argument?** The only argument to the `useState()` Hook is the initial state. Unlike with classes, the state doesn't have to be an object. We can keep a number or a string if that's all we need. In our example, we just want a number for how many times the user clicked, so pass `0` as initial state for our variable. (If we wanted to store two different values in state, we would call `useState()` twice.) |
|||
|
|||
**What does `useState` return?** It returns a pair of values: the current state and a function that updates it. This is why we write `const [count, setCount] = useState()`. This is similar to `this.state.count` and `this.setState` in a class, except you get them in a pair. If you're not familiar with the syntax we used, we'll come back to it [at the bottom of this page](/docs/hooks-state.html#tip-what-do-square-brackets-mean). |
|||
|
|||
Now that we know what the `useState` Hook does, our example should make more sense: |
|||
|
|||
```js{4,5} |
|||
import { useState } from 'react'; |
|||
|
|||
function Example() { |
|||
// Declare a new state variable, which we'll call "count" |
|||
const [count, setCount] = useState(0); |
|||
``` |
|||
|
|||
We declare a state variable called `count`, and set it to `0`. React will remember its current value between re-renders, and provide the most recent one to our function. If we want to update the current `count`, we can call `setCount`. |
|||
|
|||
>Note |
|||
> |
|||
>You might be wondering: why is `useState` not named `createState` instead? |
|||
> |
|||
>"Create" wouldn't be quite accurate because the state is only created the first time our component renders. During the next renders, `useState` gives us the current state. Otherwise it wouldn't be "state" at all! There's also a reason why Hook names *always* start with `use`. We'll learn why later in the [Rules of Hooks](/docs/hooks-rules.html). |
|||
|
|||
## Reading State |
|||
|
|||
When we want to display the current count in a class, we read `this.state.count`: |
|||
|
|||
```js |
|||
<p>You clicked {this.state.count} times</p> |
|||
``` |
|||
|
|||
In a function, we can use `count` directly: |
|||
|
|||
|
|||
```js |
|||
<p>You clicked {count} times</p> |
|||
``` |
|||
|
|||
## Updating State |
|||
|
|||
In a class, we need to call `this.setState()` to update the `count` state: |
|||
|
|||
```js{1} |
|||
<button onClick={() => this.setState({ count: this.state.count + 1 })}> |
|||
Click me |
|||
</button> |
|||
``` |
|||
|
|||
In a function, we already have `setCount` and `count` as variables so we don't need `this`: |
|||
|
|||
```js{1} |
|||
<button onClick={() => setCount(count + 1)}> |
|||
Click me |
|||
</button> |
|||
``` |
|||
|
|||
## Recap |
|||
|
|||
Let's now **recap what we learned line by line** and check our understanding. |
|||
|
|||
<!-- |
|||
I'm not proud of this line markup. Please somebody fix this. |
|||
But if GitHub got away with it for years we can cheat. |
|||
--> |
|||
```js{1,4,9} |
|||
1: import { useState } from 'react'; |
|||
2: |
|||
3: function Example() { |
|||
4: const [count, setCount] = useState(0); |
|||
5: |
|||
6: return ( |
|||
7: <div> |
|||
8: <p>You clicked {count} times</p> |
|||
9: <button onClick={() => setCount(count + 1)}> |
|||
10: Click me |
|||
11: </button> |
|||
12: </div> |
|||
13: ); |
|||
14: } |
|||
``` |
|||
|
|||
* **Line 1:** We import the `useState` Hook from React. It lets us keep local state in a function component. |
|||
* **Line 4:** Inside the `Example` component, we declare a new state variable by calling the `useState` Hook. It returns a pair of values, to which we give names. We're calling our variable `count` because it holds the number of button clicks. We initialize it to zero by passing `0` as the only `useState` argument. The second returned item is itself a function. It lets us update the `count` so we'll name it `setCount`. |
|||
* **Line 9:** When the user clicks, we call `setCount` with a new value. React will then re-render the `Example` component, passing the new `count` value to it. |
|||
|
|||
This might seem like a lot to take in at first. Don't rush it! If you're lost in the explanation, look at the code above again and try to read it from top to bottom. We promise that once you try to "forget" how state works in classes, and look at this code with fresh eyes, it will make sense. |
|||
|
|||
### Tip: What Do Square Brackets Mean? |
|||
|
|||
You might have noticed the square brackets when we declare a state variable: |
|||
|
|||
```js |
|||
const [count, setCount] = useState(0); |
|||
``` |
|||
|
|||
The names on the left aren't a part of the React API. You can name your own state variables: |
|||
|
|||
```js |
|||
const [fruit, setFruit] = useState('banana'); |
|||
``` |
|||
|
|||
This JavaScript syntax is called ["array destructuring"](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Destructuring_assignment#Array_destructuring). It means that we're making two new variables `fruit` and `setFruit`, where `fruit` is set to the first value returned by `useState`, and `setFruit` is the second. It is equivalent to this code: |
|||
|
|||
```js |
|||
var fruitStateVariable = useState('banana'); // Returns a pair |
|||
var fruit = fruitStateVariable[0]; // First item in a pair |
|||
var setFruit = fruitStateVariable[1]; // Second item in a pair |
|||
``` |
|||
|
|||
When we declare a state variable with `useState`, it returns a pair — an array with two items. The first item is the current value, and the second is a function that lets us update it. Using `[0]` and `[1]` to access them is a bit confusing because they have a specific meaning. This is why we use array destructuring instead. |
|||
|
|||
>Note |
|||
> |
|||
>You might be curious how React knows which component `useState` corresponds to since we're not passing anything like `this` back to React. We'll answer [this question](/docs/hooks-faq.html#how-does-react-associate-hook-calls-with-components) and many others in the FAQ section. |
|||
|
|||
### Tip: Using Multiple State Variables |
|||
|
|||
Declaring state variables as a pair of `[something, setSomething]` is also handy because it lets us give *different* names to different state variables if we want to use more than one: |
|||
|
|||
```js |
|||
function ExampleWithManyStates() { |
|||
// Declare multiple state variables! |
|||
const [age, setAge] = useState(42); |
|||
const [fruit, setFruit] = useState('banana'); |
|||
const [todos, setTodos] = useState([{ text: 'Learn Hooks' }]); |
|||
``` |
|||
|
|||
In the above component, we have `age`, `fruit`, or `todos` as local variables, and we can update them individually: |
|||
|
|||
```js |
|||
function handleOrangeClick() { |
|||
// Similar to this.setState({ fruit: 'orange' }) |
|||
setFruit('orange'); |
|||
} |
|||
``` |
|||
|
|||
You **don't have to** use many state variables. State variables can hold objects and arrays just fine, so you can still group related data together. However, unlike `this.setState` in a class, updating a state variable always *replaces* it instead of merging it. |
|||
|
|||
We provide more recommendations on splitting independent state variables [in the FAQ](/docs/hooks-faq.html#should-i-use-one-or-many-state-variables). |
|||
|
|||
## Next Steps |
|||
|
|||
On this page we've learned about one of the Hooks provided by React, called `useState`. We're also sometimes going to refer to it as the "State Hook". It lets us add local state to React function components -- which we did for the first time ever! |
|||
|
|||
We also learned a little bit more about what Hooks are. Hooks are functions that let you "hook into" React features from function components. Their names always start with `use`, and there are more Hooks we haven't seen yet. |
|||
|
|||
**Now let's continue by [learning the next Hook: `useEffect`.](/docs/hooks-effect.html)** It lets you perform side effects in components, and is similar to lifecycle methods in classes. |
@ -1,2 +1,3 @@ |
|||
/html-jsx.html https://magic.reactjs.net/htmltojsx.htm 301 |
|||
/tips/controlled-input-null-value.html /docs/forms.html#controlled-input-null-value |
|||
/hooks /docs/hooks-intro.html |
|||
|
Loading…
Reference in new issue