Understanding Content Projection in Angular
• • 3 min readContent projection, what is it?
Content projection (also known as transclusion) is a way to import HTML content from outside the component and insert that content into the component's template in a designated spot.
Therefore, using content-projection you can obtain the result shown in the following image.
data:image/s3,"s3://crabby-images/9e255/9e255b7f8a2cd648a310fd5b71eac9493598d1ba" alt=""
Notice that the component works like a mirror. Another more complicated example in the one shown in the following image in which a letter with user data is being configured.
data:image/s3,"s3://crabby-images/fc927/fc9275c62452a100a98425e3937772fc2ce482ab" alt=""
I saw a ng-container, what is that?
The Angular ng-container is a grouping element (syntax element) that doesn't interfere with styles or layout because Angular doesn't put it in the DOM.
And we can use structural directives with it.
Structural... What?
A Structural directive changes the DOM layout by adding, removing or manipulating DOM elements.
And then can I...?
data:image/s3,"s3://crabby-images/1250a/1250ac864e7fe92cddbb688a4a98d12e0d1ab41f" alt=""
where am-icon
is the following component:
import { Component, ViewChild, ElementRef, AfterContentChecked } from '@angular/core';
@Component({
selector: 'am-icon',
template: `
<!-- A bit tricky... 🤟🏻 -->
<span #content [hidden]="true"><ng-content></ng-content></span>
<ng-container [ngSwitch]="icon">
<ng-container *ngSwitchCase="'fire'">🔥</ng-container>
<ng-container *ngSwitchCase="'heavy'">🤟🏻</ng-container>
<ng-container *ngSwitchDefault>🖖🏻</ng-container>
</ng-container>
`,
})
export class IconComponent implements AfterContentChecked {
icon: string;
@ViewChild('content') content: ElementRef<HTMLSpanElement>;
// Using OnPush we need to run change detection by hand
ngAfterContentChecked() {
if (this.icon !== this.content.nativeElement.textContent) {
this.icon = this.content.nativeElement.textContent;
}
}
}
The result is an icon instead of four, just as we expected. 💔
data:image/s3,"s3://crabby-images/19030/19030525e0e355f1b961fb891e61ac2c03d21236" alt=""
How does ng-content work?
- ng-content doesn't produce content. It simply projects the existing content
- consistency of expectations 👀
- performance 🔥
- Following the previous rule, it can neither create nor destroy components projected (lifecycle).
- Only the last ng-content projects the content.
And... what can I do? Here ng-template comes to the rescue...
data:image/s3,"s3://crabby-images/5f8e6/5f8e6455d910e4cb15e6c6747c3a4639462cf4a6" alt=""
What is a ng-template?
As the name suggests, it is a template element, a model which you can instantiate, hence you can set a template as a component's input, which is pretty useful.
The asterisk (*) syntax
The asterisk is syntactic sugar for something a bit more complicated.
You usually see with: *ngIf, *ngFor, *anyDirective, ...
data:image/s3,"s3://crabby-images/b29d0/b29d011c1849ea2af8d37add34b3f3f81fde59dd" alt=""
Then our repeat component?
data:image/s3,"s3://crabby-images/11838/11838c401ff3b7a12a71b933306fc49f70deae83" alt=""
data:image/s3,"s3://crabby-images/62178/62178d45ae6ce003b312b608f1cec80a2ef2d51e" alt=""
Explain me what that ngTemplateOutlet is!
The ngTemplateOutlet directive receives a ng-template and it's responsible for create the instance and insert it into the DOM.
data:image/s3,"s3://crabby-images/29215/29215c546b778e330cfae6e5197ea6d6cb72cebc" alt=""
Yeah, context like in JS!
We can give an object to the instance, which is created by the ngTemplateOutlet, of your ng-template. That object can contain whatever you want to pass to your template. Here, we expose the magic of a lot of library components (datatables, angular-material, ng-bootstrap, ...).
We can do awesome things!
data:image/s3,"s3://crabby-images/331e8/331e860e48f2b768ea8d639c204f7a8c0c193adc" alt=""
More more more...
- Presentation
- The GitHub branch of this post is https://github.com/Caballerog/ng-transclusion/