DEV Community

Renuka Patil
Renuka Patil

Posted on

Angular Deep Dive: Exploring Templates, View Handling, and Directives

Mastering Angular Structural Elements and Directives

Angular is a powerful framework that offers various tools to enhance the UI and manage the DOM effectively. In this blog, we will cover the following concepts in detail with beginner-friendly examples and outputs:

  • ng-template
  • ng-container
  • ng-content
  • ElementRef
  • TemplateRef
  • ViewContainerRef
  • ngTemplateOutlet
  • Custom Directives
  • Renderer2
  • Structural Directives (*ngIf, *ngFor, etc.)

1. ng-template

The ng-template directive is used to define a reusable and non-rendered block of HTML until explicitly invoked. It is highly versatile and works with structural directives, custom directives, and dynamic content rendering. It is also instrumental in creating reusable components.

Advanced Use Cases

1. Using ng-template with ngIf

You can define templates for conditional rendering directly within an ngIf directive.

<div *ngIf="isLoggedIn; else loginTemplate">
  <p>Welcome back, user!</p>
</div>

<ng-template #loginTemplate>
  <p>Please log in to continue.</p>
</ng-template>
Enter fullscreen mode Exit fullscreen mode

2. Reusable Components

ng-template allows for content reuse in dynamic components. Here's an example:

Parent Component:

<app-message [template]="customTemplate"></app-message>

<ng-template #customTemplate>
  <p>Custom message content!</p>
</ng-template>
Enter fullscreen mode Exit fullscreen mode

Child Component:

<div>
  <ng-container *ngTemplateOutlet="template"></ng-container>
</div>
Enter fullscreen mode Exit fullscreen mode

Component Code:

import { Component, Input, TemplateRef } from '@angular/core';

@Component({
  selector: 'app-message',
  templateUrl: './message.component.html',
})
export class MessageComponent {
  @Input() template!: TemplateRef<any>;
}
Enter fullscreen mode Exit fullscreen mode

3. Dynamic Context Passing

With ngTemplateOutletContext, you can pass data dynamically to templates, enabling more flexible logic.

<ng-template #greeting let-name="name">
  <p>Hello, {{ name }}!</p>
</ng-template>

<ng-container *ngTemplateOutlet="greeting; context: { name: 'Angular Developer' }"></ng-container>
Enter fullscreen mode Exit fullscreen mode

4. Usage in Custom Directives

You can combine ng-template with custom structural directives to create powerful DOM transformations. For instance, rendering templates conditionally:

import { Directive, Input, TemplateRef, ViewContainerRef } from '@angular/core';

@Directive({
  selector: '[appCustomRender]',
})
export class CustomRenderDirective {
  @Input() set appCustomRender(condition: boolean) {
    if (condition) {
      this.vcr.createEmbeddedView(this.templateRef);
    } else {
      this.vcr.clear();
    }
  }

  constructor(private templateRef: TemplateRef<any>, private vcr: ViewContainerRef) {}
}
Enter fullscreen mode Exit fullscreen mode

Usage:

<ng-template [appCustomRender]="isActive">
  <p>Active Content Rendered!</p>
</ng-template>
Enter fullscreen mode Exit fullscreen mode

Summary

ng-template is an essential tool in Angular for defining reusable, dynamic, and efficient templates. By leveraging its advanced capabilities, you can create robust and modular components in your Angular applications. ng-template directive is used to define a reusable and non-rendered block of HTML until explicitly invoked. It is highly versatile and works with structural directives and dynamic content rendering.

Use Cases

  • Conditional rendering using *ngIf.
  • Storing reusable template code.
  • Dynamic content rendering with ngTemplateOutlet.
  • Placeholder content.
  • Creating dynamic components in reusable modules.

Example

<ng-template>
  <p>This content is rendered using ng-template!</p>
</ng-template>

<!-- Usage of ng-template -->
<div *ngIf="showContent; else noContent">
  <p>Content is shown!</p>
</div>

<ng-template #noContent>
  <p>No content to show!</p>
</ng-template>
Enter fullscreen mode Exit fullscreen mode

Component Code

import { Component } from '@angular/core';

@Component({
  selector: 'app-root',
  templateUrl: './app.component.html',
})
export class AppComponent {
  showContent = false;
}
Enter fullscreen mode Exit fullscreen mode

Output

When showContent is false:

No content to show!
Enter fullscreen mode Exit fullscreen mode

When showContent is true:

Content is shown!
Enter fullscreen mode Exit fullscreen mode

2. ng-container

The ng-container is a grouping element that doesn't render an additional DOM element. It is used to group elements logically and apply directives without affecting the DOM structure.

Use Cases

  • Logical grouping of elements for conditional rendering.
  • Avoid unnecessary wrappers in the DOM.
  • Using multiple structural directives together.
  • Creating dynamic tables or grids efficiently.

Example

<ng-container *ngIf="isLoggedIn">
  <p>Welcome, user!</p>
</ng-container>
Enter fullscreen mode Exit fullscreen mode

Component Code

isLoggedIn = true;
Enter fullscreen mode Exit fullscreen mode

Output

Welcome, user!
Enter fullscreen mode Exit fullscreen mode

If isLoggedIn is false, no output is rendered.


3. ng-content

The ng-content directive is used to project content into a component from its parent. It enables developers to create highly customizable and reusable components.

Use Cases

  • Content projection in reusable components.
  • Supporting slot-like behavior in components.
  • Projecting multiple slots with select attribute.
  • Implementing dynamic layouts in parent-child components.

Example

<!-- Parent Component -->
<app-card>
  <p>Content to be projected</p>
</app-card>

<!-- Child Component Template -->
<div class="card">
  <ng-content></ng-content>
</div>
Enter fullscreen mode Exit fullscreen mode

Output

<div class="card">
  <p>Content to be projected</p>
</div>
Enter fullscreen mode Exit fullscreen mode

4. ElementRef

ElementRef is used to directly access and manipulate DOM elements. Use it cautiously, as improper usage can lead to security risks like XSS.

Use Cases

  • Accessing and modifying native DOM elements.
  • Direct interaction with element attributes or styles.
  • Custom directive implementation.
  • Integrating third-party libraries that require direct DOM manipulation.

Example

import { Component, ElementRef, ViewChild } from '@angular/core';

@Component({
  selector: 'app-root',
  template: '<button #btn>Click me!</button>',
})
export class AppComponent {
  @ViewChild('btn', { static: true }) button!: ElementRef;

  ngAfterViewInit() {
    this.button.nativeElement.style.backgroundColor = 'blue';
  }
}
Enter fullscreen mode Exit fullscreen mode

Output

The button will have a blue background color.


5. TemplateRef

TemplateRef represents an embedded template. It is used to work with structural directives or programmatically render templates.

Use Cases

  • Creating dynamic views with ViewContainerRef.
  • Passing templates as inputs.
  • Handling reusable template logic.
  • Using TemplateRef to control rendering logic dynamically.

Example

import { Component, TemplateRef, ViewChild } from '@angular/core';

@Component({
  selector: 'app-root',
  template: `
    <ng-template #tpl>
      <p>Rendered by TemplateRef</p>
    </ng-template>
  `,
})
export class AppComponent {
  @ViewChild('tpl', { static: true }) template!: TemplateRef<any>;

  constructor() {}
}
Enter fullscreen mode Exit fullscreen mode

6. ViewContainerRef

ViewContainerRef is used to dynamically insert templates or components into the DOM at runtime.

Use Cases

  • Dynamically adding or removing components.
  • Rendering templates programmatically.
  • Creating reusable content renderers.
  • Efficiently managing dynamic lists or grids.

Example

import { Component, TemplateRef, ViewChild, ViewContainerRef } from '@angular/core';

@Component({
  selector: 'app-root',
  template: `
    <ng-template #tpl>
      <p>Dynamically rendered content!</p>
    </ng-template>
    <button (click)="render()">Render Content</button>
  `,
})
export class AppComponent {
  @ViewChild('tpl', { static: true }) template!: TemplateRef<any>;

  constructor(private vcr: ViewContainerRef) {}

  render() {
    this.vcr.createEmbeddedView(this.template);
  }
}
Enter fullscreen mode Exit fullscreen mode

7. ngTemplateOutlet

ngTemplateOutlet allows you to render an ng-template programmatically. It simplifies template management and reuse, and provides powerful options for passing data to templates dynamically.

Use Cases

  • Rendering conditional templates.
  • Reusing predefined templates dynamically.
  • Passing context to templates.
  • Simplifying repetitive template logic.

Example: Basic Usage

<ng-template #template1>
  <p>This is Template 1</p>
</ng-template>

<ng-container *ngTemplateOutlet="template1"></ng-container>
Enter fullscreen mode Exit fullscreen mode

Output

This is Template 1
Enter fullscreen mode Exit fullscreen mode

Example: Passing Context Data to Templates

Using ngTemplateOutletContext, you can pass data to an ng-template dynamically.

<ng-template #template2 let-name="name">
  <p>Hello, {{ name }}!</p>
</ng-template>

<ng-container *ngTemplateOutlet="template2; context: { name: 'Angular' }"></ng-container>
Enter fullscreen mode Exit fullscreen mode

Output

Hello, Angular!
Enter fullscreen mode Exit fullscreen mode

Example: Conditional Rendering with Context

<ng-template #loggedIn let-user="user">
  <p>Welcome back, {{ user }}!</p>
</ng-template>

<ng-template #loggedOut>
  <p>Please log in to continue.</p>
</ng-template>

<ng-container *ngTemplateOutlet="isLoggedIn ? loggedIn : loggedOut; context: { user: 'John' }"></ng-container>
Enter fullscreen mode Exit fullscreen mode

Component Code

import { Component } from '@angular/core';

@Component({
  selector: 'app-root',
  templateUrl: './app.component.html',
})
export class AppComponent {
  isLoggedIn = true;
}
Enter fullscreen mode Exit fullscreen mode

Output

If isLoggedIn is true:

Welcome back, John!
Enter fullscreen mode Exit fullscreen mode

If isLoggedIn is false:

Please log in to continue.

allows you to render an ng-template programmatically. It simplifies template management and reuse.

Use Cases

  • Rendering conditional templates.
  • Reusing predefined templates dynamically.
  • Passing context to templates.
  • Simplifying repetitive template logic.

Example

<ng-template #template1>
  <p>This is Template 1</p>
</ng-template>

<ng-container *ngTemplateOutlet="template1"></ng-container>
Enter fullscreen mode Exit fullscreen mode

Output

This is Template 1


8. Custom Directives

Directives extend the behavior of elements or components. Below is an example of a directive that changes the background color of an element.

Use Cases

  • Modifying DOM behavior dynamically.
  • Handling user interactions like hover or click events.
  • Creating reusable logic encapsulated in directives.
  • Implementing animation triggers or scroll-based interactions.
  • Building structural directives to transform the DOM dynamically.

Example: Attribute Directive

import { Directive, ElementRef, Renderer2, HostListener } from '@angular/core';

@Directive({
  selector: '[appHighlight]',
})
export class HighlightDirective {
  constructor(private el: ElementRef, private renderer: Renderer2) {}

  @HostListener('mouseenter') onMouseEnter() {
    this.renderer.setStyle(this.el.nativeElement, 'background-color', 'yellow');
  }

  @HostListener('mouseleave') onMouseLeave() {
    this.renderer.removeStyle(this.el.nativeElement, 'background-color');
  }
}
Enter fullscreen mode Exit fullscreen mode

Usage

<p appHighlight>Hover over me to highlight!</p>
Enter fullscreen mode Exit fullscreen mode

Output

The background of the paragraph changes to yellow when hovered over.

Example: Structural Directive

Structural directives can modify the DOM by adding or removing elements dynamically. Here is an example to toggle visibility based on a condition.

import { Directive, Input, TemplateRef, ViewContainerRef } from '@angular/core';

@Directive({
  selector: '[appIf]',
})
export class IfDirective {
  constructor(private templateRef: TemplateRef<any>, private viewContainer: ViewContainerRef) {}

  @Input() set appIf(condition: boolean) {
    if (condition) {
      this.viewContainer.createEmbeddedView(this.templateRef);
    } else {
      this.viewContainer.clear();
    }
  }
}
Enter fullscreen mode Exit fullscreen mode

Usage

<div *appIf="isVisible">This content is conditionally visible!</div>
Enter fullscreen mode Exit fullscreen mode

Component Code

isVisible = true;
Enter fullscreen mode Exit fullscreen mode

Output

If isVisible is true, the content will be rendered:

This content is conditionally visible!
Enter fullscreen mode Exit fullscreen mode

If isVisible is false, the content will not be displayed. extend the behavior of elements or components. Below is an example of a directive that changes the background color of an element.

Use Cases

  • Modifying DOM behavior dynamically.
  • Handling user interactions like hover or click events.
  • Creating reusable logic encapsulated in directives.
  • Implementing animation triggers or scroll-based interactions.

Example

import { Directive, ElementRef, Renderer2, HostListener } from '@angular/core';

@Directive({
  selector: '[appHighlight]',
})
export class HighlightDirective {
  constructor(private el: ElementRef, private renderer: Renderer2) {}

  @HostListener('mouseenter') onMouseEnter() {
    this.renderer.setStyle(this.el.nativeElement, 'background-color', 'yellow');
  }

  @HostListener('mouseleave') onMouseLeave() {
    this.renderer.removeStyle(this.el.nativeElement, 'background-color');
  }
}
Enter fullscreen mode Exit fullscreen mode

Usage

<p appHighlight>Hover over me to highlight!</p>
Enter fullscreen mode Exit fullscreen mode

Output

The background of the paragraph changes to yellow when hovered over.


9. Renderer2

Renderer2 is a service used for safely manipulating the DOM. It is a preferred alternative to ElementRef for DOM interactions.

Use Cases

  • Changing element attributes or styles.
  • Adding or removing DOM elements dynamically.
  • Safeguarding against potential XSS attacks.

Example

import { Directive, ElementRef, Renderer2 } from '@angular/core';

@Directive({
  selector: '[appSafeHighlight]',
})
export class SafeHighlightDirective {
  constructor(private el: ElementRef, private renderer: Renderer2) {
    this.renderer.setStyle(this.el.nativeElement, 'border', '1px solid red');
  }
}
Enter fullscreen mode Exit fullscreen mode

Usage

<p appSafeHighlight>This paragraph has a red border.</p>
Enter fullscreen mode Exit fullscreen mode

By understanding these concepts and practicing the examples, you'll gain a solid foundation for working with Angular's powerful tools and directives. Happy coding!

Top comments (1)

Collapse
 
happygate profile image
happygate

The newest version of Angular does not use the *ngIf and *ngFor directives, they have a new control flow syntax using "@if" and "@for"