Typical use cases of flexbox

In this guide, we will take a look at some of the common use cases for flexbox — those places where flexbox makes more sense than another layout method.

Why choose flexbox?

Flexbox is generally the right CSS layout solution when you want to lay out a collection of items in one dimension or control the spacing between items. In this guide, we'll look at some of the typical use cases of Flexbox.

A common pattern for navigation is to have a list of items displayed as a horizontal bar. It is likely the most common of flexbox examples, and could be considered the ideal flexbox use case.

When we have a set of items that we want to display horizontally, we may well end up with additional space. We need to decide what to do with that space, and have a couple of options. We either display the space outside of the items — therefore spacing them out with white space between or around them — or we absorb the extra space inside the items and therefore need a method of allowing the items to grow and take up this space.

Space distributed outside the items

To distribute the space between or around the items, we use the alignment properties in flexbox, and the justify-content property. You can read more about this property in Aligning items in a flex container, which deals with aligning items on the main axis.

In this example, we display the items at their natural size and use justify-content: space-between to space the items equally. You can change how the space is distributed using the space-around or space-evenly values. You could also use start to put the space at the end of the items, end to place it before them, or center to center the navigation items.

Space distributed within the items

A different pattern for navigation would be to distribute the available space within the items themselves, rather than create space between them. The flex properties allow items to grow and shrink in proportion to one another as described in Controlling ratios of flex items along the main axis.

If you wanted to respect the size property of your navigation items but have the available space shared out equally among them, you might use flex: auto, which is the shorthand for flex: 1 1 auto — all items grow and shrink from a flex-basis of auto. This would mean that the longer item would have more space because it started from a larger size, even though the same amount of available space is assigned to it as the others.

In the live example below try changing flex: auto to flex: 1. This shorthand for flex: 1 1 0 causes all of the items to become the same width, as they are working from a flex-basis of 0, allowing all of the space to be distributed evenly.

Split navigation

Another way to align items on the main axis is to use auto margins. This enables the design pattern of a navigation bar where one group of items are aligned left and another group aligned right. Here we are using the auto margins technique described in Using auto margins for main axis alignment.

The items are aligned on the main axis with normal, which behaves as start, as this is the initial behavior of flexbox. The gap property creates gaps between items. And we are aligning the last item to the right by giving it a margin-left value of auto. You can move the class from one item to another to change where the split happens.

Center item

A long-standing joke among developers is that the hardest problem in web design is vertical centering. Vertically centering content is very straightforward with flexbox alignment properties, as the following live example shows.

You can play with the alignment, aligning the item to the start with start or end with end.

With CSS box alignment properties, you can vertically center an element inside another without flexbox. In the example above, try removing the flex properties from the box and adding align-content: center. Then add margin: auto to the element you want to horizontally center.

Whether you use flexbox or grid to lay out a list of card components, these layout methods only work on direct children of the flex or grid component. This means that if you have variable amounts of content, the card will stretch to the height of the grid area or flex container. Any content inside uses regular block layout, meaning that on a card with less content the footer will rise up to the bottom of the content rather than stick to the bottom of the card.

Two card components showing that the internals of the component do not stretch with the wrapper.

Flexbox solves this. We make the card a flex container, with flex-direction: column. We then set the content area to flex: 1, which is the shorthand for flex: 1 1 0 — the item can grow and shrink from a flex basis of 0. As this is the only item that can grow, it takes up all available space in the flex container and pushes the footer to the bottom. If you remove the flex property from the live example you will see the footer moves up to sit directly under the content.

Media objects

The media object — an image or other media element with some descriptive text side-by-side — is a common pattern in web design. Media objects should be able to be flipped — moving the image from one side to the other.

This pattern is used for comments and other places where images are placed next to their descriptions. We can use flexbox to allow the part of the media object containing the image to take its sizing information from the image with the content of the media object flexing to take up the remaining space.

In this example, the media object is aligned to flex-start and the .content is set to grow, with the grow factor set to 1. These properties are the same as those used for our column layout card pattern above.

Some things that you might want to try in this live example relate to the different ways you might want to constrain the media object in your design.

To prevent the image from growing too large, you should add a max-width to the image. As that side of the media object uses the initial values of flexbox, it can shrink, but not grow, and uses a flex-basis of auto. Any width or max-width applied to the image will become the flex-basis.

css
.image img {
  max-width: 100px;
}

You could also allow both sides to grow and shrink in proportion. If you set both sides to flex: 1, they will grow and shrink from a flex-basis of 0, so you will end up with two equal-sized columns. You could either take the content as a guide and set both to flex: auto, in which case they would grow and shrink from the size of the content or any size applied directly to the flex items such as a width on the image.

css
.media .content {
  flex: 1;
  padding: 10px;
}

.image {
  flex: 1;
}

You could also give each side different flex-grow factors, for example setting the side with the image to flex: 1 and the content side to flex: 3. This will mean they use a flex-basis of 0 but distribute that space at different rates according to the flex-grow factor you have assigned. The flex properties we use to do this are described in detail in the guide Controlling ratios of flex items along the main axis.

css
.media .content {
  flex: 3;
  padding: 10px;
}

.image {
  flex: 1;
}

Flipping the media object

To switch the display of the media object and have the image on the right and content on the left, we set the flex-direction property to row-reverse.

In this example, we added a flipped class next to the media class. Remove the class from the HTML to see how the display changes.

Form controls

Flexbox is particularly useful when it comes to styling form controls. Forms have several small elements that we typically want to align with each other. A common pattern is to have a <label> and <input> pair combined with a <button>, perhaps for a search form or a newsletter sign-up form where you want your visitor to enter their email address.

Flexbox makes this type of layout easy to achieve. The <label>, <input> and <button> are contained in a wrapper that is set to display: flex. The flex properties allow the <input> field to grow, while the button and label do not grow. The text input field will grow and shrink depending on the space available.

Patterns like this can make it much easier to create a library of form elements for your design, which easily accommodate additional elements being added. You are taking advantage of the flexibility of flexbox by mixing items that do not grow with those that do.

Conclusion

While exploring the above patterns, you have hopefully started to see how you can think through the best way to use flexbox to achieve the result that you want. Quite often you have more than one choice. Mix items that cannot stretch with those that can, use the content to inform the size, or allow flexbox to share out space in proportion. It's up to you.

Think about the best way to present the content that you have and then see how flexbox or other layout methods can help you achieve it.