How to write and use custom hooks in React.js
Katsiaryna (Kate) Lupachova
Posted on November 2, 2020
Originally posted on my personal blog.
What is a Hook in react
Hooks as a new feature was introduced in React 16.8 back in 1Q of 2019 and since then became one of the most valuable features in React.
Hook is a piece of functionality containing its state or other hooks that can be reused in multiple places across the app (or even across many apps if a hook is published as an NPM package).
There are several rules of hooks, most important of them are:
- only call hooks at the top level (not inside loops, conditions, or nested functions)
- call hooks from React function components (you actually can’t use hooks inside a class component)
- and name your custom hook starting with "use". Only this way React will be able to automatically check for a rules violation.
By default, React comes with several build-in hooks. The most widely used of them are:
- useState
- useEffect
- useContext
You can find a full list here.
Writing a custom hook
In this blog post, we'll look at the writing and using three custom hooks:
- useSmoothScroll - includes only one other React's hook
- useDocTitle - uses two other React's hooks
- useRoute + useDocTitle - combine two custom hooks
1. useSmoothScroll custom hook
As we established this earlier, basically hook is a function that uses other hooks. Let's try to write our custom hook!
Assume we'd like to scroll to some element by clicking on the button somewhere in our app. And for it to look nice we want the scroll to happen smoothly.
useSmoothScroll.js
import { useRef } from "react";
const useSmoothScroll = () => {
const ref = useRef();
const smoothScroll = () => ref.current.scrollIntoView({ behavior: "smooth" });
return [ref, smoothScroll];
};
export default useSmoothScroll;
In this custom hook, we are using useRef hook - a default React's hook that returns a mutable ref object. This ref object will be used as a way to access the DOM by passing it to a ref HTML attribute.
smoothScroll is a function that incapsulates Element.scrollIntoView() method.
And finally, the custom hook returns an array that contains a ref object and a function for smooth scrolling to this object (similar to useState React's hook).
Now we can use this custom hook in any component in the app. For example,
import React from "react";
import "./App.css";
import useSmoothScroll from "./hooks/useSmoothScroll";
function App() {
const [refToScroll, smoothScroll] = useSmoothScroll();
return (
<div className="App">
<button onClick={smoothScroll}>Scroll</button>
<div style={{ marginTop: "150vh" }} ref={refToScroll}>
I wanna be seen
</div>
</div>
);
}
export default App;
Here is this app in action:
This hook can be used multiple times across the app. And the important thing is that two components using the same hook do not share the state.
For instance, we can use the useSmoothScroll hook for marking another element to which we'd like to scroll and do so by clicking on another button.
useSmoothScroll.js
import React from "react";
import "./App.css";
import useSmoothScroll from "./hooks/useSmoothScroll";
function App() {
const [refToScroll, smoothScroll] = useSmoothScroll();
const [topRef, smoothScrollToTop] = useSmoothScroll();
return (
<div className="App">
<button ref={topRef} onClick={smoothScroll}>
Scroll
</button>
<div style={{ marginTop: "150vh" }} ref={refToScroll}>
I wanna be seen
</div>
<button onClick={smoothScrollToTop}>Go UP</button>
</div>
);
}
export default App;
Demo:
2. useDocTitle custom hook
A custom hook can use not only one other hook but as many as you wish. Now let's write a custom hook that contains two other hooks.
useDocTitle.js
import { useState, useEffect } from "react";
const useDocTitle = (title) => {
const [docTitle, setDocTitle] = useState(title);
useEffect(() => {
document.title = docTitle;
}, [docTitle]);
return [docTitle, setDocTitle];
};
export default useDocTitle;
This custom hook updates the current title of the document (the name of the current browser's tab).
In this hook two other hooks are used:
- useState - for saving and updating the state of the docTitle variable. Title prop is passed as an initial state value.
- useEffect - for setting the document.title property to a new value on every change of the docTitle variable.
And this hook can be used in the following way:
import React from "react";
import "./App.css";
import useDocTitle from "./hooks/useDocTitle";
function App() {
const [, setDocTitle] = useDocTitle("Start page");
return (
<div className="App">
<button onClick={() => setDocTitle("Updated title")}>
Change doc title
</button>
</div>
);
}
export default App;
The hook in action:
3. useRoute custom hook
Also, a custom hook can be used inside another custom hook.
Let's create a new custom hook - useRoute, which purpose is to update the browser URL without page reload.
useRoute.js
import { useState, useEffect } from "react";
const useRoute = (initialRoute) => {
const [route, setRoute] = useState(initialRoute);
useEffect(() => {
window.history.pushState(null, "", route);
}, [route]);
return [route, setRoute];
};
export default useRoute;
The structure of this hook is very similar to our previous useDocTitle hook structure. The main difference is that we use window.history.pushState() method inside the useEffect hook.
Now we are going to add this hook into the useDocTitle hook.
useDocTitle.js
import { useState, useEffect } from "react";
import useRoute from "./useRoute";
const useDocTitle = (title) => {
const [docTitle, setDocTitle] = useState(title);
const [, setRoute] = useRoute();
useEffect(() => {
document.title = docTitle;
setRoute(docTitle);
}, [docTitle]);
return [docTitle, setDocTitle];
};
export default useDocTitle;
Now the browser's URL will be updated at the same time as when the document title is changed.
So we can use this hook like this:
import React from "react";
import "./App.css";
import useDocTitle from "./hooks/useDocTitle";
function App() {
const [, setDocTitle] = useDocTitle("home");
return (
<div className="App">
<button onClick={() => setDocTitle("updated")}>
Change doc title and route
</button>
</div>
);
}
export default App;
And the demo of how it works:
Conclusion
As you see, writing and using custom hooks is pretty easy and straightforward. Depending on your use cases, you can write as many custom hooks as you like and then easily share them between different components of your app. So just have fun! And don't forget to wear a mask and wash your hands!
The complete source code of the tutorial is available in this GitHub repo.
Posted on November 2, 2020
Join Our Newsletter. No Spam, Only the good stuff.
Sign up to receive the latest update from our blog.