Série de bibliotecas lit — 1: Um mergulho profundo na biblioteca lit-html
O objetivo desta série de blogs é aprofundar a biblioteca Lit e obter uma compreensão abrangente de seus recursos exclusivos e como ela se diferencia de outras bibliotecas e estruturas do mercado.
A série é dividida em três partes, abordando tópicos como
1. Como funciona o lit-html?
2. Como o Lit Element realiza atualizações em lote no DOM?
3. Uma comparação e benchmarking do Lit com outras estruturas e bibliotecas, juntamente com seus possíveis casos de uso.
Neste blog em particular, vamos nos concentrar em explorar o lit-html e seu funcionamento interno. Mas antes de nos aprofundarmos no lit-html, vamos primeiro ter uma visão geral da própria biblioteca Lit.
O que é Lit?
O Lit é uma biblioteca leve de componentes da Web com um tamanho pequeno de aproximadamente 5 KB.
Ele foi projetado especificamente para o desenvolvimento eficiente e rápido de componentes de interface do usuário usando especificações de componentes da Web e aprimora ainda mais o desempenho ao introduzir atualizações em lote no DOM.
De fato, de acordo com um benchmark público realizado pelahttps://krausest.github.io/js-framework-benchmark/index.html, o Lit Element demonstrou um desempenho mais rápido em comparação com o React.
Para saber mais sobre o Lit Element, você pode visitar o site oficial emhttps://lit.dev/. No entanto, o foco desta série é entender como o Lit Element funciona e como ele é implementado.
O Lit Element é construído sobre o lit-html, que é uma biblioteca de modelos usada para criar modelos e a lógica de atualização do DOM em lote. O Lit Element estende o lit-html fornecendo funcionalidade adicional para atualizações em lote.
Criação do template lit-html:
O lit-html é uma biblioteca de modelo JavaScript que permite a criação de modelos usando literais marcados JavaScript e funções de marca, aproveitando recursos nativos de JavaScript e componentes da web.
Ao contrário do JSX ou de outras bibliotecas de modelos, o lit-html não requer uma etapa de construção para converter modelos ou sintaxe específicos da estrutura em modelos ou sintaxe compatíveis com o navegador, pois utiliza recursos nativos do JavaScript diretamente.
Vamos criar um modelo usando a biblioteca lit-html:
Neste exemplo, importamos duas funções da biblioteca lit-html:
1. html
, que é uma função de tag.
2. render
, que é usado para renderizar o modelo para o DOM.
Em seguida, definimos uma função chamada myTemplate
que recebe um argumento name
.
O modelo é definido usando a html
função tag, entre aspas (``) e contém um trecho de HTML <div>Hello, ${name}</div>
.
O ${name}
é um espaço reservado que pode ser substituído dinamicamente por um valor em tempo de execução com base no argumento passado para a função myTemplate
.
A html
função tag retorna um TemplateResult
objeto, que representa o template que pode ser renderizado no DOM.
Nas próximas seções, exploraremos a função de tag JavaScript e como a função de tag html é implementada na biblioteca lit-html.
Na linha final do código, estamos invocando o render
método, que renderiza o modelo.
O render
método aceita dois argumentos:
a) TemplateResult
objeto que representa o template a ser renderizado.
b) Referência ao elemento DOM onde o template será renderizado.
Função de marcação em JavaScript:
Observação: se você já estiver familiarizado com as funções de tag, pode pular esta seção.
A função de tag em JavaScript é um tipo especial de função que pode ser usado para processar literais de modelo, que são strings delimitadas por back ticks (``).
A função de tag é invocada imediatamente antes da avaliação do literal de modelo e recebe o literal de modelo e seus valores interpolados como argumentos separados.
Por exemplo, considere o seguinte código:
var name = "Whatfix";
function sayHello(literalString, ...values) {
console.log(literalString); //Output: ['Hello, ', '', raw: Array(2)]
console.log(values); //Output: ['Whatfix']
}
sayHello`Hello, ${name}`;
a. Uma matriz de partes estáticas do literal de modelo: literalString
.
No nosso exemplo, o valor do literalString
parâmetro é['Hello, ', '', raw: Array(2)]
b. Um ou mais argumentos correspondentes ao número de expressões no literal marcado: values
. Em nosso exemplo, estamos usando o operador spread ( ...
) para representá-lo como um array. Seu valor é ['Whatfix']
, que contém os valores das expressões usadas no literal de modelo.
Um mergulho profundo no método html:
Aqui está uma implementação da html
função de tag do lit-html
repositório —https://github.com/lit/lit. (Fiz algumas melhorias no código removendo partes desnecessárias para torná-lo mais compreensível).
const html =
<T extends ResultType>(type: T) =>
(strings: TemplateStringsArray, ...values: unknown[]): TemplateResult<T> => {
return {
['_$litType$']: type,
strings,
values,
};
};
Em seguida, forma um objeto, conhecido como TemplateResult
, com três parâmetros e o retorna.
Os três parâmetros de TemplateResult
incluem _$litType$, strings e values[].
a. O parâmetro _$litType$ contém o tipo.
b. O parâmetro strings contém uma matriz das partes estáticas dos literais.
c. O parâmetro values[] contém os valores das partes dinâmicas do literal.
Em nosso exemplo anterior (Modelo — A), myTemplate("whatfix”)
o método retornará:
{
_$litType$: 1,
strings: ["<div>Hello, ", "</div>", raw: Array(2)],
values: ["whatfix"]
}
Um mergulho profundo no método de renderização:
O método render em lit-html é responsável por renderizar o componente na interface do usuário.
Aceita três argumentos:
a. value — Um TemplateResult
que é gerado pela função de tag html, conforme discutido na seção anterior.
b. container — Uma referência ao elemento DOM onde o modelo será renderizado.
c. opções — Para efeito de simplificação, este argumento pode ser ignorado para melhor compreensão do processo.
Podemos entender visualmente o funcionamento do método render em nosso exemplo, utilizando um fluxograma.
Em seguida, vamos processar nosso componente de exemplo, Template A, usando o fluxograma mencionado anteriormente. Como já geramos um TemplateResult
objeto usando a função tag html, conforme discutido na seção anterior, podemos agora prosseguir com o processo de renderização.
O TemplateResult
objeto do nosso componente de exemplo, Template A, terá a seguinte estrutura:
A etapa subsequente envolve a geração do <template>
elemento a partir do TemplateResult
objeto.
Em nosso componente de exemplo, Template A, que inclui uma parte dinâmica representada por {name}
, estamos gerando um número aleatório de nove dígitos e o adicionamos como um nó de comentário no lugar de {name}
.
Como resultado, o <template>
elemento aparecerá da seguinte forma:
The subsequent step involves cloning the template element and updating the dynamic parts, either during the first-time rendering or subsequent updates. The cloning the template element happens only in initial render of the component.
The resulting structure will appear as follows:
Then using the second argument passed to the render method, we are adding the final component structure in the Document.
Conclusion:
In this blog, we’ve explored the internal workings of lit-html. In the next part of this series, we’ll delve deeper into the batching operations performed in the lit library with lit elements.