Como criar animações CSS de alto desempenho

Este guia ensina como criar animações CSS de alto desempenho.

Consulte Por que algumas animações são lentas? para saber a teoria por trás dessas recomendações.

Todas as propriedades CSS recomendadas neste guia têm bom suporte entre navegadores.

transform

Compatibilidade com navegadores

  • Chrome: 36.
  • Edge: 12.
  • Firefox: 16.
  • Safari: 9.

Origem

opacity

Compatibilidade com navegadores

  • Chrome: 1.
  • Edge: 12.
  • Firefox: 1.
  • Safari: 2.

Origem

will-change

Compatibilidade com navegadores

  • Chrome: 36.
  • Edge: 79.
  • Firefox: 36.
  • Safari: 9.1.

Origem

Mover um elemento

Para mover um elemento, use os valores da palavra-chave translate ou rotation da propriedade transform.

Por exemplo, para deslizar um item para a visualização, use translate.

.animate {
  animation: slide-in 0.7s both;
}

@keyframes slide-in {
  0% {
    transform: translateY(-1000px);
  }
  100% {
    transform: translateY(0);
  }
}

Use rotate para girar elementos. O exemplo a seguir gira um elemento 360 graus.

.animate {
  animation: rotate 0.7s ease-in-out both;
}

@keyframes rotate {
  0% {
    transform: rotate(0);
  }
  100% {
    transform: rotate(360deg);
  }
}

Redimensionar um elemento

Para redimensionar um elemento, use o valor da palavra-chave scale da propriedade transform.

.animate {
  animation: scale 1.5s both;
}

@keyframes scale {
  50% {
    transform: scale(0.5);
  }
  100% {
    transform: scale(1);
  }
}

Mudar a visibilidade de um elemento

Para mostrar ou ocultar um elemento, use opacity.

.animate {
  animation: opacity 2.5s both;
}

@keyframes opacity {
  0% {
    opacity: 1;
  }
  50% {
    opacity: 0;
  }
  100% {
    opacity: 1;
  }
}

Evite propriedades que acionam layout ou pintura

Antes de usar qualquer propriedade CSS para animação (exceto transform e opacity), determine o impacto da propriedade no pipeline de renderização. Evite qualquer propriedade que acione o layout ou a pintura, a menos que seja absolutamente necessário.

Forçar a criação de camadas

Conforme explicado em Por que algumas animações estão lentas?, colocar elementos em uma nova camada permite que o navegador os repinte sem precisar repintar o restante do layout.

Os navegadores geralmente podem tomar boas decisões sobre quais itens devem ser colocados em uma nova camada, mas você pode forçar manualmente a criação de camadas com a propriedade will-change. Como o nome sugere, essa propriedade informa ao navegador que esse elemento vai ser alterado de alguma forma.

No CSS, é possível aplicar will-change a qualquer seletor:

body > .sidebar {
  will-change: transform;
}

No entanto, a especificação sugere que você só faça isso para elementos que estão sempre prestes a mudar. Por exemplo, isso pode ser verdade para uma barra lateral que o usuário pode deslizar para dentro e para fora. Para elementos que não mudam com frequência, recomendamos aplicar will-change usando JavaScript quando uma mudança provavelmente vai acontecer. Aguarde tempo suficiente para que o navegador faça as otimizações necessárias e remova a propriedade quando a alteração for interrompida.

Se você quiser forçar a criação de camadas em um navegador que não oferece suporte a will-change (provavelmente o Internet Explorer), defina transform: translateZ(0).

Depurar animações lentas ou com falhas

O Chrome DevTools e o Firefox DevTools têm muitas ferramentas para ajudar você a descobrir por que as animações estão lentas ou falhas.

Verificar se uma animação aciona o layout

Uma animação que move um elemento usando algo diferente de transform provavelmente será lenta. O exemplo a seguir compara uma animação que usa transform a uma que usa top e left.

O que não fazer
.box {
  position: absolute;
  top: 10px;
  left: 10px;
  animation: move 3s ease infinite;
}

@keyframes move {
  50% {
     top: calc(90vh - 160px);
     left: calc(90vw - 200px);
  }
}
O que fazer
.box {
  position: absolute;
  top: 10px;
  left: 10px;
  animation: move 3s ease infinite;
}

@keyframes move {
  50% {
     transform: translate(calc(90vw - 200px), calc(90vh - 160px));
  }
}

Você pode testar isso nos dois exemplos de Glitch a seguir e conferir a performance usando o DevTools.

Chrome DevTools

  1. Abra o painel Performance.
  2. Registre o desempenho de execução enquanto a animação estiver acontecendo.
  3. Confira a guia Resumo.

Se você encontrar um valor diferente de zero para Renderização na guia Resumo, isso pode significar que a animação está fazendo o navegador trabalhar no layout.

O painel "Resumo" mostra 37 ms para renderização e 79 ms para pintura.
O exemplo animation-with-top-left causa trabalho de renderização.
O painel "Resumo" mostra valores zero para renderização e pintura.
O exemplo de animação-com-transformação não causa trabalho de renderização.

DevTools do Firefox

No Firefox DevTools, a Cascata (link em inglês) pode ajudar a entender onde o navegador está passando tempo.

  1. Abra o painel Performance.
  2. Comece a gravar a performance enquanto a animação está acontecendo.
  3. Pare a gravação e inspecione a guia Cascata.

Se você encontrar entradas para Recalculate Style, isso significa que o navegador precisa retornar ao início da cascata de renderização para renderizar a animação.

Verificar se há frames descartados

  1. Abra a guia Renderização no Chrome DevTools.
  2. Marque a caixa de seleção FPS meter.
  3. Observe os valores enquanto a animação é executada.

Preste atenção no rótulo Frames na parte de cima da interface do medidor de QPS. Isso mostra valores como 50% 1 (938 m) dropped of 1878. Uma animação de alto desempenho tem uma porcentagem alta, como 99%, o que significa que poucos frames estão sendo descartados e a animação parece suave.

O medidor de QPS mostra que 50% dos frames foram descartados
O exemplo de animation-with-top-left faz com que 50% dos frames sejam descartados
O medidor de QPS mostra que apenas 1% dos frames foram descartados
O exemplo animation-with-transform faz com que apenas 1% dos frames sejam descartados.

Verificar se uma animação aciona a pintura

Algumas propriedades são mais caras para o navegador pintar do que outras. Por exemplo, tudo que envolve um desfoque (como uma sombra) leva mais tempo para pintar do que desenhar uma caixa vermelha. Essas diferenças nem sempre são óbvias no CSS, mas as ferramentas do navegador podem ajudar a identificar quais áreas precisam ser pintadas novamente, além de outros problemas de desempenho relacionados à pintura.

Chrome DevTools

  1. Abra a guia Renderização no Chrome DevTools.
  2. Selecione Pintura.
  3. Mova o ponteiro pela tela.
Um elemento da interface destacado em verde para demonstrar que ele será repintado
Neste exemplo do Google Maps, é possível ver os elementos sendo repintados.

Se a tela inteira estiver piscando ou áreas destacadas que você acha que não deveriam mudar, investigue mais.

Se você precisar determinar se uma propriedade específica está causando problemas de desempenho relacionados à pintura, o Paint Profiler no Chrome DevTools pode ajudar.

Firefox DevTools

  1. Abra Configurações e adicione um botão da caixa de ferramentas para Alternar pintura intermitente.
  2. Na página que você quer inspecionar, ative o botão e mova o mouse ou role a tela para ver as áreas destacadas.

Conclusão

Sempre que possível, restrinja as animações a opacity e transform para manter as animações no estágio de composição do caminho de renderização. Use o DevTools para verificar qual etapa do caminho está sendo afetada pelas animações.

Use o gerador de perfis de pintura para saber se alguma operação de pintura é particularmente cara. Se você encontrar algo, verifique se outra propriedade CSS oferece a mesma aparência com um desempenho melhor.

Use a propriedade will-change com moderação e somente se você encontrar um problema de desempenho.