View (visões)
Uma View é uma base primitiva usada pelo angular para rederizar uma árvore DOM. Uma ViewContainer(Conteiner de Visões) é um local na View o qual aceita Views filhas. Cada ViewContainer tem ViewContainerRef(Referência a um ViewContainer) associado que pode conter um infinidade de Views filhas. Views formam uma estrutura de uma árvore que imita um ávore DOM.
- View é o construtor principal de renderização. Uma aplicação em execução é apenas uma coleção de View que são aninhadas em uma estrutura que lembram árvores. Uma árvore de Views é uma versão simplificada de uma árvore DOM. Uma View pode ter um único Elemento DOM ou grandes estruturas DOM. A chave é que a árvore DOM em na View pode não sofrer mudanças estruturais (apenas mudanças de propriedades).
- Views representam uma instância sendo executada de uma View DOM. Isto implica que enquanto elementos em uma View pode mudar propriedades, eles não podem mudar estruturalmente, (Mudanças estruturais como, adição ou remoção de elementos, requer remoção de View filhas dos ViewContainers).
- Views podem ter zero ou mais ViewContainers. Uma ViewContainer é um marcador no DOM que permite a inserção de Views filhas.
- Views são criados de uma ProtoView (Protítipo de Visão). Uma ProtoView é compilado para uma View DOM o qual é bastante eficiente em criar Views.
- Views contém um objeto de contexto. Um contexto representa a instância de objeto contra o qual todas as expressõe são avaliadas.
- View contém um ChangeDetector (Detector de mudanças) para busca por mudanças detectadas no modelo.
- Views contém ElementInjector(Injetor de Elementos) para criação de Diretivas.
View Simples
Vamos examinar uma View simples e todas suas partes em detalhes.
Assuma o seguinte Componente:
class Greeter {
greeting:string;
constructor() {
this.greeting = 'Olá';
}
}
E assuma a seguinte View HTML:
<div>
Seu nome:
<input var="name" type="Text">
<br>
{{greeting}} {{name.value}}!
</div>
O template acima é compilado pelo Compilador para criar uma ProtoView. A ProtoView é então usada para criar instâncias da View. O processo de instanciação envolve a clonagem do template acima e localizar todos os elementos que contenham vínculos e finalmente é instanciado as Diretivas associadas ao template. (Veja a seção de Compilação para mais detalhes.)
<div> | viewA(greeter)
Your name: | viewA(greeter)
<input var="name" type="Text"> | viewA(greeter): variável local 'name'
<br> | viewA(greeter)
{{greeting}} {{name.value}}! | viewA(greeter): expressão de vínculo 'greeting' & 'name.value'
</div> | viewA(greeter)
A instância resultante da View é algo semelhante a isso (pseudocódigo simplificado):
viewA = new View({
template: ...,
context: new Greeter(),
localVars: ['name'],
watchExp: ['greeting', 'name.value']
});
Nota:
- Views usam instâncias de
Greeter
como contexto de avaliação. - View reconhece variáveis locais como
name
. - View sabe quais expressões precisam ser observadas.
- Views sabem o que é preciso para ser atualizado caso uma expressão observada mude.
- Todos os elementos DOM são pertencentes por uma única instância da View.
- A estrutura da DOM pode não mudar durante o tempo de execução. Para permitir mudanças estruturais no DOM necessitamos entender Views Compostas.
View Compostas
Uma importante parte da aplicação é permitir mudanças na estrutura do DOM para renderizar os dados para o usuário. Em Angular isto é feito pela inserção de View filhas em um ViewContainer.
Vamos começas com uma View como a seguinte:
<ul>
<li template="ng-for: #person of people">{{person}}</li>
</ul>
Durante o processo de compilação o Compilador quebra o template HTML em estas duas ProtoViews:
<li>{{person}}</li> | protoViewB(Locals)
e
<ul> | protoViewA(someContext)
<template></template> | protoViewA(someContext): protoViewB
</ul> | protoViewA(someContext)
O próximo passo é compor estas PropoViews em um View de fato oqual é rederizada para o usuário.
Passo 1: Instancia-se viewA
<ul> | viewA(someContext)
<template></template> | viewA(someContext): new NgFor(new ViewContainer(protoViewB))
</ul> | viewA(someContext)
Passo 2: Intancia-se a diretiva NgFor
o qual irá receber a ViewContainerRef
. (A ViewContainerRef tem uma referência a protoViewA
).
Passo 3: Como a diretiva irá executar irá pergunta ao ViewContainerRef
para instanciar protoViewB
e irá inserí-lo logo após a âncora do ViewContainer
. Isto é repetitivo a cada person
in people
.
Note que
<ul> | viewA(someContext)
<template></template> | viewA(someContext): new NgFor(new ViewContainer(protoViewB))
<li>{{person}}</li> | viewB0(locals0(someContext))
<li>{{person}}</li> | viewB1(locals0(someContext))
</ul> | viewA(someContext)
Passo 4: Todos os vínculkos nas Views filhas são atualizadas. Note que no caso de NgFor
a avaliação contextual para a viewB0
e viewB1
são locals0
e locals1
respectivamente. Locals permite a introdição de variáveis locais visíveis apenas dentro do escopo da View, e delega cada referência desconhecida para o contexto pai.
<ul> | viewA
<template></template> | viewA: new NgFor(new ViewContainer(protoViewB))
<li>Alice</li> | viewB0
<li>Bob</li> | viewB1
</ul> | viewA
Cada View pode ter entre zero ou mais ViewContainers. Pela inserção ou remoção de View filhas para e de um ViewContainers, a aplicação pode mudar a estrutura DOM em qualquer estado desejado. Uma View pode conter nós individuais ou complexas estruturas DOM. A inserção de pontos para View filhas, conhecido como ViewContainers, contém um elemento DOM o qual age como uma âncora. A âncora pode ser tanto um template
quanto um elemento script dependendo do seu browser; Isto é usado para identificar onde as Views filhas irá ser inseridas.
Componente de Views
Uma View pode ter também conter Componentes. Componentes contém Shadow DOM para encapsulamento de seu estado interno de renderização. Ao contrário aos ViewContainers que podem ter zero ou mais Views, o Componente sempre terá exatamente um único Shadow View.
<div> | viewA
<my-component> | viewA
#SHADOW_ROOT | (fronteira de encapsulamento)
<div> | viewB
renderização encapsulada | viewB
</div> | viewB
</my-component> | viewA
</div> | viewA
Contexto de Avaliação
Cada View age como um contexto para avaliação de suas expressões. Há dois tipos de contextos:
- Uma instância de um componente Controller e
- um contexto
Locals
para introdução de variáveis locais na View.
Vamos assumir o seguinte componente:
class Greeter {
greeting:string;
constructor() {
this.greeting = 'Olá';
}
}
E assumir a seguinte View HTML:
<div> | viewA(greeter)
Your name: | viewA(greeter)
<input var="name" type="Text"> | viewA(greeter)
<br> | viewA(greeter)
{{greeting}} {{name.value}}! | viewA(greeter)
</div> | viewA(greeter)
A Interface do Usuário é construída usando uma única View, e consequentemente um único contexto para gretter
. Isto pode ser expressado em pseudo-código:
var greeter = new Greeter();
A View contém dois vínculos:
gretting
: limita a propriedadegreeting
na instânciaGreeter
.name.value
: isto põe um problema. Não há propriedadename
na instânciaGreeter
. Para resolver nós envolver a instância Greeter em uma instânciaLocal
como abaixo:
var greeter = new Locals(new Greeter(), {name: referência_ao_elemento_de_entrada })
Por envolvermos a instância de Greeter
em Locals
permitimos que a View introduza variáveis as quais são adicionadas a instância de Greeter
. Durante a resolução das expressões primeiramente checamos o Locals, e somente então a instância de Greeter
.
Ciclo de Vida da View (Hidratação e Desidratração)
Transição de View através de um particular conjunto de estados:
- Views podem ser criadas de um ProtoView.
- Views podem ser anexadas em uma ViewContainerRef.
- Sobre a anexação da View para uma ViewContainerRef, a View precisa ser hidratada. A hidratação envolve a instanciação de todas as Diretivas associadas com a View corrente.
- Até este ponto a View está preparada e renderizável. Múltiplas mudanças pode ser entregues por Diretivas vindas da Detecção de Mudanças.
- Em algum momento a View pode ser removida. Neste ponto todas as diretivas são destruídas durante o processo de Desidratração e a View torna-se inativa.
- A View tem que esperar até que seja desanexada do DOM. O atraso em desanexar pode ser causado por alguma animação que esteja animando a saída da View.
- Após a view ser desanexada do DOM está pronta para ser reutilizada. O reuso de Views permite a aplicação ser mais rápida en rederizações subsequentes.