Flexbox

Um padrão de design que pode ser complicado em design responsivo é uma barra lateral que fica alinhada com algum conteúdo. Onde há espaço de janela de visualização, esse padrão funciona muito bem, mas onde o espaço é condensado, esse layout rígido pode se tornar problemático.

O modelo de layout de caixa flexível (flexbox) é um modelo de layout projetado para conteúdo unidimensional. Ele se destaca em pegar vários itens de tamanhos diferentes e retornar o melhor layout para esses itens.

Este é o modelo de layout ideal para este padrão de barra lateral. O Flexbox não apenas ajuda a colocar a barra lateral e o conteúdo em linha, mas onde não há espaço suficiente restante, a barra lateral se quebra em uma nova linha. Em vez de definir dimensões rígidas para o navegador seguir, com o flexbox, você pode fornecer limites flexíveis para sugerir como o conteúdo pode ser exibido.

O que você pode fazer com um layout flexível?

Os layouts flexíveis possuem os seguintes recursos, que você poderá explorar neste guia.

  • Eles podem ser exibidos como uma linha ou coluna.
  • Eles respeitam o modo de escrita do documento.
  • Eles são uma linha por padrão, mas podem ser solicitados a quebrar em várias linhas.
  • Os itens no layout podem ser reordenados visualmente, longe de sua ordem no DOM.
  • O espaço pode ser distribuído dentro dos itens, de forma que eles fiquem maiores e menores de acordo com o espaço disponível em seus pais.
  • O espaço pode ser distribuído ao redor dos itens e linhas flexíveis em um layout encapsulado, usando as propriedades do Alinhamento de caixa.
  • Os próprios itens podem ser alinhados no eixo transversal.

O eixo principal e o eixo transversal

A chave para entender o flexbox é entender o conceito de eixo principal e eixo transversal. O eixo principal é aquele definido por sua propriedade flex-direction Se for row seu eixo principal está ao longo da linha, se for column seu eixo principal está ao longo da coluna.

Três caixas próximas umas das outras com uma seta apontando da esquerda para a direita. A seta é rotulada como Eixo principal

Os itens flexíveis se movem como um grupo no eixo principal. Lembre-se: temos um monte de coisas e estamos tentando obter o melhor layout para eles como um grupo.

O eixo cruzado corre na direção oposta ao eixo principal, portanto, se flex-direction for row o eixo cruzado corre ao longo da coluna.

Três caixas de alturas diferentes, uma ao lado da outra com uma seta apontando da esquerda para a direita. A seta está rotulada como Eixo principal. inferior. Este é rotulado como eixo cruzado

Você pode fazer duas coisas no eixo transversal. Você pode mover os itens individualmente ou como um grupo para que eles se alinhem uns com os outros e com o contêiner flexível. Além disso, se você encapsulou as linhas flexíveis, pode tratar essas linhas como um grupo para controlar como o espaço é atribuído a essas linhas. Você verá como tudo isso funciona na prática ao longo deste guia, por enquanto, apenas tenha em mente que o eixo principal segue sua flex-direction.

Criação de um contêiner flexível

Vamos ver como o flexbox se comporta pegando um grupo de itens de diferentes tamanhos e usando o flexbox para distribuí-los.

<div class="container" id="container">
  <div>One</div>
  <div>Item two</div>
  <div>The item we will refer to as three</div>
</div>

Para usar o flexbox, você precisa declarar que deseja usar um contexto de formatação flexível e não bloco regular e layout embutido. Faça isso alterando o valor da propriedade display flex .

.container {
  display: flex;
}

Como você aprendeu no guia de layout, isso lhe dará uma caixa em nível de bloco, com filhos de itens flexíveis. Os itens flex imediatamente começam a exibir algum comportamento do flexbox, usando seus valores iniciais.

Os valores iniciais significam que:

  • Os itens são exibidos como uma linha.
  • Eles não se enrolam.
  • Eles não crescem para encher o recipiente.
  • Eles se alinham no início do contêiner.

Controlando a direção dos itens

Mesmo que você não tenha adicionado uma flex-direction ainda, os itens são exibidos como uma linha porque o valor inicial de flex-direction é row. Se você quiser uma linha, não precisa adicionar a propriedade. Para alterar a direção, adicione a propriedade e um dos quatro valores:

  • row : os itens são dispostos como uma fileira.
  • row-reverse: os itens são dispostos como uma linha a partir do final do contêiner flexível.
  • column : os itens são dispostos como uma coluna.
  • column-reverse : os itens são dispostos como uma coluna a partir do final do contêiner flexível.

Você pode experimentar todos os valores usando nosso grupo de itens na demonstração abaixo.

Reverter o fluxo de itens e acessibilidade

Você deve ter cuidado ao usar quaisquer propriedades que reordenem a exibição visual de como as coisas estão ordenadas no documento HTML, pois isso pode afetar negativamente a acessibilidade. Os column-reverse row-reverse coluna são um bom exemplo disso. O reordenamento só acontece para a ordem visual, não para a ordem lógica. É importante entender isso porque a ordem lógica é a ordem em que um leitor de tela lerá o conteúdo e qualquer pessoa que navegue usando o teclado a seguirá.

Você pode ver no vídeo a seguir como, em um layout de coluna invertido, a tabulação entre os links se desconecta, pois a navegação do teclado segue o DOM e não a exibição visual.

Qualquer coisa que possa alterar a ordem dos itens no flexbox ou na grade pode causar esse problema. Portanto, qualquer reordenamento deve incluir testes completos para verificar se isso não tornará o seu site difícil de usar para algumas pessoas.

Para mais informações, veja:

Modos e direção de escrita

Os itens flexíveis são dispostos como uma linha por padrão. Uma linha segue na direção em que as frases fluem em seu modo de escrita e na direção do script. Isso significa que se você estiver trabalhando em árabe, que tem uma direção de script da direita para a esquerda (rtl), os itens serão alinhados à direita. A ordem das tabulações também começa à direita, pois é assim que as frases são lidas em árabe.

Se você estiver trabalhando com um modo de escrita vertical, como algumas fontes japonesas, uma linha será exibida verticalmente, de cima para baixo. Tente mudar a flex-direction neste demo que está usando um modo de escrita vertical.

Portanto, a maneira como os flex items se comportam por padrão está vinculada ao modo de escrita do documento. A maioria dos tutoriais é escrita em inglês ou outro modo de escrita horizontal, da esquerda para a direita. Isso tornaria mais fácil presumir que os itens flexíveis se alinham à esquerda e rodam horizontalmente.

Com o eixo principal e cruzado mais o modo de escrita a ser considerado, o fato de falarmos sobre início e fim em vez de topo, base, esquerda e direita no flexbox pode ser mais fácil de entender. Cada eixo tem um início e um fim. O início do eixo principal é conhecido como início principal. Portanto, nossos itens flexíveis inicialmente se alinham do main-start. O fim desse eixo é o fim principal. O início do eixo cruzado é um início cruzado e o final um final cruzado.

Um diagrama rotulado dos termos acima

Envolvendo itens flexíveis

O valor inicial da propriedade flex-wrap nowrap. Isso significa que, se não houver espaço suficiente no contêiner, os itens transbordarão.

Um contêiner flexível com nove itens dentro, os itens diminuíram de modo que uma palavra está em uma linha, mas não há espaço suficiente para mostrá-los lado a lado para que os itens flexíveis tenham se estendido para fora da caixa do contêiner.
Assim que atingirem o tamanho mínimo do conteúdo, os itens flexíveis começarão a estourar seu contêiner

Os itens exibidos com os valores iniciais serão reduzidos o máximo possível, até o min-content antes que ocorra o estouro.

Para fazer com que os itens embrulhem, adicione flex-wrap: wrap ao contêiner flexível.

.container {
  display: flex;
  flex-wrap: wrap;
}

Quando um contêiner flexível é enrolado, ele cria várias linhas flexíveis. Em termos de distribuição de espaço, cada linha atua como um novo contêiner flexível. Portanto, se você estiver agrupando linhas, não é possível fazer com que algo na linha 2 se alinhe com algo acima dela na linha 1. Isso é o que significa que o flexbox é unidimensional. Você pode controlar o alinhamento em um eixo, uma linha ou uma coluna, não os dois juntos como podemos fazer na grade.

A abreviação do flex-flow

Você pode definir as propriedades flex-direction e flex-wrap usando a abreviação flex-flow. Por exemplo, para definir flex-direction para column e permitir que os itens sejam agrupados:

.container {
  display: flex;
  flex-flow: column wrap;
}

Controlando o espaço dentro dos itens flexíveis

Supondo que nosso contêiner tenha mais espaço do que o necessário para exibir os itens, os itens se alinham no início e não crescem para preencher o espaço. Eles param de crescer em seu tamanho máximo de conteúdo. Isso ocorre porque o valor inicial das flex- é:

  • flex-grow: 0 : os itens não crescem.
  • flex-shrink: 1 : os itens podem encolher até menos do que sua flex-basis.
  • flex-basis: auto : os itens têm um tamanho base de auto.

Isso pode ser representado por um valor de palavra-chave flex:initial. A flex shorthand, ou as longhands de flex-grow, flex-shrink e flex-basis são aplicadas aos filhos do contêiner flexível.

Para fazer com que os itens cresçam, enquanto permite que itens grandes tenham mais espaço do que os pequenos, use flex:auto. Você pode tentar fazer isso usando a demonstração acima. Isso define as propriedades para:

  • flex-grow: 1 : os itens podem crescer mais do que sua flex-basis.
  • flex-shrink: 1 : os itens podem encolher até menos do que sua flex-basis.
  • flex-basis: auto : os itens têm um tamanho base de auto.

Usar flex: auto significará que os itens terão tamanhos diferentes, já que o espaço que é compartilhado entre os itens é compartilhado depois que cada item é disposto como tamanho máximo do conteúdo. Portanto, um item grande ganhará mais espaço. Para forçar todos os itens a terem um tamanho consistente e ignorar o tamanho do conteúdo, altere flex:auto para flex: 1 na demonstração.

Isso descompacta para:

  • flex-grow: 1 : os itens podem crescer mais do que sua flex-basis.
  • flex-shrink: 1 : os itens podem encolher até menos do que sua flex-basis.
  • flex-basis: 0 : os itens têm um tamanho base de 0.

Usando flex: 1 diz que todos os itens têm tamanho zero, portanto, todo o espaço no contêiner flexível está disponível para ser distribuído. Como todos os itens têm um fator de flex-grow 1, todos eles crescem igualmente e o espaço é compartilhado igualmente.

Permitindo que os itens cresçam em taxas diferentes

Você não precisa dar a todos os itens um fator de flex-grow 1. Você pode dar aos seus itens flexíveis diferentes fatores de flex-grow. Na demonstração abaixo, o primeiro item tem flex: 1, o segundo flex: 2 e o terceiro flex: 3. Conforme esses itens crescem de 0 o espaço disponível no contêiner flexível é dividido em seis. Uma parte é dada ao primeiro item, duas partes ao segundo, três partes ao terceiro.

Você pode fazer a mesma coisa em uma flex-basis de auto, embora seja necessário especificar os três valores. O primeiro valor sendo flex-grow, o segundo flex-shrink e o terceiro flex-basis.

.item1 {
  flex: 1 1 auto;
}

.item2 {
  flex: 2 1 auto;
}

Este é um caso de uso menos comum, pois a razão para usar uma flex-basis do auto é permitir que o navegador descubra a distribuição do espaço. Se você quiser que um item cresça um pouco mais do que o algoritmo decide, pode ser útil.

Reordenando itens flexíveis

Os itens em seu contêiner flexível podem ser reordenados usando a propriedade order Esta propriedade permite a ordenação de itens em grupos ordinais. Os itens são dispostos na direção ditada pela flex-direction, os valores mais baixos primeiro. Se mais de um item tiver o mesmo valor, ele será exibido com os outros itens com esse valor.

O exemplo abaixo demonstra essa ordem.

Check your understanding

Test your knowledge of flexbox

The default flex-direction is

column
row

By default, a flex container wraps children.

true
false

A flex child item appears squished, which flex property helps mitigate this?

flex-basis
flex-grow
flex-shrink

Visão geral do alinhamento Flexbox

O Flexbox trouxe consigo um conjunto de propriedades para alinhar itens e distribuir espaço entre os itens. Essas propriedades foram tão úteis que, desde então, foram movidas para suas próprias especificações, você também as encontrará no Layout de grade. Aqui você pode descobrir como eles funcionam quando você está usando o flexbox.

O conjunto de propriedades pode ser colocado em dois grupos. Propriedades para distribuição espacial e propriedades para alinhamento. As propriedades que distribuem o espaço são:

  • justify-content: distribuição do espaço no eixo principal.
  • align-content: distribuição de espaço no eixo transversal.
  • place-content: uma abreviação para definir as duas propriedades acima.

As propriedades usadas para alinhamento no flexbox:

  • align-self: alinha um único item no eixo cruzado
  • align-items: alinha todos os itens como um grupo no eixo cruzado

Se você estiver trabalhando no eixo principal, as propriedades começam com justify-. No eixo transversal, eles começam com align-.

Distribuindo espaço no eixo principal

Com o HTML usado anteriormente, os itens flexíveis dispostos como uma linha, há espaço no eixo principal. Os itens não são grandes o suficiente para encher completamente o recipiente flexível. Os itens se alinham no início do contêiner flexível porque o valor inicial de justify-content é flex-start. Os itens se alinham no início e qualquer espaço extra fica no final.

Adicione a justify-content ao contêiner flexível, atribua a ele um valor flex-end e os itens se alinham no final do container e o espaço livre é colocado no início.

.container {
  display: flex;
  justify-content: flex-end;
}

Você também pode distribuir o espaço entre os itens com justify-content: space-between.

Experimente alguns dos valores da demonstração e consulte o MDN para obter o conjunto completo de valores possíveis.

Com flex-direction: column

Se você alterou sua flex-direction para column, justify-content funcionará na coluna. Para ter espaço livre em seu contêiner ao trabalhar como uma coluna, você precisa dar a ele uma height ou block-size. Caso contrário, você não terá espaço livre para distribuir.

Experimente os diferentes valores, desta vez com um layout de coluna flexbox.

Distribuindo espaço entre as linhas flexíveis

Com um contêiner flexível encapsulado, você pode ter espaço para distribuir no eixo transversal. Nesse caso, você pode usar a align-content com os mesmos valores que justify-content. Ao contrário de justify-content que alinha itens para flex-start por padrão, o valor inicial de align-content é stretch. Adicione a propriedade align-content ao contêiner flexível para alterar esse comportamento padrão.

.container {
  align-content: center;
}

Experimente na demonstração. O exemplo tem linhas quebradas de itens flexíveis e o contêiner tem um block-size para que tenhamos algum espaço livre.

A abreviatura de place-content

Para definir justify-content e align-content você pode usar place-content com um ou dois valores. Um único valor será usado para ambos os eixos, se você especificar que o primeiro é usado para align-content e o segundo para justify-content.

.container {
  place-content: space-between;
  /* sets both to space-between */
}

.container {
  place-content: center flex-end;
  /* wrapped lines on the cross axis are centered,
  on the main axis items are aligned to the end of the flex container */
}

Alinhando itens no eixo cruzado

No eixo cruzado, você também pode alinhar seus itens dentro da linha flexível usando align-items e align-self. O espaço disponível para este alinhamento dependerá da altura do contêiner flexível, ou linha flexível no caso de um conjunto de itens envolvidos.

O valor inicial de align-self é stretch, que é o motivo pelo qual os itens flexíveis em uma linha se estendem até a altura do item mais alto por padrão. Para mudar isso, adicione a align-self a qualquer um dos seus itens flex.

.container {
  display: flex;
}

.item1 {
  align-self: flex-start;
}

Use qualquer um dos seguintes valores para alinhar o item:

  • flex-start
  • flex-end
  • center
  • stretch
  • baseline

Veja a lista completa de valores no MDN.

A próxima demonstração tem uma única linha de flex items com flex-direction: row. O último item define a altura do contêiner flexível. O primeiro item possui a align-self com um valor de flex-start. Tente alterar o valor dessa propriedade para ver como ela se move dentro do seu espaço no eixo cruzado.

A align-self é aplicada a itens individuais. A align-items pode ser aplicada ao contêiner flexível para definir todas as propriedades individuais do align-self como um grupo.

.container {
  display: flex;
  align-items: flex-start;
}

Na próxima demonstração, tente alterar o valor de align-items para alinhar todos os itens no eixo cruzado como um grupo.

Por que não há justify-self no flexbox?

Os itens flexíveis atuam como um grupo no eixo principal. Portanto, não existe o conceito de separar um item individual desse grupo.

No layout de grade, as propriedades justify-self e justify-items funcionam no eixo em linha para fazer o alinhamento dos itens naquele eixo dentro de sua área de grade. Devido à maneira como os layouts flexíveis tratam os itens como um grupo, essas propriedades não são implementadas em um contexto flexível.

Vale a pena saber que o flexbox funciona muito bem com margens automáticas. Se você encontrar a necessidade de separar um item de um grupo, ou separar o grupo em dois grupos, pode aplicar uma margem para fazer isso. No exemplo abaixo, o último item tem uma margem esquerda auto. A margem automática absorve todo o espaço na direção em que é aplicada. Isso significa que ele empurra o item para a direita, dividindo os grupos.

Como centralizar um item vertical e horizontalmente

As propriedades de alinhamento podem ser usadas para centralizar um item dentro de outra caixa. A justify-content alinha o item no eixo principal, que é a linha. A align-items no eixo cruzado.

.container {
  width: 400px;
  height: 300px;
  display: flex;
  justify-content: center;
  align-items: center;
}

Check your understanding

Test your knowledge of flexbox

.container {
  display: flex;
  direction: ltr;
}

To vertically align with flexbox, use

justify keywords
align keywords
.container {
  display: flex;
  direction: ltr;
}

To horizonally align with flexbox, use

align keywords
justify keywords
.container {
  display: flex;
  direction: ltr;
}

By default, flex items are aligned to stretch. If you want content size used for child items, which of the following styles would you use?

height: auto
align-items: flex-start
justify-content: flex-start
align-content: start

Recursos