Baseado nesta minha publicação no Twitter

Eu queria adicionar um carrossel de imagens em um README.md de projeto no GitHub. Mas o componente de carrossel não é algo tão trivial de se fazer. Então, como contornar?

O carrossel

Normalmente carrossel se dá ao nome de um componente que tem uma sequência de imagens e fica girando entre elas, até voltar para o começo. Não há um componente nativo para ele em HTML. Então, como fazer?

Uma alternativa é usar JS para esse fim (a parte Liquid será processada para gerar o endereço definitivo dos assets):

<style>
    .hidden {
        display: none
    }
</style>
<script>
    function prepareCarousel(carouselRoot) {
        const children = carouselRoot.children
        let idx = 0
        setInterval(() => {
            const nextIdx = (idx + 1) % children.length
            children[idx].className = "hidden"
            children[nextIdx].className = ""
            idx = nextIdx
        }, 1000)
    }
    const tryAndRepeat = () => {
        const carousels = document.querySelectorAll(".carousel")
        if (carousels.length == 0) {
            setTimeout(tryAndRepeat, 200)
            return
        }
        for (const carousel of carousels) {
            prepareCarousel(carousel)
        }
    }
    tryAndRepeat()
</script>
<div class='carousel'>
    <img                src='{{ page.base-assets | append: "girassol-girl1.png" | relative_url }}' height='80' />
    <img class="hidden" src='{{ page.base-assets | append: "girassol-girl2.png" | relative_url }}' height='80' />
    <img class="hidden" src='{{ page.base-assets | append: "girassol-kisses.png" | relative_url }}' height='80' />
    <img class="hidden" src='{{ page.base-assets | append: "girassol-steampunk.png" | relative_url }}' height='80' />
    <img class="hidden" src='{{ page.base-assets | append: "girassol.png" | relative_url }}' height='80' />
</div>

Nessa implementação, estou usando um CSS para controlar visibilidade dos itens e mantendo um índice de qual foi a última coisa visível. Estou assumindo que tudo dentro do carrossel são elementos que precisam ficar girando. Também estou assumindo que só vou ter isso de classe nesses elementos (vai, pressuposições aceitáveis para um componente específico que não irá ver os raios de sol, que estará para sempre preso em uma única página).

A função que deixa tudo pronto é essa:

function prepareCarousel(carouselRoot) {
    const children = carouselRoot.children
    let idx = 0
    setInterval(() => {
        const nextIdx = (idx + 1) % children.length
        children[idx].className = "hidden"
        children[nextIdx].className = ""
        idx = nextIdx
    }, 1000)
}

A cada janela de tempo eu determino qual será o próximo elemento visível, então torno o elemento atual invisível e só depois torno o seguinte visível. Após terminar essa atribuição no atributo class, atualizamos o valor do idx atual com o próximo que foi calculado. Pequena nota para perceber que nextIdx já leva em consideraçãoa possibilidade de se chegar no final da lista e precisar loopar.

Note que eu poderia fazer assim, que estaria “certo”:

const nextIdx = idx + 1
children[idx % children.length].className = "hidden"
children[nextIdx % children.length].className = ""
idx = nextIdx

Só que isso pode levar a idx estourar, caindo no ponto em que a precisão do ponto flutuante impeça a atualização correta, pois em ponto flutuante a + 1 == a pode ser verdade. Iria demorar um tempo absurdamente lobgo? Iria, mas melhor garantir que isso nunca aconteça, nõa é mesmo?

Para detectar os elementos que eu queria como carrossel, tentei uma abordagem para executar apenas ao estar carregado, mas como não tenho acesso ao <body onload='...'> de maneira trivial no blog, resolvi não seguir por esse caminho.

Após falhar de diversas maneiras a captura do evento de carregamento da página, resolvi usar uma alterativa covarde:

  • se eu não detectei os elementos do carrossel, tenta de novo daqui um tempinho
  • se eu detectei os elementos do carrossel, para cada um deles chamar o prepareCarousel com esse elemento

Para fazer essa estratégia usei o tryAndRepeat. Ao ser invocado, ele mesmo detecta se está faltando algo, então ele se cadastra novamente para se invocar:

const tryAndRepeat = () => {
    const carousels = document.querySelectorAll(".carousel")
    if (carousels.length == 0) {
        setTimeout(tryAndRepeat, 200)
        return
    }
    for (const carousel of carousels) {
        prepareCarousel(carousel)
    }
}
tryAndRepeat()

Download dos PNGs

As imagens que estou usando foram geradas usando algumas ferramentas de IA generativa de imagens, como MidJourney. Eu as salvei dentro do repositório do Girassol (projeto que no momento da escrita deste artigo está intocado desde a concepção, 2 anos atrás). Para deixar este artigo auto-contido, melhor deixar tudo ao meu dispor dentro do repositório do blog do Computaria.

Então, para baixar as imagens, precisava iterar em cada imagem do repositório original e baixar. Tradicionalmente se faz isso com curl. Peguemos um exemplo de imagem:

Garota

A URL inteira dela é https://raw.githubusercontent.com/girassol-rb/girassol/refs/heads/main/assets/girassol-girl1.png. Vamos analisar por partes?

Primeiro, o domínio é um lugar para indicar onde estão guardadas as coisas raw do GitHub. É bem padrão isso, qualquer arquivo que você pedir o link para o objeto raw ele dará algo lá dentro.

No path da URL, temos uma parte significativa aqui: girassol-rb/girassol. Isso contém o identificador do repositório, no formato OWNER/REPO.

Após a indicação do repositório, temos refs/heads/main. Isso simplesmente indica qual a referência do repositório que estamos usando. No caso específico é um branch chamado main.

Finalmente, chegamos na última parte do path da URL: assets/girassol-girl1.png. Essa parte indica o caminho do arquivo a partir da raiz do repositório.

https://raw.githubusercontent.com/girassol-rb/girassol/refs/heads/main/assets/girassol-girl1.png
        \_______________________/\___________________/\______________/\________________________/
          domínio de coisas raw     identificador do    git-ref            path do arquivo
                                    repo, no formato    aqui é o branch
                                    OWNER/REPO          main

No meu caso específico, tudo que vou pegar vai ser dentro de assets, então bastaria ter a lista com os nomes dos arquivos que eu poderia iterar neles e passar pro curl. E é fácil obter essa lista:

  • girassol-girl1.png
  • girassol-girl2.png
  • girassol-kisses.png
  • girassol-steampunk.png
  • girassol.png

Vamos testar o download?

#!/bin/bash

BASE_URL=https://raw.githubusercontent.com/girassol-rb/girassol/refs/heads/main/assets/

for img in girassol-girl1.png girassol-girl2.png girassol-kisses.png girassol-steampunk.png girassol.png; do
        echo $img
        curl -o $img $BASE_URL/$img
done

Hmmm, o VSCode disse que não conseguiu abrir o arquivo, por que será? Abrindo com o VIM o arquivo…

<a href="/girassol-rb/girassol/refs/heads/main/assets/girassol-girl1.png">Moved Permanently</a>.

Oops? Ok, se eu adicionar o -L o curl seguirá redirects, quem sabe seja só isso? Vamos testar e…

Ok, era só isso mesmo. Agora, para onde ele estaria redirecionando? Vou pegar um único exemplo e pedir para ele me mostrar mais sobre as interações, com -i:

> curl -i https://raw.githubusercontent.com/girassol-rb/girassol/refs/heads/main/assets/girassol-girl1.png
HTTP/2 200 
cache-control: max-age=300
content-security-policy: default-src 'none'; style-src 'unsafe-inline'; sandbox
content-type: image/png
etag: "XXXX"
strict-transport-security: max-age=31536000
x-content-type-options: nosniff
x-frame-options: deny
x-xss-protection: 1; mode=block
x-github-request-id: XXXX
accept-ranges: bytes
date: XXXX
via: 1.1 varnish
x-served-by: XXXX
x-cache: HIT
x-cache-hits: 0
x-timer: XXXX
vary: Authorization,Accept-Encoding,Origin
access-control-allow-origin: *
cross-origin-resource-policy: cross-origin
x-fastly-request-id: XXXX
expires: XXXX
source-age: 293
content-length: 1650809

Warning: Binary output can mess up your terminal. Use "--output -" to tell 
Warning: curl to output it to your terminal anyway, or consider "--output 
Warning: <FILE>" to save to a file.

Hmmm, nada demais. O que será que tá acontecendo? Vou tentar reproduzir o resultado num comando. Vou usar a característica de criar variáveis de ambiente específicas de um comando em bash, colocando no começo da linha todas as variáveis que eu quero dar valor para o comando específico, e depois de criar essas variáveis executar o comando.

Como o comando que eu quero usar vai usar essas variáveis, então eu preciso delegar para o comando em questão fazer a expansão dessas variáveis em texto, caso contrário, caso eu tentasse expandir o valor da variável de ambiente na linha que estou declarando, como a expansão é feita antes da execução do comando, eu estaria com o valor errado desses valores. Para executar um comando que execute um comando, nada como o bash -c CMD, assim eu posso passar bash -c 'echo $LALALA' e a expansão $LALALA será feita pelo comando bash como ele começar a executar.

> BASE_URL=https://raw.githubusercontent.com/girassol-rb/girassol/refs/heads/main/assets/ img=girassol-girl1.png bash -c 'curl -i $BASE_URL/$img'
HTTP/2 301 
content-type: text/html; charset=utf-8
location: /girassol-rb/girassol/refs/heads/main/assets/girassol-girl1.png
x-github-request-id: XXXX
accept-ranges: bytes
date: XXXX
via: 1.1 varnish
x-served-by: XXXX
x-cache: HIT
x-cache-hits: 1
x-timer: XXXX
access-control-allow-origin: *
cross-origin-resource-policy: cross-origin
x-fastly-request-id: XXXX
expires: XXXX
source-age: 425
vary: Authorization,Accept-Encoding
content-length: 98

<a href="/girassol-rb/girassol/refs/heads/main/assets/girassol-girl1.png">Moved Permanently</a>.

Hmmm, aqui obtivemos o resultado que obtivemos ao usar o script que não seguia links. Agora, por que será que usando as variáveis obtive o resultado, mas usando a URL diretamente eu consegui baixar a imagem corretamente?

O uso de variáveis para fazer isso com certeza não é um fator pra isso, pois o bash vai fazer a expansão de variável antes de executar o comando. Então vamos focar na expansão de $BASE_URL/$img.

> BASE_URL=https://raw.githubusercontent.com/girassol-rb/girassol/refs/heads/main/assets/ img=girassol-girl1.png bash -c 'echo $BASE_URL/$img'   
https://raw.githubusercontent.com/girassol-rb/girassol/refs/heads/main/assets//girassol-girl1.png

Conseguiu ver o problema? Pois bem, como eu declarei a variável com um / no final, e usei o texto $BASE_URL/$img, isso significou que eu coloquei uma barra a mais. Portanto, estava gerando o nome do arquivo com duas barras ali, e o GitHub, ao perceber isso, redirecionou de /assets//girassol-girl1.png para /assets/girassol-girl1.png.

Para provar a hipótese, removi do script a barra no final de BASE_URL e o -L do curl.

E deu certo! O script para download se encontra aqui.

Limitações no GitHub

Bem, o GitHub limita as tags HTML que posso usar para escrever markdown. Especificamente ele vai impedir uso das tags <script> e <style>. Por exemplo, usar o seguinte markdown em comentários (que em tese passa pelo mesmo trecho de renderização de um README.md):

<style>
  .marm {
    background-color: pink !important;
  }
</style>
<script>
console.log("lala")
</script>

I like to <span class='marm'>MOVE IT</span>

Gerou isso daqui:

Não renderizou as tags acima com a semântica HTML, mas
como se fossem texto plano

Esse comportamento está mapeado na especificação do GitHub Flavored Markdown, mais especificamente na seção 6.11, Disallowed Raw HTML (extension).

Com isso, não consigo usar o componente de carrossel criado neste post.

Seria tão bom se eu pudesse usar alguma coisa que eu conseguisse enganar as tags bloqueadas do GitHub sobre markdown…

Surge uma esperança: SVG

Vendo esta resposta no Stack Overflow, o pessoal já usava essa estratégia de se usar SVG para gerar a imagem desejada e com isso exibir no GitHub.

Basicamente a ideia é fazer com um SVG oco, contendo uma tag <foreignObject> e dentro dela um carrossel feito em HTML/CSS puros. Então, com esse SVG em mãos, podemos mencionar ele como uma imagem dentro do repositório do GitHub.

Então, vamos fazer a animação em CSS?

Criando uma transição com CSS

Na própria resposta o autor já dá um caminho muito importate: animações e keyframes. Uma alternativa para usar imagem no CSS é definir a imagem como sendo um background-image, e usar uma <div> de tamanho fixo para caber a imagem.

Para a parte da animação, preciso definir os @keyframes. Como vai ser um loop em cima de diversas imagens, uma das maneiras de seguir é definir a porcentagem de “andamento” da animação, casa porcengagem com uma imagem de fundo diferente:

@keyframes carousel {
    0% {
        background-image: url("girassol.png");
    }
    25% {
        background-image: url("girassol-girl1.png");
    }
    50% {
        background-image: url("girassol-girl2.png");
    }
    75% {
        background-image: url("girassol-kisses.png");
    }
    100% {
        background-image: url("girassol-steampunk.png");
    }
}

E para o estilo que usa essa animação? Bem, eu quero que ela se repita indefinidamente, que tenha uma duração de 10s (para não parecer muito corrida a troca de imagens) e suave, igual para todas as imagens. Portanto a animação vai ser carousel 10s linear infinite.

Além disso, copiei alguns atributos CSS cujo conteúdo não entendi de verdade. background-repeat: no-repeat é para ter uma certeza, e background-image: url("girassol.png") é um placeholder, inclusive é a primeira imagem das animações no keyframe. background-size: contain foi uma novidade para mim; basicamente, ele garante que a imagem fique dentro dos limites do componente, sem que ela seja esticada ou cortada para caber lá. MDN falando sobre o assunto.

.girassol {
    margin: 0;
    height: 400px;
    aspect-ratio: 1 / 1;
    background-image: url("girassol.png");
    background-repeat: no-repeat;
    background-size: contain;
    animation: carousel 10s linear infinite;
}

Adaptando os links das URLs para poder usar <iframe> inline (leia sobre o atributo srcdoc) e com isso eu tenho esse HTML:

Ok, estou satisfeito com o resultado.

Embarcando HTML no SVG e o SVG no HTML

Vamos ver se funciona? Basicamente criei um SCG com view-box de 400x400, começando do ponto (0,0). Clica aqui para abrir ele.

Se eu fosse usar diretamente no markdown, fica assim (ajustando porque inline ele não está no mesmo diretório do que as imagens):

<svg fill="none" viewBox="0 0 400 400" width="400" height="400" xmlns="http://www.w3.org/2000/svg">
    <foreignObject width="100%" height="100%">
        <div xmlns="http://www.w3.org/1999/xhtml">
            <style>
                @keyframes carousel {
                    0% {
                        background-image: url("/blog/assets/github-carousel/girassol.png");
                    }
                    25% {
                        background-image: url("/blog/assets/github-carousel/girassol-girl1.png");
                    }
                    50% {
                        background-image: url("/blog/assets/github-carousel/girassol-girl2.png");
                    }
                    75% {
                        background-image: url("/blog/assets/github-carousel/girassol-kisses.png");
                    }
                    100% {
                        background-image: url("/blog/assets/github-carousel/girassol-steampunk.png");
                    }
                }

                .girassol {
                    margin: 0;
                    height: 400px;
                    aspect-ratio: 1 / 1;
                    background-image: url("/blog/assets/github-carousel/girassol.png");
                    background-repeat: no-repeat;
                    background-size: contain;
                    animation: carousel 10s linear infinite;
                }
            </style>
            <div class="girassol">
            </div>
        </div>
    </foreignObject>
</svg>

Mas a ideia não é embarcar um objeto SVG inline no markdown do github, até porque isso é coibido pelo GFM. Mas, mas será que tem alguma maneira de adicionar o SVG no documento? Bem, podemos usar um <object> para deixar a coisa embarcada (já que a tag <svg> não aceita URLs com recursos externos):

<object type="image/svg+xml" data="/blog/assets/github-carousel/girassol-orig.svg">
</object>

Pois bem, o GitHub de toda sorte impede o uso do <object>, mas pelo menos consegui mostrar uma alternativa de como colocar um SVG na página HTML citando o link, não um SVG inline dentro da tag <svg>.

Pois então, acho que chegou a hora da verdade, não é mesmo? Pois vejam!!

Um SVG com um carrossel de imagens

Ué… O que ocorreu? Será que o elemento tá no canto? Clica aqui embaixo para pintar de rosa o backgound:

Hmmm, então esse objeto está aí… mas não carregou?

Depurando com CSS

Em primeiro ponto, para saber se o objeto estava no canto, queria algo que ficasse óbvio e marcante a seleção. Tons de rosa bem impactantes foram a primeira coisa que me vieram a cabeça. Inicialmente usar pink como a cor parecia uma boa ideia, mas na verdade essa cor não chamava tanto a atenção. Procurei no MDN uma lista de cores nomeadas, e nelas escolhi o deeppink porque me pareceu mais chamativo.

No lugar de alterar o estilo do componente diretamente (como fiz no poost Editando SVG na mão pra pedir café), resolvi adotar uma outra abordagem: que o CSS escolha o componente HTML correto. Para isso, fiz com que o CSS olhasse para um atributo da tag e que só fosse ativo caso tivesse um valor específico. Para deixar mais amarradinho, prendi que esse destaque fosse feito não somente na presença do atributo, mas também em uma classe específica highlight-opt. Então com essa intenção, temos a primeira parte do seletor CSS:

.highlight-opt[data-highlight="true"] {
    background-color: deeppink;
}

Do jeito que está construído o seletor acima, ele só será ativado caso o elemento HTML tenha a classe .highlight-opt e também o atributo data-highlight com o valor true.

Mas agora tem um problema, como consigo adicionar a classe ao <img> da importação do SVG? O SVG eu importei através da diretiva de markdown ![alt-text](url), que não me permite trabalhar diretamente com o nó HTML. Inclusive estou usando markdown para poder abstrair o DOM a maior parte do tempo e tornar isso transparente para mim, então é uma feature desejável essa limitação.

Para contornar isso, no lugar de tentar manipular diretamente o <img>, coloquei a imagem dentro de um <span> para poder interagir. Poderia ter escolhido <span> ou <div> para esse exemplo, mas o <span> tem como premissa não quebrar o fluxo dos elementos, então seria mais um envelope em cima do que vem dentro dele, já a <div> delimita um local contíguo.

Ficou assim o posicionamento da imagem no blog:

<span id='carrossel-falho' class="highlight-opt">
![Um SVG com um carrossel de imagens]({{ page.base-assets | append: "girassol-orig.svg" | relative_url }})
</span>

Com isso o posicionamento não deve ser absurdamente afetado pelo <span> e eu consigo selecionar especificamente o elemento do <span>. Em cima disso eu pego o elemento pelo ID específico e posso adicionar/remover o atributo dele:

function toggleHilightCarrosselFalho() {
    const spanCarrosselFalho = document.getElementById("carrossel-falho")
    if (spanCarrosselFalho.hasAttribute("data-highlight")) {
        spanCarrosselFalho.removeAttribute("data-highlight")
    } else {
        spanCarrosselFalho.setAttribute("data-highlight", "true")
    }
}

E dentro do botão eu chamo a função toggleHilightCarrosselFalho() ao ser clicado:

<button onclick="toggleHilightCarrosselFalho()">Toggle destaque</button>

Mas isso não deixa pintado todo o comprimento da imagem, e sim a linha que o span supostamente envelopa. Para resolver essa questão, posso colocar no seletor do CSS para pegar as tags img filhas daquele seletor que foi definido inicialmente:

.highlight-opt[data-highlight="true"]>img {
    background-color: deeppink;
}

Esse > vai pegar o elemento img que seja filho imediato de um nó que satisfaça .highlight-opt[data-highlight="true"]. Se eu colocar um espaço ` ` no lugar do > (.highlight-opt[data-highlight="true"] img), a interpretação é “um elemento img que seja descendente de um nó que satisfaça .highlight-opt[data-highlight="true"]”.

Inclusive essa questão do espaçamento pode gerar um incômodo no backender que está escrevendo o CSS. Por exemplo, eu costumeiramente colocava espaços entre as partes que deveriam ser satisfeitas do seletor CSS. Por exemplo, colocar .highlight-opt [data-highlight="true"] no lugar de .highlight-opt[data-highlight="true"]. Só que as semânticas são distintas demais entre esses dois seletores, e minha cabeça de backender se confundia com isso. Ao botar tudo junto, estou dizendo “um nó que seja da classe highlight-opt e que tenha o atributo data-highlight com valor true”. Ao por o espaço, a interpretação é “um nó que tenha o atributo data-highlight com valor true que seja descendente de um nó com classe highlight-opt”.

O que aconteceu com o SVG na imagem?

Bem, nesse caso específico é que SVG ao ser importado como imagem precisa atender algumas restrições a mais. Importar como <object> é tranquilo e o SVG é feliz para trabalhar como quiser, mas como <img> não. Aqui na documentação da MDN sobre a tag SVG como imagem:

External resources (e.g. images, stylesheets) cannot be loaded

Em tradução livre:

Recursos externos (tais como imagens, folhas de estilo) não podem ser carregados

Evitando a carga de recursos externos

O problema é recurso exerno? Então não tem problema. No lugar de apontar para o recurso externo, posso usar o recurso para a contrução do meu objeto. Como fazer isso? Usando URLs data: para descrever o conteúdo das imagens de fundo.

Basicamente, onde antes se tinha background-image: url("girassol.png");, agora preciso embarcar esse girassol.png dentro do CSS. Então vamos ter um background-image: url("data:...")), só falta descobrir o miolo desse data.

A primeira parte é indicar o que é que está sendo carregado. Então é data:image/png. Mas só isso não é o suficiente, preciso carregar o resto da imagem. Mas preciso que seja enviado textualmente a imagem, não pode ser um binário. E advinha só qual o esquema de representação de dados binários para apenas caracteres imprimíveis? Isso mesmo, base64!

Dentro do data então preciso definir que tem mais coisa além do image/png. O jeito que eu fiz para alcançar isso foi data:image/png;base64,..., onde a elipse é a representação em base64 do blob da imagem.

Geração automática do SVG

Vamos fazer em bash? Primeiro, preciso deixar claro que esse programa vai inspecionar o seu ambiente para determinar quais são as imagens, sem haver possibilidade de passar para o programa as imagens via argumentos de CLI.

Vamos definir o formato geral do SVG primeiro?

<svg fill="none" viewBox="0 0 400 400" width="400" height="400" xmlns="http://www.w3.org/2000/svg">
	<foreignObject width="100%" height="100%">
		<div xmlns="http://www.w3.org/1999/xhtml">
			<style>
				@keyframes carousel {
					/* injetar os frames aqui */
				}
				.girassol {
					margin: 0;
					height: 400px;
                    aspect-ratio: 1 / 1;
					background-image: url(/* inserir imagem do girassol padrão */);
					background-repeat: no-repeat;
					background-size: contain;
					animation: carousel 10s linear infinite;
				}
			</style>
			<div class="girassol">
			</div>
		</div>
	</foreignObject>
</svg>

Estamos usando bash, então posso imprimir com heredoc:

cat <<EOG
<svg fill="none" viewBox="0 0 400 400" width="400" height="400" xmlns="http://www.w3.org/2000/svg">
	<foreignObject width="100%" height="100%">
		<div xmlns="http://www.w3.org/1999/xhtml">
			<style>
				@keyframes carousel {
					/* injetar os frames aqui */
				}
				.girassol {
					margin: 0;
					height: 400px;
                    aspect-ratio: 1 / 1;
					background-image: url(/* inserir imagem do girassol padrão */);
					background-repeat: no-repeat;
					background-size: contain;
					animation: carousel 10s linear infinite;
				}
			</style>
			<div class="girassol">
			</div>
		</div>
	</foreignObject>
</svg>
EOG

Ok, bacana. Vamos aproveitar que o Bash faz substituição textual no heredoc e já deixar marcado que precisam pegar sol:

cat <<EOG
<svg fill="none" viewBox="0 0 400 400" width="400" height="400" xmlns="http://www.w3.org/2000/svg">
	<foreignObject width="100%" height="100%">
		<div xmlns="http://www.w3.org/1999/xhtml">
			<style>
				@keyframes carousel {
					`create_frames`
				}
				.girassol {
					margin: 0;
					height: 400px;
                    aspect-ratio: 1 / 1;
					background-image: url(`create_base_64 girassol.png`);
					background-repeat: no-repeat;
					background-size: contain;
					animation: carousel 10s linear infinite;
				}
			</style>
			<div class="girassol">
			</div>
		</div>
	</foreignObject>
</svg>
EOG

Note que antes onde estava comentário foi substituído por shell expansions.

Ok, e como seria o create_base_64? Pois bem, já ouviu falar do comando base64?

Basicamente esse comando vai criar um binário e irá produzir o output em base 64. Mas esse comando tem um problema: ele insere uma quebra de linha no final. Para evitar isso, que iria ficar indesejado, eu passo pelo tr para remover o conteúdo:

create_base_64() {
    base64 "$1" | tr -d $"\r\n"
}

Aqui o tr -d indica que estou apagando o caracter, não trocando. O comando tr vem de “translate”, mas aqui a interpretação é mais literal: a tradução se dá ao nível de cacarter.

O -d indica ao tr que ele precisa de comportamento específico: não precisa traduzir, apenas deletar. Então chegamos na expansão shell de string: $"\r\n". Isso basicamente vai indicar ao shell para fazer a interpretação desses scape sequencies.

Agora adicionemos o cabeçalho necessário:

create_base_64() {
    echo -n "data:image/png;base64,"
    base64 "$1" | tr -d $"\r\n"
}

E pronto! Aqui o -n passado para o echo indica que ela não insira a quebra de linha no final, pois assim a representação em base 64 do banco estará na linha correta.

Ok, mas e como funciona o create_frames? Basicamente aqui eu vou pegar a quantidade de fotos e, em cima disso, ir calculando quantos porcento vai avançar. Basicamente para cada paso vou criar um frame passando um arquivo e o porcentual de progressão (com exceção da última imagem, que é 100%).

Vamos primeiro examinar o create_frame em si? E depois voltar para o create_frames? Basicamente aqui vou dar o step da animação. Só isso. O formato geral é:

$progression% {
    background-image: url("`create_base_64 "$image"`");
}

E só isso. O create_base_64 já foi discutido. O resto é apenas receber os argumentos e parsear corretamente:

create_frame() {
    local image="$1"
    local progression="$2"

    cat <<FRAME
$progression% {
    background-image: url("`create_base_64 "$image"`");
}
FRAME
}

Ok, agora só faltou os detalhes do create_frames. Logo no começo desta seção mencionei que o script iria fazer uma inspeção e em cima disso ele pegava os dados. Então, essa inspeção vai preencher a variável GIRASSOLES:

GIRASSOLES=( girassol*.png )

Como a variável está sendo criada com o valor dentor de parênteses, a ideia é transformar em vetor. Ser vetor ajuda bastante a trabalhar com índices e listagem de elementos de dentro das variáveis. O jeito que foi construída a expansão glob ( girassol*.png ) vai gerar um vetor com os arquivos que começam com girassol e terminam com .png. Isso implica que automaticamente o tamanho do vetor vai se adequar, basta que se coloque o nome da imagem do arquivo de imagem com esse formato.

Então, precisamos agora contar quantos passos iremos dar. Se temos 3 fotos, comecemos da foto 0, então damos um passo para a foto 1 no meio, e finalmente chegamos no segudo passo no 100% com a foto 2. A quantidade de passos vai ser igual a quantidade de imagens disponíveis menos uma, que vai ser usado no 0% (ou no 100%).

Mas como fazer isso em cima da variável GIRASSOLES? Bem, o Bash oferece uma expansão de variável específica de vetores: ${#GIRASSOLES[@]}. Aqui o ${...} é só para indicar que haverá uma expansão de variável. O # no começo da expansão é para fazer uma contagem. Só que se eu fizer apenas ${#GIRASSOLES} ele irá contar a quantidade de caracteres do primeiro elemento do vetor. Para contornar isso, colocamos o “índice” [@], ficando finalmente ${#GIRASSOLES[@]}.

Então a quantidade de steps vai ser a quantidade de fotos menos 1. Para poder fazer aritmética, eu posso declarar a variável de interesse como sendo inteira, e em cima disso fazer as contas:

create_frames() {
    local -i n_frames=${#GIRASSOLES[@]}-1
    # ...
}

Ok, agora vamos calcular o quanto avançamos a cada step? Basicamente 100 dividido pela quantidade de passos:

create_frames() {
    local -i n_frames=${#GIRASSOLES[@]}-1
    local -i step=100/$n_frames
    # ...
}

Ok, agora é só iterar, não é? Existem várias maneiras de iterar em cima de GIRASSOLES. Mas nesse caso específico eu quero ignorar o último elemnto, para ter uma lida diferente. Então usar o for-each não satisfaria:

for girassol in "${GIRASSOLES[@]}"; do
    # blablabla
done

Note que a expansão do vetor protegido com aspas e com “índice” @ fará com que esse vetor se expanda em cada um de seus itens individuais. Se eu não tivesse protegido com aspas, o IFS entraria em ação para separar os pedaços da string.

E a expansão protegida com aspas em cima do índice asterisco * teria um resultado diferente: seria gerado um único elemento que é a concatenação de todos os elementos: "${GIRASSOLES[@]}" é muito diferente de "${GIRASSOLES[*]}".

Para poder lidar especificamente com o valor final, posso iterar usando o “for c-style”:

for (( i=0; i < $n_frames; i++ )); do
    # corpo do laço...
done

A sintaxe para usar esse tipo de for é:

  • palavra-chave for
  • dois parênteses abrindo juntos
  • campos de inicialização, verificação e avanço separados pom ;
  • uma quebra de comando (por exemplo com ;) e a palavra-chave do
  • o corpo do laço
  • fechamento do laço com palavra-chave done

No caso, preciso calcular a progressão para enviar para o create_frame. De modo geral, ficaria assim:

create_frames() {
    local -i n_frames=${#GIRASSOLES[@]}-1
    local -i step=100/$n_frames
    local -i progression

    for (( i=0; i < $n_frames; i++ )); do
        progression=... # cálculo mágico...
        create_frame ${GIRASSOLES[$i]} $progression
    done
    create_frame ${GIRASSOLES[$n_frames]} 100
}

E, bem. Temos o tamanho do passo, e temos o índice. Podemos simplesmente dizer que a progressão do momento é simplesmente usar progression=$step*$i. Note que eu posso fazer a multiplicação porque o tipo de progression é inteiro e que step é inteiro:

create_frames() {
    local -i n_frames=${#GIRASSOLES[@]}-1
    local -i step=100/$n_frames
    local -i progression

    for (( i=0; i < $n_frames; i++ )); do
        progression=$step*$i
        create_frame ${GIRASSOLES[$i]} $progression
    done
    create_frame ${GIRASSOLES[$n_frames]} 100
}

Vamos testar como que fica?

O SVG passou a ter essa cara geral (o base 64 só está sendo exibido parcialmente, não pus ele todo aqui):

<svg fill="none" viewBox="0 0 400 400" width="400" height="400" xmlns="http://www.w3.org/2000/svg">
	<foreignObject width="100%" height="100%">
		<div xmlns="http://www.w3.org/1999/xhtml">
			<style>
				@keyframes carousel {
					0% {
    background-image: url("...");
}
25% {
    background-image: url("...");
}
50% {
    background-image: url("...");
}
75% {
    background-image: url("...");
}
100% {
    background-image: url("...");
}
				}
				.girassol {
					margin: 0;
					height: 400px;
                    aspect-ratio: 1 / 1;
					background-image: url("...");
                    background-repeat: no-repeat;
                    background-size: contain;
					animation: carousel 10s linear infinite;
				}
			</style>
			<div class="girassol">
			</div>
		</div>
	</foreignObject>
</svg>

Carrossel de fotos

Voi là! Deu certo! Se quiser observar no GitHub, só clicar neste link, que foi o proeto que disparou minha necessidade de colocar um carrossel de fotos no painel do GitHub.

Resumindo

Devido as restrições do GitHub, para fazer um carrossel só se for animação CSS. Mas GitHub não permite que você suba assim coisas de CSS. Para contornar, você precisa importar um SVG como uma imagem e dentro do SVG determinar via CSS a animação.

SVG como imagem é proibido de pegar recursos externos, então se tiver dependência de outras coisas precisa importar manualmente dentro do SVG para ser tudo local. Por exemplo, carregar imagens PNG internamente usando as URLs data:... com o conteúdo da imagem em base 64.