Recently I cloned the UI of Netflix's Landing Page using just React and Sass to practice CSS. Though it was simple, there were a lot of learning opportunities for someone like me to play with frontend development, and I quite enjoyed that challenge. Here's the link to the project if you are interested : Netflix Landing Page Clone.
In this post, we will be focusing on a section that you might have already seen on many other online platforms and learn how to implement that. It is called accordion, not the piano one thought 😂.
What is a web accordion?
It helps you manage collapsable content. They are useful when you want to toggle between hiding and showing a large amount of content.
This is what I built for the clone project:
If you click on one of the blocks, it would expand the content and look like this:
Let's get building 🔨
Note: For the sake of simplicity, I will use plain HTML, CSS, and Javascript to build this instead of React and Sass.
✏ First, let's create the structure of the accordion with HTML:
<div class="qna">
<button class="accordion">Question 1</button>
<div class="panel panel-closed">
<p>Answer of Question 1</p>
</div>
<button class="accordion">Question 2</button>
<div class="panel panel-closed">
<p>Answer of Question 2</p>
</div>
<button class="accordion">Question 3</button>
<div class="panel panel-closed">
<p>Answer of Question 3</p>
</div>
<button class="accordion">Question 4</button>
<div class="panel panel-closed">
<p>Answer of Question 4</p>
</div>
</div>
🎨 Now, we will add some basic styles and reset defaults with CSS:
/* Centers horizontally */
.qna{
max-width: 500px;
margin: 0 auto;
}
/* Button default resets */
button{
border: none;
outline: none;
cursor: pointer;
}
/* Styling the accordion buttons */
.accordion{
width: 100%;
background-color: #303030;
color: #fff;
font-size: 20px;
padding: 18px 20px;
margin-bottom: 1px;
text-align: left;
/* To push the plus sign to the right */
display: flex;
justify-content: space-between;
align-items: center;
}
/* adding the "plus" sign (+) */
.accordion::after{
content: '\02795';
font-size: 20px;
color: #fff;
}
.panel{
background-color: #303030;
color: #fff;
margin-bottom: 4px;
overflow: hidden;
}
.panel > p {
font-size: 20px;
padding: 10px 20px;
}
.panel-closed{
max-height: 0;
overflow: hidden;
}
⚙ Adding Functionality with Javascript
We should roughly note down what we want. When someone clicks on the accordion button, we would want to:
- show the panel below the accordion (with sliding down animation)
- change the "plus sign (+)" to something else, like a "cross sign (x)"
- auto close the previous panel if another button is clicked
Let's code that down, one by one:
Here, we can add a class panel-open
and remove the panel-closed
class on the panel div when someone clicks the accordion button and similarly do the opposite when the button is clicked again. We already have the closed class, let's write the open class:
.panel-open{
max-height: 700px;
}
Coding the class add and remove logic in javascript:
let accordion = document.getElementsByClassName("accordion");
const totalAccordions = accordion.length;
for (let i = 0; i < totalAccordions; i++) {
accordion[i].addEventListener("click", (e) => {
let panel = e.target.nextElementSibling;
if (panel.classList[1] === "panel-closed") {
panel.classList.add("panel-open");
panel.classList.remove("panel-closed");
} else {
panel.classList.add("panel-closed");
panel.classList.remove("panel-open");
}
});
}
This would toggle max-height
property based on the click event on the accordion button.
Now, to change the plus sign to a cross sign, we can add another class to the accordion button on active state and javascript would handle that on the same click event handler function:
/* active class for accordion, adds the "cross sign (x)" */
.accordion-active::after {
content: "\274C";
}
After adding the js code:
let accordion = document.getElementsByClassName("accordion");
const totalAccordions = accordion.length;
for (let i = 0; i < totalAccordions; i++) {
accordion[i].addEventListener("click", (e) => {
// toggle the class 'accordion-active' on accordion button
e.target.classList.toggle("accordion-active");
let panel = e.target.nextElementSibling;
if (panel.classList[1] === "panel-closed") {
panel.classList.add("panel-open");
panel.classList.remove("panel-closed");
} else {
panel.classList.add("panel-closed");
panel.classList.remove("panel-open");
}
});
}
A more Clean way of doing that would be to use toggle instead to add and remove:
let accordion = document.getElementsByClassName("accordion");
const totalAccordions = accordion.length;
for (let i = 0; i < totalAccordions; i++) {
accordion[i].addEventListener("click", (e) => {
// toggle the class 'accordion-active' on accordion button
e.target.classList.toggle("accordion-active");
let panel = e.target.nextElementSibling;
if (panel.classList[1] === "panel-closed") {
panel.classList.toggle("panel-open");
}
});
}
🌈 Adding smooth slide down animation:
To make the sliding of the panel look smooth, we can also add a transition
property that would look for the max-height
, all we have to do is to add that on the panel
class.
.panel{
transition: max-height .33s cubic-bezier(.5,0,.1,1);
}
Here's a nice visual comparison between the above cubic-bezier
curve and a linear transition.
This is how it looks like after adding the transition property:
We are almost done, but something is missing. Yep, the previous panel doesn't auto-close on opening a new panel.
Let's create a function for that, this function would take in the currently open panel as an argument and close all the other open panels (if any) on opening another accordion.
const closeAllExcept = (pan) => {
for (let i = 0; i < totalAccordions; i++) {
let panelToClose = accordion[i].nextElementSibling;
if(panelToClose !== pan){
accordion[i].classList.remove("accordion-active");
panelToClose.classList.remove("panel-open");
}
}
}
We now need to invoke this function. Below is the final javascript code, you would also notice that I had moved the accordion-active
class toggle code inside the if
block because now closeAllExcept
also has the control to toggle the class.
let accordion = document.getElementsByClassName("accordion");
const totalAccordions = accordion.length;
for (let i = 0; i < totalAccordions; i++) {
accordion[i].addEventListener("click", (e) => {
let panel = e.target.nextElementSibling;
if (panel.classList[1] === "panel-closed") {
e.target.classList.toggle("accordion-active");
panel.classList.toggle("panel-open");
closeAllExcept(panel);
}
});
}
const closeAllExcept = (pan) => {
for (let i = 0; i < totalAccordions; i++) {
let panelToClose = accordion[i].nextElementSibling;
if(panelToClose !== pan){
accordion[i].classList.remove("accordion-active");
panelToClose.classList.remove("panel-open");
}
}
}
Hooray, it's working perfectly now 😍
It was fun building this, isn't it? 😎 If you want to play and mess around, here's a link to my Codepen for this project. If you enjoyed reading this, let me know in the comments or reach out to me on Twitter @AyushCodes 🙌
Top comments (6)
Nice example but you have made things hard for yourself and forgotten about accessibility.
Why not start with the correct HTML elements to make life easy?
The <summary> and <details> elements are designed for this and work without JS reasonably well.
If you don't support Internet Explorer anymore they are fully supported (and they fall back gracefully anyway in IE)
You can then improve that functionality with JS.
As it stands your accordion is not accessible for people who use a screen reader as I mentioned.
You need to add
aria-expanded
when a section is open,aria-controls
to point to the section that the button is related to, you need to add loads of keyboard controls such as down arrow to jump to the next accordion header, Home to jump to the first header and about 4 other things!As you can see there is loads to think about, so let HTML take care of some of it for you so you only have to implement the arrow keys, home key etc.
Anyway, not meaning to bash your article in any way, it was clear and well written so have a ❤🦄!
Thanks for pointing that out. 🙏 I really forgot about accessibility because I might be too concerned about doing it the way netflix has done it. I will try to edit or might as well write another article to improve the accessibility.
Very well explained and I hope it will help to build website like netflix as the streaming industry is been growing at a tremendous speed and if its planned correctly you could make a good profit.
Thank you!
A great addition could be to handle the max height of the accordion element dynamically.
Yeah, that would be great.
Glad it helped you, Hiep 🙌, thanks for reading.