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

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 id
cparameter 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.