A grid layout with responsive squares

35,279

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.

Fiddle demo - Flexbox

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 units

The 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 or vh.
  • vmax unit - Equal to the larger of vw or vh.

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>
Share:
35,279
Stretch0
Author by

Stretch0

Php and Javascript developer

Updated on July 09, 2022

Comments

  • Stretch0
    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
    Stretch0 over 6 years
    Unfortunately .square-container is nested within a few containers so using vw causes the squares to overflow outside the .square-container wrapper
  • Michael Benjamin
    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
    Michael Benjamin over 6 years
    Try it on Firefox.
  • Michael Benjamin
    Michael Benjamin over 6 years
    Same problem with flexbox: stackoverflow.com/q/36783190/3597276
  • Michael Benjamin
    Michael Benjamin over 6 years
    Post a full demo. That may be a problem that can be overcome.
  • Stretch0
    Stretch0 over 6 years
    This seems to be working well across all browsers I've tested so far. Thank you
  • Mr. Pyramid
    Mr. Pyramid about 6 years
    this solution works like charm in any browser wow! and guess what I have implemented it with bootstrap and works like charmer!
  • aaaidan
    aaaidan almost 6 years
    LGSon, could you add a little explanation of what the magic is and how it works?
  • Asons
    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
    kukkuz about 5 years
    as the solution is now valid, I have un-deleted the answer, thanks all :)
  • Stuart Casarotto
    Stuart Casarotto about 5 years
    I really prefer this answer as it doesn't require the padding trick.
  • FluffyBeing
    FluffyBeing over 4 years
    what's the browser support like for this solution?
  • FluffyBeing
    FluffyBeing over 4 years
    how can i vertical align the text in the flex item?
  • Etienne Coumont
    Etienne Coumont almost 4 years
    If 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
    Asons almost 4 years
    Updated the answer with a vertical aligned text sample.
  • NickCoder
    NickCoder almost 4 years
    This is a perfect responsive square grid that worked very well.
  • low_rents
    low_rents almost 3 years
    you are using square images - so this doesn't work with images which are not square-scaled;