Динамическое создание компонентов Angular на лету
- вторник, 28 ноября 2023 г. в 00:00:12
В этой статье мы поговорим о создании компонентов динамически, шаг за шагом пройдем этот путь. Помимо простого создания компонентов, мы поговорим о более продвинутых вещах, которые можно сделать в рамках этого процесса.
В первую очередь нам нужен сам компонент, который мы будем динамически создавать.
@Component({
selector: "alert",
template: `
<h1>Alert {{type}}</h1>
`,
})
export class AlertComponent {
@Input() type: string = "success";
}
Будем использовать простой компонент alert, который принимает тип оповещения как инпут свойство
Стоит сказать, что динамические компоненты, а вернее те, которые создаются динамически, являются элементами DOM, и, соответственно, в шаблоне необходимо обеспечить место, куда его нужно добавить.
@Component({
selector: 'my-app',
template: `
<template #alertContainer></template>
`,
})
export class App {}
В компоненте my-app
создаем шаблон, используя тег ng-template
и добавляем к нему переменную через #. Этот template как раз является местом, куда добавится компонент, дальше будем называть это контейнером. В роли контейнера может выступать любой компонент или элемент DOM.
Теперь в компоненте нужно получить ссылку на контейнер. Это можно сделать с помощью ViewChild
. Кстати, у меня уже есть статья, которая поможет разобраться в процессах, связанных с взаимодействием с DOM элементами, где как раз подробно рассказывается о декораторе ViewChild
. Манипулируй DOM правильно.
@Component({
selector: 'my-app',
template: `
<template #alertContainer></template>
`,
})
export class App {
@ViewChild("alertContainer", { read: ViewContainerRef }) container;
}
Дефолтное значение, которое возвращается из ViewChild
, это экземпляр компонента или элемента DOM, но в нашем случае нужен ViewContainerRef
. Название этого класса говорит само за себя, ViewContainerRef
содержит в себе ссылку на контейнер, а также методы, которые позволяют создавать компоненты.
Подготовим небольшой интерфейс для создания компонентов. Добавим на страницу две кнопки, а также заинжектим один сервис в наш компонент.
@Component({
selector: 'my-app',
template: `
<template #alertContainer></template>
<button (click)="createComponent('success')">Create success alert</button>
<button (click)="createComponent('danger')">Create danger alert</button>
`,
})
export class App {
@ViewChild("alertContainer", { read: ViewContainerRef }) container;
constructor(private resolver: ComponentFactoryResolver) {}
}
Сервис ComponentFactoryResolver
содержит метод resolveComponentFactory
, который принимает в себя компонент, и возвращает ComponentFactory
, в котором, в свою очередь присутствует метод create()
, именно он и будет использоваться нашим контейнером для создания компонента.
Итак, теперь есть всё необходимое для метода, создающего компоненты.
createComponent(type) {
this.container.clear();
const factory: ComponentFactory = this.resolver.resolveComponentFactory(AlertComponent);
this.componentRef: ComponentRef = this.container.createComponent(factory);
}
Давайте пройдемся по порядку и посмотрим, что тут происходит
this.container.clear();
Каждый раз, когда мы нажмем кнопу для создания компонента, нам необходимо очистить контейнер от предыдущего представления. Иначе это будет подобно методу push()
для массивов, что в данной ситуации не является хорошей практикой.
const factory: ComponentFactory =
this.resolver.resolveComponentFactory(AlertComponent);
resolveComponentFactory()
принимает класс компонента и возвращает ComponentFactory
, где есть всё необходимое чтобы его создать.
this.componentRef: ComponentRef = this.container.createComponent(factory);
Мы вызываем метод createComponent()
и передаем переменную типа ComponentFactory
. Под капотом этот метод вызывает create()
и добавляет готовый компонент как дочерний в контейнер.
После всех этих взаимодействий у нас есть ссылка на компонент, и мы можем установить значение его Input свойства
this.componentRef.instance.type = type;
Так же можно и подписаться на Output свойство компонента.
this.componentRef.instance.output.subscribe(event => console.log(event));
Чтобы избежать утечек памяти, стоит уничтожить компонент после использования
ngOnDestroy() {
this.componentRef.destroy();
}
Метод для создания компонента готов, все операции изучены и понятны, и, в конце концов, остается только добавить динамический компонент в массив entryComponents в модуле.
@NgModule({
entryComponents: [ AlertComponent ],
bootstrap: [ App ]
})
export class AppModule {}
Ангуляр обеспечивает ряд возможностей по оптимизации работы вашего проекта. Одна из таких особенностей – это ленивая загрузка модулей, что позволяет подгружать различные модули с определенным функционалом по мере необходимости, а не сразу при старте приложения. Это позволяет сократить размер изначально загружаемого кода, уменьшить нагрузку на клиенте и т.д. Об этой возможности Ангуляр можно найти довольно много материалов и ознакомиться с ними подробнее.
Однако мы можем пойти дальше, и загружать лениво не только целые модули, но и отдельные компоненты на странице.
Давайте посмотрим, как это можно сделать, используя уже изученные методы из первой части статьи.
Создадим компонент BarComponent
@Component({
template: `
<h1 (click)="titleChanges.emit('changed')">{{ title }}</h1>
`
})
export class BarComponent implements OnInit {
title = 'Default';
titleChanges = new EventEmitter();
Тут присутствует свойство title
, а также свойство titleChanges
, на которое можно подписаться и принимать исходящие из компонента события.
Теперь давайте посмотрим, как лениво загрузить этот компонент, динамически его создать, и получить доступ к его свойствам.
@Component({
template: `
<button (click)="loadBar()">Load BarComponent</button>
<ng-template #vcr></ng-template>
`
})
export class MyComponent {
@ViewChild('vcr', { read: ViewContainerRef }) vcr: ViewContainerRef;
barRef: ComponentRef<BarComponent>;
constructor(private resolver: ComponentFactoryResolver) {}
async loadBar() {
if (!this.barRef) {
const { BarComponent } = await import(`./bar/bar.component`);
const factory = this.resolver.resolveComponentFactory(BarComponent);
this.barRef = this.vcr.createComponent(factory);
this.barRef.instance.title = 'Changed';
// Don't forget to unsubscribe
this.barRef.instance.titleChanges.subscribe(console.log);
}
}
}
Сперва, как и в первой части статьи, получим ссылку на контейнер через ViewChild
. После этого нужно загрузить сам компонент BarComponent
. Синтаксис этой загрузки очень похож на синтаксис, используемый при ленивой загрузке модулей.
const { BarComponent } = await import(`./bar/bar.component`);
Затем инжеткируем сервис ComponentFactoryResolver
, чтобы пользоваться его методами для создания компонента.
Дальнейшая последовательность действий аналогична динамическому созданию компонента.
После выполнения всех действий, у нас есть доступ к компоненту и его свойствам. Тут нужно обратить внимание, что мы не используем декораторы Input
и Output
, потому что мы общаемся с компонентом не через шаблон, а напрямую взаимодействуем с экземпляром компонента прямо в коде.
Однако если посмотреть на созданный компонент, можно сделать вывод что он довольно бесполезный и пригодится только чтобы отображать данные из входных свойств. Для полноты картины, было бы здорово обеспечить возможность использования остальные возможности Ангуляра в нем, например: другие компоненты, директивы, пайпы. Для решения этой проблемы мы создадим BarModule
, и добавим туда все необходимое. (В последней версии Ангуляр, а на данный момент — это 16 версия, появилась возможность использовать standalone
компоненты и напрямую подключать в него все используемые компоненты, модули, директивы и т.д.)
@NgModule({
imports: [ReactiveFormsModule],
declarations: [FooComponent]
})
class BarModule {
}
Самое интересное, что благодаря механизму tree-shaking, если единственное место, где мы будем использовать модуль ReactiveFormsModule
— это наш компонент BarComponent
, то исходный код реактивных форм загрузится только при загрузке BarComponent
. Это открывает новые возможности в вопросах оптимизации приложения.
Вышеописанный синтаксис отлично подойдет для использования в проектах, где версия Ангуляр 12 и ниже. Однако начиная с 13 версии класс ComponentFactoryResolver
был отмечен как deprecated
, и это повлияло на процесс создания динамических компонентов, так как теперь из последовательности действий ушел один шаг, а именно создание фабрики компонента.
const factory = this.resolver.resolveComponentFactory(BarComponent);
Теперь же для создания компонента можно напрямую обращаться к методу createComponent
и передавать класс создаваемого компонента в качестве параметра, на выходе получится тот же результат.
Динамическое создание компонентов открывает большие возможности по взаимодействию с ними и их управлению. Помимо всего этого в шаблонах компонента остается гораздо меньше кода с описанием всех input и output свойств, что так же хорошо влияет на весь проект. Ну и самое главное, это вопросы оптимизации. Благодаря более тонкой работе с компонентами, их динамическому созданию и ленивой загрузке открываются всё большие возможности по оптимизации приложения и улучшению пользовательского опыта при работе с ним. Данная статья не раскрывает собой все вопросы по данной теме, поэтому я советую вам более подробно ознакомиться со всеми материалами, которые существуют по тематике динамических компонентов.
Динамическое создание компонентов мощный инструмент, способный дать вашему проекту новые преимущества. В свою очередь вы как разработчик, изучив досконально этот вопрос, добавите себе в копилку ценные и важные навыки.
Буду рад обратной связи в карму или в комментариях.