Dynamic component supports bindings and directives

Angular 20's createComponent function will support bindings and directives. The feature is in 20.0.0-next.1; therefore, it can be tested after updating the Angular dependencies to the next version. ng update @angular/cli --next ng update @angular/core --next This demo will show how to invoke the createComponent method to create a dynamic component displaying the information of Star Wars characters. The dynamic component has inputs, an output, and a host directive that needs to be set during creation. Define the AppStarWarCharacterComponent Compnent @Component({ selector: 'app-star-war-character', template: ` @if(person(); as person) { Id: {{ id() }} @if (isSith()) { A Sith, he is evil. } Name: {{ person.name }} Height: {{ person.height }} Alert parent } @else { No info } `, changeDetection: ChangeDetectionStrategy.OnPush, }) export class AppStarWarCharacterComponent { id = input(1); isSith = input(false); alertStarWars = output(); getPersonFn = getPerson(); person = toSignal(toObservable(this.id) .pipe( switchMap((id) => this.getPersonFn(id)), ) ); } The AppStarWarCharacterComponent component displays the details of a Star Wars character. The component has two signal inputs, id and isSith, which hold the value of the ID and whether or not the character is a Sith fighter. It also has an alertStarWars output that notifies the parent component to show an alert box. Finally, the component is dynamically bound to a host directive, AppLabelColorDirective, that changes the text color of the span elements. Implement of the Host Directive .blue span { color: blue; } .rebeccapurple span { color: rebeccapurple; } .red span { color: red; } The global stylesheet has some classes that override the text color of the span elements. @Directive({ selector: '[appLabelColorDirective]', host: { '[class]': 'spanClass()' } }) export class AppLabelColorDirective { spanClass = input('red'); } The AppLabelColorDirective is a host directive that sets the host class. The default value of the spanClass input is 'red', which changes the text color of the span elements to red. When the input value is 'blue', the span elements display blue text color. Finally, the texts are in rebeccapurple color when the input is 'rebeccapurple'. Create AppStarWarCharacterComponent Dynamically @Component({ selector: 'app-root', imports: [FormsModule, NgTemplateOutlet], template: ` @for (item of items; track item.id) { {{ item.name }} } @let text = isSith ? 'Add a Sith' : 'Add a Jedi'; {{ text }} `, changeDetection: ChangeDetectionStrategy.OnPush, }) export class AppComponent { vcr = viewChild.required('vcr', { read: ViewContainerRef }); componentRefs = [] as ComponentRef[]; jediId = signal(1); jediFighters = signal([ { id: 1, name: 'Luke' }, { id: 10, name: 'Obi Wan Kenobe' }, ]); async addAJedi(id: number, isSith = false) { const { AppStarWarCharacterComponent } = await import ('./star-war/star-war-character.component'); const componentRef = this.vcr() .createComponent(AppStarWarCharacterComponent, { bindings: [ inputBinding('id', () => id), inputBinding('isSith', () => isSith), outputBinding('alertStarWars', (name) => alert(`${name} alerts the parent component`)), ], directives: [ { type: AppLabelColorDirective, bindings: [ inputBinding('spanClass', () => isSith ? 'red' : 'rebeccapurple') ] } ] } ); this.componentRefs.push(componentRef); } } The inline template has a with a vcr template variable. It also displays a drop-down list containing names of the Jedi fighters. vcr = viewChild.required('vcr', { read: ViewContainerRef }); In the component class, the viewChild function queries the VierContainerRef and assigns to the vcr field. When users select a Jedi fighter and click the "Add a Jedi" button, the addAJedi method is triggered. The method imports the AppStarWarCharacterComponent class to create the component dynamically. const componentRef = this.vcr() .createComponent(AppStarWarCharacterComponent, { bindings: [ inputBinding('id', () => id), inputBinding('isSith', () => isSith), outputBinding('alertStarWars', (name) => alert(`${name} alerts the parent component`)), ], directives: [ { type: AppLabelColorDirective, bindings: [ inputBinding('spanClass', () => isSith ? 'red' : 'rebeccapurple') ] } ] } ); The ViewContainerRef class has a createComponent method; therefore, it is executed to create an instance of AppStarWarCharacterComponent

Mar 16, 2025 - 16:18
 0
Dynamic component supports bindings and directives

Angular 20's createComponent function will support bindings and directives.

The feature is in 20.0.0-next.1; therefore, it can be tested after updating the Angular dependencies to the next version.

ng update @angular/cli --next
ng update @angular/core --next

This demo will show how to invoke the createComponent method to create a dynamic component displaying the information of Star Wars characters. The dynamic component has inputs, an output, and a host directive that needs to be set during creation.

Define the AppStarWarCharacterComponent Compnent

@Component({
 selector: 'app-star-war-character',
 template: `
   
@if(person(); as person) {

Id: {{ id() }} @if (isSith()) {

A Sith, he is evil. }

Name: {{ person.name }}

Height: {{ person.height }} } @else {

No info }

`
, changeDetection: ChangeDetectionStrategy.OnPush, }) export class AppStarWarCharacterComponent { id = input(1); isSith = input(false); alertStarWars = output<string>(); getPersonFn = getPerson(); person = toSignal(toObservable(this.id) .pipe( switchMap((id) => this.getPersonFn(id)), ) ); }

The AppStarWarCharacterComponent component displays the details of a Star Wars character. The component has two signal inputs, id and isSith, which hold the value of the ID and whether or not the character is a Sith fighter. It also has an alertStarWars output that notifies the parent component to show an alert box. Finally, the component is dynamically bound to a host directive, AppLabelColorDirective, that changes the text color of the span elements.

Implement of the Host Directive

.blue span {
   color: blue;
}

.rebeccapurple span {
   color: rebeccapurple;
}

.red span {
   color: red;
}

The global stylesheet has some classes that override the text color of the span elements.

@Directive({
   selector: '[appLabelColorDirective]',
   host: {
       '[class]': 'spanClass()'
   }
})
export class AppLabelColorDirective {
   spanClass = input('red');
}

The AppLabelColorDirective is a host directive that sets the host class. The default value of the spanClass input is 'red', which changes the text color of the span elements to red. When the input value is 'blue', the span elements display blue text color. Finally, the texts are in rebeccapurple color when the input is 'rebeccapurple'.

Create AppStarWarCharacterComponent Dynamically

@Component({
 selector: 'app-root',
 imports: [FormsModule, NgTemplateOutlet],
 template: `
   
@let text = isSith ? 'Add a Sith' : 'Add a Jedi'; `
, changeDetection: ChangeDetectionStrategy.OnPush, }) export class AppComponent { vcr = viewChild.required('vcr', { read: ViewContainerRef }); componentRefs = [] as ComponentRef<any>[]; jediId = signal(1); jediFighters = signal([ { id: 1, name: 'Luke' }, { id: 10, name: 'Obi Wan Kenobe' }, ]); async addAJedi(id: number, isSith = false) { const { AppStarWarCharacterComponent } = await import ('./star-war/star-war-character.component'); const componentRef = this.vcr() .createComponent(AppStarWarCharacterComponent, { bindings: [ inputBinding('id', () => id), inputBinding('isSith', () => isSith), outputBinding<string>('alertStarWars', (name) => alert(`${name} alerts the parent component`)), ], directives: [ { type: AppLabelColorDirective, bindings: [ inputBinding('spanClass', () => isSith ? 'red' : 'rebeccapurple') ] } ] } ); this.componentRefs.push(componentRef); } }

The inline template has a with a vcr template variable. It also displays a drop-down list containing names of the Jedi fighters.

vcr = viewChild.required('vcr', { read: ViewContainerRef });

In the component class, the viewChild function queries the VierContainerRef and assigns to the vcr field.

When users select a Jedi fighter and click the "Add a Jedi" button, the addAJedi method is triggered. The method imports the AppStarWarCharacterComponent class to create the component dynamically.

const componentRef = this.vcr()
.createComponent(AppStarWarCharacterComponent,
     {
       bindings: [
         inputBinding('id', () => id),
         inputBinding('isSith', () => isSith),
         outputBinding<string>('alertStarWars', (name) => alert(`${name} alerts the parent component`)),
       ],
       directives: [
         {
           type: AppLabelColorDirective,
           bindings: [
             inputBinding('spanClass', () => isSith ? 'red' : 'rebeccapurple')
           ]
         }
       ]
     }
);

The ViewContainerRef class has a createComponent method; therefore, it is executed to create an instance of AppStarWarCharacterComponent and set the bindings and host directive. The idcparameter is bound to the id signal input and isSith parameter is bound to the isSith signal input. The callback function listens to the alertStarWars output to open an alert box displaying the character name. The directive array applies the AppLabelColorDirective directive to the component. When the isSith parameter is true, the spanClass signal input receives 'red'. When the parameter is false, the spanClass signal input receives 'rebeccaPurple'.

The createComponent method returns a ComponentRef that is inserted to the ViewContainerRef. The ComponentRef is tracked by the componentRefs array. When the App component is destroyed, the ngOnDestroy lifecycle hook method also runs to destroy the ComponentRef to release the memory to avoid memory leak.

Creating a dynamic component is easier in v20 when it supports custom events by output binding and changing DOM behavior through host directive.

References: