DEV Community

Timevolt
Timevolt

Posted on

CSS Grid vs Flexbox: When to Reach for Which

CSS Grid vs Flexbox: When to Reach for Which

Quick context (why you're writing this)

I was building a dashboard last month and kept switching between Grid and Flexbox like I was trying to decide which coffee to order. One minute I’d swear Flexbox was the answer for a card layout, the next minute I’d wrestle with wrapping behavior and end up pulling my hair out. After a few wasted hours I realized I wasn’t picking the right tool for the job—I was just using whatever felt familiar. If you’ve ever felt that tug‑of‑war, you’re not alone.

The Insight

The short answer: Flexbox shines when you’re laying out items along a single axis (think rows or columns that need to stretch or shrink together). Grid shines when you need two‑dimensional control—both rows and columns at the same time.

It’s not a religion; it’s about the problem you’re solving. Flexbox gives you powerful distribution and alignment with minimal markup. Grid gives you a true canvas where you can place items anywhere, define explicit tracks, and even overlap elements if you want. Knowing which dimension you actually need to control saves you from writing weird hacks just to get things to line up.

How (with code)

A classic Flexbox win: a responsive navigation bar

<nav class="navbar">
  <a href="#" class="logo">Brand</a>
  <ul class="nav-links">
    <li><a href="#">Home</a></li>
    <li><a href="#">Features</a></li>
    <li><a href="#">Pricing</a></li>
    <li><a href="#">Contact</a></li>
  </ul>
</nav>
Enter fullscreen mode Exit fullscreen mode
.navbar {
  display: flex;
  align-items: center;          /* vertical centering */
  justify-content: space-between;
  padding: 1rem;
  background: #333;
  color: #fff;
}

.nav-links {
  display: flex;
  gap: 1.5rem;                  /* space between items */
}

.nav-links a {
  color: inherit;
  text-decoration: none;
}

/* Stack on small screens */
@media (max-width: 600px) {
  .navbar {
    flex-direction: column;
  }
  .nav-links {
    flex-direction: column;
    align-items: flex-start;
  }
}
Enter fullscreen mode Exit fullscreen mode

What makes Flexbox the right call here?

We only care about distributing items along the main axis (horizontal) and maybe wrapping them on narrow screens. The vertical centering is a simple align-items. No need to define rows and columns explicitly—Flexbox does the heavy lifting with a couple of lines.

Where Flexbox trips you up: a two‑column card grid

A common mistake is trying to force a responsive card grid with Flexbox alone:

<section class="card-grid">
  <article class="card"></article>
  <article class="card"></article>
  <article class="card"></article>
  <!-- more cards -->
</section>
Enter fullscreen mode Exit fullscreen mode
.card-grid {
  display: flex;
  flex-wrap: wrap;
  gap: 1rem;
}

.card {
  flex: 1 1 200px;   /* grow, shrink, basis */
}
Enter fullscreen mode Exit fullscreen mode

At first glance this looks fine—cards wrap and keep a minimum width. But notice what happens when the number of cards isn’t a multiple of the row count: the last row gets uneven spacing, and you end up with “orphan” cards that stretch oddly or leave awkward gaps. You start adding margin hacks, :last-child tricks, or even JavaScript to fix it. That’s a sign you’re using the wrong tool.

The Grid fix for the same UI

.card-grid {
  display: grid;
  gap: 1rem;
  /* auto‑fit as many 200px columns as will fit, then share remaining space */
  grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
}
Enter fullscreen mode Exit fullscreen mode
<section class="card-grid">
  <!-- same cards as before -->
</section>
Enter fullscreen mode Exit fullscreen mode

Now the browser calculates the exact number of columns that fit, gives each column an equal share of any extra space, and the last row always aligns perfectly—no orphan stretching, no extra markup. You get true two‑dimensional control with a single declaration.

A Grid‑only scenario: a complex dashboard layout

Imagine a layout with a header, a sidebar, a main content area, and a footer that spans the full width. With Flexbox you’d need nested containers and careful ordering; with Grid it’s straightforward:

<div class="dashboard">
  <header>Header</header>
  <nav>Sidebar</nav>
  <main>Main</main>
  <footer>Footer</footer>
</div>
Enter fullscreen mode Exit fullscreen mode
.dashboard {
  display: grid;
  min-height: 100vh;
  grid-template-rows: auto 1fr auto;   /* header, main+sidebar, footer */
  grid-template-columns: 200px 1fr;    /* sidebar, main */
  grid-template-areas:
    "header header"
    "sidebar main"
    "footer footer";
}

header   { grid-area: header; }
nav      { grid-area: sidebar; }
main     { grid-area: main; }
footer   { grid-area: footer; }
Enter fullscreen mode Exit fullscreen mode

The grid-template-areas naming makes the layout self‑documenting. Change a single line and the whole structure shifts—no need to juggle flex directions or wrap properties.

Why This Matters

Choosing the right layout tool isn’t just about saving a few lines of CSS; it’s about predictability. When you reach for Flexbox for a one‑dimensional problem, you get behavior that’s intuitive and easy to reason about. When you need true rows and columns, Grid gives you a declarative way to define the whole space up front, reducing the chance of brittle hacks later on.

I’ve seen teams waste days debugging alignment issues that vanished the moment they swapped a Flexbox container for a Grid one (or vice‑versa). The mental model shift is small, but the payoff is huge: less CSS, fewer media queries, and layouts that adapt gracefully as content changes.

So next time you start a new component, ask yourself: Do I need to control layout along a single axis, or do I need to define both rows and columns? Answer that, and you’ll reach for the right tool without second‑guessing.

Your turn

Try taking a component you’ve built recently with Flexbox and see if Grid simplifies it—or the other way around. Drop a CodePen or a GitHub link in the comments and let us know what you discovered. I’m curious to see where the line falls for you. Happy styling!

Top comments (0)