How to Create a Responsive Navbar in React
Navigation menus are a central part of every website. In this tutorial, we will be creating a responsive navbar in React.
Prerequisites: Basic knowledge in HTML, CSS, and React.
The demo and source code can be found at:
Setup and Directory Structure
We will be using the Create React App boilerplate to get started.
After that command executes, let’s modify the structure of the created directory. In the src folder, let’s create a components folder. Then, in the components folder, we’ll create a layout folder. So our directory structure should now look like:
Even though we are just creating a navbar, it’s still a good idea to keep the project organized, that way the navbar we are creating can be easily reused on different projects.
Finally, let’s create a layout.js file in our layout folder, and add the following code to it:
Now, let’s make sure everything is working correctly by returning this component in our app.js file.
After running the development server, we should now see:
Creating the Navbar Component
Now that everything is set up, let’s begin creating the navbar component. In the layout folder, let’s add a navbar.js file, and a navbar.module.css file. So, our layout directory now looks like:
In the navbar.js file, add the following code:
All we are doing here is creating the markup for a navigation menu. Now, let’s add the styles in a mobile-first approach.
One thing to note with the styling is that we set the position of the .nav class to relative, that way we can place the hamburger menu exactly where we would like it. In this case, our hamburger menu is given the class name .hamburger and will be positioned at the top right corner relative to the html element with the class name of .nav. Other than that, the styling is straightforward.
Finally, let’s return the navbar component in our layout.js file.
So, at this point, our navigation menu should look like:
We are making progress, but we definitely need to make some improvements. For starters, we should prevent the content from hitting the sides of the viewport. Let’s address that right now.
Creating the Container Component
To ensure the page content does not hit the sides of the viewport, let’s create a container component. We will do this by first creating a new folder called container inside the components folder. In the container folder let’s add a container.js file, and a container.module.css file. So, the directory structure show now look like
The code for the container component is super simple:
And for the styling:
All we are doing here is wrapping the child elements in a div with a CSS class of container. This ensures that the child elements will never stretch beyond the defined width in our CSS class, which in this case is 1100px. In the case that the viewport is smaller than 1100px, we also add padding on the left and right side of the container, which ensures that the page content will never hit the side edges of the viewport.
Now, let’s put this container component to use. In our layout.js file let’s add the highlighted lines of code:
Also, let’s add the same three lines in our navbar.js file:
With these changes in place, our navigation menu should now look like:
This is much better, however, we still need to fix a few more issues. Right now, the hamburger menu is always open, which is not desirable. Let’s add some code that will allow us to open and close the menu.
Opening and Closing the Hamburger Menu
To open and close the hamburger menu, we will add state to our layout component. This state will keep track of when to show the hamburger links and when to hide them. To add state, let’s convert the layout component to a class based component.
Let’s also add a method in this class that will change whether or not the hamburger links should show.
In this method, we are using the opposite value of the previous state to set the new state. So, if hamburgerLinksShouldShow was true, it will now be switched to false. In other words, if the hamburger menu is open, then the next time this method is called, the hamburger menu should close, and vice-versa.
Let’s now pass down whether or not the hamburger links should show, and the toggle method as props to our navbar component. This will look like:
The full code should look like:
Now, let’s use these props in our navbar component.
Before I explain what’s going on, let’s add the following styles to our navbar stylesheet.
So, now by default, our hamburger links will not show at all. When the hamburger menu is clicked, then our toggle method will be called, and the hamburgerLinksShouldShow value will be set to true. Since this is now true, our showHamburgerLinks CSS class will get added, and our hamburger menu will open.
Improving the Styling For Large Screens
Great, so at this point we have a menu that will open and close, and is perfect for smaller devices. However, when the screen size is big enough, we should remove the hamburger menu altogether, and instead use a normal navigation bar. Let’s work on that right now.
In our navbar stylesheet, let’s add the following media query underneath all of the existing styles:
So, the full code should look like:
And with those styles added, the navbar now looks great on both smaller devices and larger devices!
Adding Animation to the Hamburger Menu
The last modification we are going to make to our hamburger menu is to add animation to it. Currently, when we open and close it, the change is instant. Let’s improve this so that it opens and closes smoothly.
We are going to use fixed height values in order to add this animation. In our navbar stylesheet, let’s add the highlighted code:
So, instead of going from display: none, to display: block, we go from a height of 0 to a height of 114px. Using these fixed heights allows us to add the transition property, which creates a smooth effect. The height of 114px was used because that was the original height of the element when it’s display property was set to block. Chrome’s Developer Tools were used to check this.
Conclusion
Awesome, we created a navbar that can now be used on any React project! Here are the links again for the demo, and the source code.
Hopefully, this tutorial was useful to you, and thanks for reading!