dev-resources.site
for different kinds of informations.
An Easy Way to Make an Auto Responsive Menu
In this article Iām gonna show you a little hack about how to automatically collapse headerās menu into a āburgerā and vice-versa.
So the idea is simpleāāāwhen the menuās items cannot fit into the header they should be hidden and the mobile menu toggle button (the burger) will be shown. And it should be done automatically, without css media queries. So your menu still looks good on any device regardless of number of items, their font-sizes etc.
This is how it looks like:
Iāve made a very basic markup, with just a logo and a menu.
Then added some CSS to make it look nicer.
So in the ādesktopā mode the header looks like this:
But when the viewport is too small to fit that menu, the menu items become āwrappedā:
We donāt need this! If thereās no space for themāāāthen itās time to hide the menu and show the menu toggle instead.
This is where a bit of JS gets handy. We need just to detect when the top position of any of the menu items will be different from the rest ones.
Hereās the trick:
/**
* Detect when elements become wrapped
*
* @param {NodeList} items - list of elements to check
* @returns {array} Array of items that were wrapped
*/
const detectWrap = (items) => {
let wrappedItems = [];
let prevItem = {};
let currItem = {};
for (let i = 0; i < items.length; i++) {
currItem = items[i].getBoundingClientRect();
if (prevItem) {
let prevItemTop = prevItem.top;
let currItemTop = currItem.top;
// if current's item top position is different from previous
// that means that the item is wrapped
if (prevItemTop < currItemTop) {
wrappedItems.push(items[i]);
}
}
prevItem = currItem;
}
return wrappedItems;
};
Then we can use that function to check if menu items were wrapped and if soāāāhide the menu and show a mobile toggle button.
const addWrapClasses = () => {
const menu = document.querySelector(".menu");
const menuItems = document.querySelectorAll(".menu > ul > li");
// remove ".wrapped" classes to detect which items was actually wrapped
menu.classList.remove("wrapped");
// only after that detect wrap items
let wrappedItems = detectWrap(menuItems); // get wrapped items
// if there are any elements that were wrapped - add a special class to menu
if (wrappedItems.length > 0) {
menu.classList.add("wrapped");
}
};
// execute function on page load
addWrapClasses();
// execute function on window resize
window.addEventListener("resize", addWrapClasses);
Now it looks fine:
Hereās the full demo of what we ahcieved:
Check another trick for responsive font-size with only CSS:
Dynamic font-size using only CSS3
If you find this article helpfulāāādonāt hesitate to like, subscribe and leave your thoughts in the comments š
Read more posts on my Medium blog
Thanks for reading!
Stay safe and peace to you!
Featured ones: