A grid layout with responsive squares
Solution 1
The padding-bottom
trick is the most used to accomplish that.
You can combine it with both Flexbox and CSS Grid, and since using percent for margin/padding gives inconsistent result for flex/grid items (on older browser versions, see edit note below), one can add an extra wrapper, or like here, using a pseudo, so the element with percent is not the flex/grid item.
Edit: Note, there's an update made to the specs., that now should give consistent result when used on flex/grid items. Be aware though, the issue still occurs on older versions.
Note, if you will add content to the content
element, it need to be position absolute to keep the square's aspect ratio.
Edit 2: In a comment I were asked how to have a centered text, so I added that in below snippet.
.square-container {
display: flex;
flex-wrap: wrap;
}
.square {
position: relative;
flex-basis: calc(33.333% - 10px);
margin: 5px;
border: 1px solid;
box-sizing: border-box;
}
.square::before {
content: '';
display: block;
padding-top: 100%;
}
.square .content {
position: absolute;
top: 0; left: 0;
height: 100%;
width: 100%;
display: flex; /* added for centered text */
justify-content: center; /* added for centered text */
align-items: center; /* added for centered text */
}
<div class="square-container">
<div class="square">
<div class="content">
<span>Some centered text</span>
</div>
</div>
<div class="square">
<div class="content spread">
</div>
</div>
<div class="square">
<div class="content column">
</div>
</div>
<div class="square">
<div class="content spread">
</div>
</div>
<div class="square">
<div class="content column">
</div>
</div>
</div>
CSS Grid version
.square-container {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(30%, 1fr));
grid-gap: 10px;
}
.square {
position: relative;
border: 1px solid;
box-sizing: border-box;
}
.square::before {
content: '';
display: block;
padding-top: 100%;
}
.square .content {
position: absolute;
top: 0; left: 0;
height: 100%;
width: 100%;
}
<div class="square-container">
<div class="square">
<div class="content">
</div>
</div>
<div class="square">
<div class="content spread">
</div>
</div>
<div class="square">
<div class="content column">
</div>
</div>
<div class="square">
<div class="content spread">
</div>
</div>
<div class="square">
<div class="content column">
</div>
</div>
</div>
Solution 2
Try using viewport percentage units.
jsFiddle
.square-container {
display: grid;
grid-template-columns: repeat(3, 30vw);
grid-template-rows: 30vw;
grid-gap: 2.5vw;
padding: 2.5vw;
background-color: gray;
}
.square {
background-color: lightgreen;
}
body {
margin: 0; /* remove default margins */
}
<div class="square-container">
<div class="square">
<div class="content"></div>
</div>
<div class="square">
<div class="content spread"></div>
</div>
<div class="square">
<div class="content column"></div>
</div>
</div>
From the spec:
5.1.2. Viewport-percentage lengths: the
vw
,vh
,vmin
,vmax
unitsThe viewport-percentage lengths are relative to the size of the initial containing block. When the height or width of the initial containing block is changed, they are scaled accordingly.
- vw unit - Equal to 1% of the width of the initial containing block.
- vh unit - Equal to 1% of the height of the initial containing block.
- vmin unit - Equal to the smaller of
vw
orvh
.- vmax unit - Equal to the larger of
vw
orvh
.
Solution 3
You can use the fact that padding is calculated based on the width and set padding-top: 100%
directly to the square
grid items (the grid items would be square now).
2019 update
Note that for flex items as well as grid items earlier this doesn't used to work - see the post linked in the comments to this answer:
Now that there is a consensus between browsers (newer versions) to have the same behaviour for
padding
for flex items and grid items, you can use this solution.
See demo below:
.square-container {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(30%, 1fr));
grid-gap: 10px;
}
.square {
background: cadetblue;
padding-top: 100%; /* padding trick directly on the grid item */
box-sizing: border-box;
position: relative;
}
.square .content { /* absolutely positioned */
position: absolute;
top: 0;
right:0;
left: 0;
bottom: 0;
}
<div class="square-container">
<div class="square">
<div class="content"> some content here</div>
</div>
<div class="square">
<div class="content"> some content here</div>
</div>
<div class="square">
<div class="content"> some content here</div>
</div>
<div class="square">
<div class="content"> some content here</div>
</div>
<div class="square">
<div class="content column">some content here and there is a lot of text here</div>
</div>
<div class="square">
<div class="content spread">text</div>
</div>
<div class="square">
<div class="content column">some text here</div>
</div>
</div>
![Stretch0](https://i.stack.imgur.com/MURuw.jpg?s=256&g=1)
Comments
-
Stretch0 almost 2 years
I am wanting to create a grid layout with responsive squares.
I feel like I should be able to do this with CSS Grid layout but having trouble setting the height of each square to be equal to the width.
Also having trouble setting a gutter between each square.
Would I be better off using flexbox?
Currently my HTML looks like this but will be dynamic so more squares may be added. And of course it needs to be responsive so will ideally use a media query to collapse it to one column.
<div class="square-container"> <div class="square"> <div class="content"> </div> </div> <div class="square"> <div class="content spread"> </div> </div> <div class="square"> <div class="content column"> </div> </div> </div>
Using css grid, this is as far as I got
.square-container{ display: grid; grid-template-columns: 30% 30% 30%; .square { } }
I was able to get a bit further with flexbox and able to use space-between to align squares with a nice gutter but was still struggling to get the height to match the width of each square.
I wasn't able to find any examples of this being done with either flexbox or grid but any examples would be appreciated as well.
Thanks
-
Stretch0 over 6 yearsUnfortunately
.square-container
is nested within a few containers so usingvw
causes the squares to overflow outside the.square-container
wrapper -
Michael Benjamin over 6 years"Authors should avoid using percentages in paddings or margins on grid items entirely, as they will get different behavior in different browsers." stackoverflow.com/q/42708323/3597276
-
Michael Benjamin over 6 yearsTry it on Firefox.
-
Michael Benjamin over 6 yearsSame problem with flexbox: stackoverflow.com/q/36783190/3597276
-
Michael Benjamin over 6 yearsPost a full demo. That may be a problem that can be overcome.
-
Stretch0 over 6 yearsThis seems to be working well across all browsers I've tested so far. Thank you
-
Mr. Pyramid about 6 yearsthis solution works like charm in any browser wow! and guess what I have implemented it with bootstrap and works like charmer!
-
aaaidan almost 6 yearsLGSon, could you add a little explanation of what the magic is and how it works?
-
Asons almost 6 years@aaaidan The main magic is the "padding". To keep an element a square, its height and width should be the same, and why e.g.
padding-top
works, is that when setting a top (or bottom ) padding using percent, it uses its own width to calculate the padding's value, which also will become its height, hence it becomes a square. The second part, when it comes to its content, it needs in this case to be absolute positioned, so it doesn't affect the square's content, or else the padding calculation would need to also take that into account (and for that, a script would be needed). -
kukkuz about 5 yearsas the solution is now valid, I have un-deleted the answer, thanks all :)
-
Stuart Casarotto about 5 yearsI really prefer this answer as it doesn't require the padding trick.
-
FluffyBeing over 4 yearswhat's the browser support like for this solution?
-
FluffyBeing over 4 yearshow can i vertical align the text in the flex item?
-
Etienne Coumont almost 4 yearsIf there is some vertical scrollbar, that solution dosn't work anymore, as 100vw becomes equal to the body width + the scrollbar width. Here is the same example as above with a vertical scrollbar that breaks the display : jsfiddle.net/0obt9cvs/3
-
Asons almost 4 yearsUpdated the answer with a vertical aligned text sample.
-
NickCoder almost 4 yearsThis is a perfect responsive square grid that worked very well.
-
low_rents almost 3 yearsyou are using square images - so this doesn't work with images which are not square-scaled;