I have created a mega menu where the values will be dynamic (coming from the API). So you never know how many links will be there. But for testing purpose I’ve put 50 texts in array menu
and trying to loop over that so that it look Like this . So there’s a parent row and inside that there are 5 columns, each containing 5 links. Per row will contain 5 columns or 25 links.
JSX
const menu = [ "one", "two", "three", "four", "five", "one", "two", "three", "four", "five", ... (total 50) ]; <Box className={classes.rows}> <Box className={classes.cols}> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> </Box> <Box className={classes.cols}> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> </Box> <Box className={classes.cols}> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> </Box> <Box className={classes.cols}> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> </Box> <Box className={classes.cols}> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> <Typography variant="body1" className={classes.link}> Link 1 </Typography> </Box> </Box>
I tried using for loop and then create another 2D array. And later planned to map through that array inside JSX.
let links = []; if (menu.length != 0) { for (let i = 0; i < menu.length; i++) { for (let j = 0; j < 5; j++) { links.push(menu[i]); } } }
But it won’t work I know. Can anyone help me with this?
Advertisement
Answer
First you have to divide your links up into groups of 5. Each group of 5 links will be put in a box with a defined width of 20% (or just under to allow for padding/margins, this will give 5 columns per page).
Then use flexbox with flexwrap: ‘wrap’ to handle your dynamic links and map your group boxes into that.
Here is an example:
https://codesandbox.io/s/youthful-chaum-83m4vl?file=/src/App.js
import { Box, Typography } from "@mui/material"; import "./styles.css"; const links = [...]; export default function App() { // how many groups of 5 given the length of the links array const numberOfGroups = Math.floor(links.length / 5) + (links.length % 5 > 0 ? 1 : 0); const linkGroups = []; for (let index = 0; index < numberOfGroups; index++) { // push each group of 5 into the linkGroups array linkGroups.push(links.slice(index * 5, (index + 1) * 5)); } return ( <div className="App"> <Box sx={{ display: "flex", width: "100vw", flexWrap: "wrap", backgroundColor: "lightgrey" }} > {linkGroups.map((linkGroup) => ( <Box sx={{ width: "18%", mb: "16px", borderRight: "1px solid white" }} > {linkGroup.map((link) => { return <Typography>{link}</Typography>; })} </Box> ))} </Box> </div> ); }