Design de edifícios

Um vislumbre do processo e das ferramentas usadas para criar a experiência de Designcember com estilo de calendário de feriados.

Em dezembro, e com tantos calendários que as pessoas usam para fazer a contagem regressiva e comemorar, queremos destacar o conteúdo da Web da comunidade e da equipe do Chrome. Todos os dias, destacamos um conteúdo relacionado ao desenvolvimento de IU e design, totalizando 31 destaques, entre eles 26 novos sites de demonstração, ferramentas, anúncios, podcasts, vídeos, artigos e estudos de caso.

Confira a experiência completa em designcember.com.

O site da Designcember.

Visão geral

Nosso objetivo era oferecer uma experiência na Web acessível, moderna, caprichosa e responsiva com o menor número possível de bytes. Queríamos destacar novas APIs responsivas, como consultas de contêiner, e incluir um belo exemplo de modo escuro em um site focado em design e com muitos recursos. Para isso, compactamos arquivos, oferecemos vários formatos, usamos ferramentas de build otimizadas para a geração de sites estáticos, lançamos um novo polyfill e muito mais.

Começar com a fantasia

A ideia do site do calendário do Designcember era funcionar como um espaço para mostrar todo o trabalho que queríamos destacar durante o mês de dezembro, atuando como um site de demonstração. Decidimos criar um edifício de apartamentos responsivo que pudesse ficar mais alto e estreito ou mais curto e largo, com janelas que se reorganizavam dentro do frame. Cada janela representava um dia (e, portanto, um conteúdo). Trabalhamos com a ilustradora Alice Lee para dar vida à nossa visão.

Esboços do esqueleto da página do Designcember.

Alice foi inspiradora, compartilhando processos e esboços que eram interessantes mesmo nos primeiros conceitos. Enquanto ela trabalhava na arte, nós hackeamos a arquitetura. As primeiras discussões foram sobre o layout macro, o edifício e as janelas. Como as janelas se adaptariam a uma, duas ou três colunas à medida que mais espaço de visualização ficasse disponível? Até onde eles podem encolher ou esticar? Qual seria o tamanho máximo do edifício? Qual seria o deslocamento das janelas?

Confira uma prévia de um protótipo responsivo usando grid-auto-flow: dense, mostrando como as janelas podem ser colocadas automaticamente pelo algoritmo de grade. Percebemos rapidamente que, embora as grids de proporção funcionassem muito bem para mostrar artes, elas não ofereciam a oportunidade de deixar as janelas crescerem e encolherem em um espaço disponível não uniforme e mostrar o poder das consultas de contêiner.

Animação que mostra como esse wireframe responde a diferentes tamanhos de tela.
Confira esta demonstração no CodePen.

Quando a grade geral ficou relativamente estável e transmitiu um senso de direção para a capacidade de resposta do edifício e das janelas, pudemos nos concentrar em uma única janela. Algumas janelas se alongaram, encolheram, apertaram, cresceram e se recompuseram mais do que outras na grade.

Wireframes mostrando como as janelas aparecem em diferentes pontos de interrupção.

Cada janela precisaria processar uma certa quantidade de turbulência de redimensionamento. Confira abaixo um protótipo de uma janela que demonstra a resposta dela a turbulências, mostrando o quanto podemos esperar que cada janela interativa se ajuste.

Animação de janela com spritesheets

Algumas janelas têm animações para aumentar a interação com a experiência. As animações são desenhadas à mão, frame a frame, no Photoshop. Cada frame é exportado, transformado em uma spritesheet com este gerador de spritesheet e otimizado com o Squoosh. A animação CSS usa background-position-x e animation-timing-function, conforme mostrado no exemplo a seguir.

.una
  background: url("/day1/una_sprite.webp") 0% 0%;
  background-size: 400% auto;
}

.day:is(:hover, :focus-within) .una {
  animation: una-wave .5s steps(1) alternate infinite;
}

@keyframes una-wave {
  0%  { background-position-x: 0%; }
  25% { background-position-x: 300%; }
  50% { background-position-x: 200%; }
  75% { background-position-x: 100%; }
}

Animação mostrando a janela do primeiro dia.

Algumas animações, como a caixa de dinheiro do dia seis, foram animações CSS baseadas em etapas. Conseguimos esse efeito com uma técnica semelhante, usando steps(), com a diferença de que os keyframes eram posições de transformação do CSS em vez de posições de plano de fundo.

Mascaramento de CSS

Algumas janelas tinham formas únicas. Usamos máscaras e aspect-ratio para criar uma janela adaptável, com formato exclusivo e escalonável.

Para criar uma máscara, como esta para a janela 8, foram necessárias algumas habilidades clássicas do Photoshop, além de um pouco de conhecimento sobre como as máscaras na Web funcionam. Vamos analisar a janela do dia 8.

A janela do dia 8.

Para se tornar uma máscara, a forma interna do trevo de quatro folhas precisa ser isolada como uma forma e preenchida com a cor branca. O branco informa ao CSS qual conteúdo permanece e tudo fora do branco não permanece. No Photoshop, o interior da janela foi selecionado, com borda de 1 pixel (para remover problemas de aliasing), preenchido com branco e exportado com a mesma altura e largura do frame da janela. Dessa forma, o frame e a máscara podem ser sobrepostos diretamente, mostrando o conteúdo interno dentro do frame conforme o esperado.

Imagem da máscara de trevo

Depois de concluído, o conteúdo da janela pode ser modificado e sempre parecer permanecer dentro do frame personalizado. A imagem a seguir mostra a versão do modo escuro da janela, com um gradiente de plano de fundo diferente e um filtro CSS de brilho aplicado à luz.

Janela do oitavo dia no modo escuro.

O mascaramento também oferece suporte a janelas responsivas baseadas em consultas de contêineres. Na janela nove, há um personagem que está oculto atrás de uma máscara até que a janela fique mais estreita. Para garantir que o usuário não pudesse ajustar a imagem fora do enquadramento, Alice completou o personagem inteiro para nós. O personagem está mascarado na janela, mas as plantas não estão. Outro desafio foi sobrepor elementos mascarados com camadas sem máscara e garantir que todos fossem dimensionados juntos.

A imagem a seguir mostra como fica sem a máscara na janela e no personagem.

A imagem da janela nove sem a máscara.

Esmagar a arte

Para manter a fidelidade da ilustração e garantir que as telas de alta definição não tivessem uma experiência do usuário desfocada, Alice trabalhou com uma proporção de 3x pixels. O plano era usar o imgix e exibir imagens e formatos otimizados no servidor, mas descobrimos que os ajustes manuais com a ferramenta Squoosh poderiam economizar 50% ou mais.

Como usar o Squoosh para compactar imagens.

A ilustração tem desafios únicos para compactação, especialmente o estilo de borda áspera e o traço de pincel que Alice usou. Escolhemos comprimir cada imagem PNG exportada do Photoshop 3x para um png, webp e avif menores. Cada tipo de arquivo tem suas próprias capacidades de compactação especiais, e foram necessárias mais de 50 imagens compactadas para encontrar algumas configurações de otimização comuns.

A CLI do Squoosh se tornou crucial com mais de 200 imagens para otimizar. Fazer tudo isso manualmente levaria dias. Depois de definir as configurações de otimização comuns, as disponibilizamos como instruções de linha de comando e processamos em lote pastas inteiras de imagens PNG em suas contrapartes compactadas WebP e AVIF.

Confira um exemplo de comando squoosh da CLI AVIF usado:

npx @squoosh/cli --quant '{"enabled":true,"zx":0,"maxNumColors":256,"dither":1}' --avif '{"cqLevel":19,"cqAlphaLevel":17,"subsample":1,"tileColsLog2":0,"tileRowsLog2":0,"speed":6,"chromaDeltaQ":false,"sharpness":5,"denoiseLevel":0,"tune":0}' image-1.png image-2.png image-3.png

Com a arte otimizada verificada no repositório, podemos começar a carregá-la do HTML:

<picture>
  <source srcset="/day1/inner-frame.avif" type="image/avif">
  <source srcset="/day1/inner-frame.webp" type="image/webp">
  <img alt="" decoding="async" role="presentation" src="/day1/inner-frame.png">
</picture>

Era repetitivo escrever o código-fonte da imagem. Por isso, criamos um componente Astro para incorporar imagens com uma linha de código.

<Pic filename="day1/inner-frame" role="presentation" />

Usuários de leitores de tela e teclados

Grande parte da experiência do Designcember é feita por meio de janelas interativas e de arte. Para nós, era importante que um usuário de teclado pudesse usar o site e conferir as janelas e que os usuários de leitores de tela tivessem uma experiência narrada agradável.

Por exemplo, ao incorporar as imagens, usamos role="presentation" para marcar a imagem como de apresentação para leitores de tela. Achamos que uma experiência do usuário com 5 a 12 descrições de alt fragmentadas seria ruim. Então, marcamos as imagens como de apresentação e fornecemos uma narração geral da janela. A navegação pelas janelas em um leitor de tela tem uma sensação de narrativa agradável, o que esperamos que ajude a transmitir a fantasia e a diversão que o site quer compartilhar.

O vídeo a seguir mostra uma demonstração da experiência do teclado. As teclas Tab, Enter, barra de espaço e Escape são usadas para orquestrar o foco nos pop-ups e nas janelas.

A experiência do leitor de tela tem atributos ARIA especiais que tornam o conteúdo mais claro. Por exemplo, os links dos dias só dizem "um" ou "dois", mas com algumas ARIAs adicionadas, eles são anunciados como "Dia um" e "Dia dois". Além disso, todas as imagens são resumidas em um único rótulo para que cada janela tenha uma descrição.

Astro, gerador de sites estático e baseado em componentes

O Astro facilitou o trabalho em equipe no site. O modelo de componente era familiar para desenvolvedores do Angular e do React, e o sistema de estilo de nome de classe com escopo ajudou cada desenvolvedor a saber que o trabalho deles em uma janela não entraria em conflito com ninguém.

Dias como componentes

Cada dia era um componente que buscava o status de uma armazenagem de dados do tempo de build. Isso nos permite executar a lógica do modelo antes que o HTML chegue ao navegador. A lógica determinaria se o dia precisa mostrar a dica ou não, já que os dias inativos não têm pop-ups.

Os builds são executados a cada hora, e o repositório de dados do tempo de build desbloqueia um novo dia quando o servidor de build passa da meia-noite. Esses pequenos sistemas auto-atualizados e autossuficientes mantêm o site atualizado.

Estilos com escopo e propriedades abertas

O Astro escreve estilos de escopo no modelo de componente, o que facilita a distribuição da carga de trabalho entre vários membros da equipe e também torna o uso de Open Props divertido. Os estilos Open Props normalize.css foram úteis com o tema adaptável (claro e escuro) e ajudaram a organizar conteúdo como parágrafos e cabeçalhos.

Como adotantes iniciais do Astro, enfrentamos alguns problemas com o PostCSS. Por exemplo, não foi possível atualizar para a versão mais recente do Astro devido a muitos problemas de build. Mais tempo pode ser gasto aqui, otimizando os fluxos de trabalho de build e de desenvolvimento.

Contêineres flexíveis

Algumas janelas aumentam e diminuem, mantendo a proporção para preservar a arte. Usamos outras janelas para mostrar o poder da arquitetura baseada em componentes com consultas de contêiner. As consultas de contêiner significam que as janelas podem ter informações de estilo responsivo individuais e se ajustar com base nos próprios tamanhos. Algumas janelas passaram de estreitas para largas e precisaram ajustar o tamanho e a posição da mídia.

Uma demonstração de como as janelas mudam à medida que têm mais espaço.

À medida que mais espaço fica disponível para uma janela, podemos adaptar o tamanho ou os elementos filhos da janela para caber. Para atender às janelas adaptativas, as consultas de contêiner não seriam apenas divertidas de mostrar, mas também seriam necessárias e simplificariam drasticamente a orquestração de determinados layouts.

.day {
  container: inline-size;
}

.day > .pane {
  min-block-size: 250px;

  @container (min-width: 220px) {
    min-block-size: 300px;
  }

  @container (min-width: 260px) {
    min-block-size: 310px;
  }

  @container (min-width: 360px) {
    min-block-size: 450px;
  }
}

Essa abordagem é diferente de manter uma proporção. Ele oferece mais controle e mais oportunidades. Em um determinado tamanho, muitas crianças se movem para se adaptar a um novo layout.

As consultas de contêiner também nos permitiram oferecer suporte à contenção de direção de bloco (vertical), de modo que, à medida que uma janela aumentava de comprimento, podíamos ajustar os estilos para que se encaixassem adequadamente. Isso pode ser visto nas consultas baseadas em altura, que usamos de forma independente, e nas consultas baseadas em largura:

.person {
  place-self: flex-end;
  margin-block: 25% 50%;
  margin-inline-start: -15%;
  z-index: var(--layer-1);

  @container (max-height: 350px) and (max-width: 425px) {
    place-self: center flex-end;
    inline-size: 50%;
    inset-block-end: -15%;
    margin-block-start: -2%;
    margin-block-end: -25%;
    z-index: var(--layer-2);
  }
}

Também usamos consultas de contêiner para mostrar e ocultar detalhes, já que a arte ficava cada vez mais lotada em tamanhos menores e mais vazia em tamanhos maiores. A janela 9 é um ótimo exemplo de como isso funciona:

Suporte a vários navegadores

Para criar uma ótima experiência moderna entre navegadores, especialmente para APIs experimentais, como consultas de contêiner, precisamos de um ótimo polyfill. Enviamos um aviso à nossa equipe, e Surma liderou a criação de um novo polyfill de consulta de contêiner. O polyfill depende de ResizeObserver, MutationObserver e da função CSS :is(). Portanto, todos os navegadores modernos são compatíveis com o polyfill, especificamente o Chrome e o Edge a partir da versão 88, o Firefox a partir da versão 78 e o Safari a partir da versão 14. O uso do polyfill permite qualquer uma das seguintes sintaxes:

/* These are all equivalent */
@container (min-width: 200px) {
  /* ... */
}
@container (width >= 200px) {
  /* ... */
}
@container size(width >= 200px) {
  /* ... */
}

Modo escuro

As versões do modo claro e escuro do site da Designcember, lado a lado.

Um último toque essencial para o site do Designcember foi um belo tema escuro. Queríamos mostrar como você pode usar a arte para participar ativamente da criação de uma ótima experiência no modo escuro. Para isso, ajustamos os estilos de plano de fundo de cada janela de forma programática e usamos o máximo de CSS possível ao criar a arte da janela. A maioria dos planos de fundo eram gradientes CSS para facilitar o ajuste dos valores de cor. Em seguida, colocamos a arte por cima.

Outros ovos de Páscoa

Toques pessoais

Adicionamos alguns toques pessoais à página para dar mais personalidade ao site. O primeiro foi o elenco de personagens, inspirado na nossa equipe. Também incluímos um cursor de estilo antigo em dias inativos e brincamos com o estilo do ícone.

Estilos de cursor personalizados e opções de favicon

Toques funcionais

Um dos toques funcionais adicionais é a funcionalidade "Ir para hoje", com um pássaro sentado no topo do edifício. Clicar ou pressionar Enter no pássaro faz com que você vá para o dia atual do mês na página, para que você possa acessar rapidamente os lançamentos mais recentes.

O Designcember.com também tem uma folha de estilo de impressão especial, em que exibimos uma imagem específica que funciona melhor em papel de 21,5 x 28 cm para que você possa imprimir o calendário e comemorar o ano todo.

Impressão em tamanho de pôster do design do calendário.
Una segurando uma impressão grande do calendário.

No geral, foi necessário muito trabalho para criar uma experiência da Web moderna, divertida e inusitada para comemorar o desenvolvimento da interface durante todo o mês de dezembro. Esperamos que você tenha gostado!

Partes da agenda com anotações visuais e comentários