Learn Angular ngFor built-in directive and trackBy
Introduction
*ngFor is a structural and built-in directive that is used to iterate over collections like an array and create a template for each item.
What is Structural Directive?
A structural directive adds and removes elements in the DOM tree. Here, *ngFor does the same.
Basic Syntax of ngFor
- <tr *ngFor="let s of students">
- <td>{{s.Name}}</td>
- </tr>
Example
Let’s take an example and understand this with the help of it - Suppose we have list of student objects with some of their information, like StudentID, Name, DOB, StuClass, Gender, Age & Address.
- [
- { studentID: 1, studentName: 'Steve', gender: 'Male', age: 35, course: 'MCA' },
- { studentID: 2, studentName: 'Bobby', gender: 'Male', age: 32, course: 'MBA' },
- { studentID: 3, studentName: 'Rina', gender: 'Female', age: 45, course: 'B.Tech' },
- { studentID: 4, studentName: 'Alex', gender: 'Female', age: 24, course: 'M.Tech' },
- ]
And, we want to have this data in HTML table format, like below.
This is how we can do using ngFor directive introduced in Angular.
Let’s look at it via code.
Step 1
Let’s create a folder named Student under src/app folder.
Let’s create a folder named Student under src/app folder.
Add a TypeScript file named student.component.ts under this folder and add the below code into it.
- import {Component} from '@angular/core'
- @Component({
- selector: 'list-student',
- templateUrl: 'app/student/student.component.html',
- styles: [
- `
- td {
- border: 1px solid black;
- padding:4px;
- }
- th {
- border: 1px solid black;
- padding:4px;
- }
- `
- ]
- })
- export class StudentListComponent {
- students: any[] =
- [
- { studentID: 1, studentName: 'Steve', gender: 'Male', age: 35, course: 'MCA' },
- { studentID: 2, studentName: 'Bobby', gender: 'Male', age: 32, course: 'MBA' },
- { studentID: 3, studentName: 'Rina', gender: 'Female', age: 45, course: 'B.Tech' },
- { studentID: 4, studentName: 'Alex', gender: 'Female', age: 24, course: 'M.Tech' },
- ];
- }
For making the class named StudentistComponent an Angular component, I decorated it with component decorator by importing component from Angular core.
templateUrl
For using view of this component, I used templateUrl property to assign an external HTML file.
For using view of this component, I used templateUrl property to assign an external HTML file.
Styles
For applying CSS styling over table output, I included some style code under styles property. We can use also styleUrls property in case we want to use some external CSS file for styling.
For applying CSS styling over table output, I included some style code under styles property. We can use also styleUrls property in case we want to use some external CSS file for styling.
Step 2
Let’s add a view file named student.component.html under student folder,
Let’s add a view file named student.component.html under student folder,
And add the below HTML code into this this file,
- <table>
- <thead>
- <tr>
- <th>Student ID</th>
- <th>Name</th>
- <th>Gender</th>
- <th>Age</th>
- <th>Course</th>
- </tr>
- </thead>
- <tbody>
- <tr *ngFor="let s of students">
- <td>{{s.studentID}}</td>
- <td>{{s.studentName}}</td>
- <td>{{s.gender}}</td>
- <td>{{s.age}}</td>
- <td>{{s.course}}</td>
- </tr>
- </tbody>
- </table>
*ngFor
It’s a built-in directive that is used to iterate over student list and add student item in the DOM tree.
It’s a built-in directive that is used to iterate over student list and add student item in the DOM tree.
Step 3
Under app folder, open app.component.ts file and add the list-student directive as view,
Under app folder, open app.component.ts file and add the list-student directive as view,
- import { Component } from '@angular/core';
- @Component({
- selector: 'my-app',
- template: `<list-student></list-student>`,
- })
- export class AppComponent {
- name = 'Angular2 Student Demo';
- }
Step 4
Under the same app folder, open app.module.ts file and import the student component as well as add your component under declarations like below,
Under the same app folder, open app.module.ts file and import the student component as well as add your component under declarations like below,
- import { NgModule } from '@angular/core';
- import { BrowserModule } from '@angular/platform-browser';
- import { AppComponent } from './app.component';
- import { StudentListComponent } from './student/student.component'
- @NgModule({
- imports: [BrowserModule],
- declarations: [AppComponent, StudentListComponent],
- bootstrap: [ AppComponent ]
- })
- export class AppModule { }
Step 5
Now, our application is ready to show the expected output. Let’s run our project using F5 and see the output in browser.
There are couple of local variables with ngFor built-in structural directive provided by Angular, which is used for getting the index of current element while iteration and for getting last or first element from the collection etc. as mentioned below –
- Index – Used for provide the index for current element while iteration.
- First – Return true if current element is first element in the iteration otherwise false.
- Last – Return true if current element is last element in the iteration otherwise false.
- Even – Return true if current element is even element based on index in the iteration otherwise false.
- Odd – Return true if current element is odd element based on index in the iteration otherwise false.
Let’s understand all these local variables with the help of an example –
Here, I am using the same student application which I used in my previous article for *ngFor directive. Look at my previous article on *ngFor directive.
Index with *ngFor directive
The index gives the index of the current element in the iteration. Let’s look into the example below. I have a list of students which I am displaying in the table on the web page as below. Now, I want to add one more column of index at first position, which will represent of index value of each item like below,
Let’s add the code for displaying the index of the current item in student.component.html file and run it for the expected output – (highlighted code)
- <table>
- <thead>
- <tr>
- <th>Index</th>
- <th>Student ID</th>
- <th>Name</th>
- <th>Gender</th>
- <th>Age</th>
- <th>Course</th>
- </tr>
- </thead>
- <tbody>
- <tr *ngFor="let s of students;let i=index">
- <td>{{i}}</td>
- <td>{{s.studentID}}</td>
- <td>{{s.studentName}}</td>
- <td>{{s.gender}}</td>
- <td>{{s.age}}</td>
- <td>{{s.course}}</td>
- </tr>
- </tbody>
- </table>
Output
First & Last with *ngFor directive
First & Last both provides the Boolean value. Returns true if current element value is first element or last element otherwise false. Let’s look into highlighted code & example below,
- <th>Course</th>
- <th>First Element</th>
- <th>Last Element</th>
- </tr>
- </thead>
- <tbody>
- <tr *ngFor="let s of students;let i=index;let f=first;let l=last">
- <td>{{i}}</td>
- <td>{{s.studentID}}</td>
- <td>{{s.studentName}}</td>
- <td>{{s.gender}}</td>
- <td>{{s.age}}</td>
- <td>{{s.course}}</td>
- <td>{{f}}</td>
- <td>{{l}}</td>
- </tr>
- </tbody>
- </table>
Output
Even & Odd with *ngFor directive
Even & odd both provides the Boolean value. Returns true if current element value is even element based on index or odd element based on index, otherwise false. Let’s look into highlighted code & example below,
- <th>Last Element</th>
- <th>Even Element</th>
- <th>Odd Element</th>
- </tr>
- </thead>
- <tbody>
- <tr *ngFor="let s of students;let i=index;let f=first;let l=last;let ev=even;let od=odd">
- <td>{{i}}</td>
- <td>{{s.studentID}}</td>
- <td>{{s.studentName}}</td>
- <td>{{s.gender}}</td>
- <td>{{s.age}}</td>
- <td>{{s.course}}</td>
- <td>{{f}}</td>
- <td>{{l}}</td>
- <td>{{ev}}</td>
- <td>{{od}}</td>
- </tr>
- </tbody>
- </table>
Output
trackBy with *ngFor directive
To iterate over a collection in Angular 2, I used *ngFor directive in above example.
Let’s understand with the same example why we needed trackBy function with *ngFor directive,
Suppose we have some data coming from some API request into the collection and we need to change the data over the web page using ngFor directive. In this case, Angular will remove all the DOM elements that associated with the data and will create them again in the DOM tree, even the same data is coming (as shown in below screenshot). That means a lot of Dom manipulations will happen if a large amount of data coming from the API.
The solution is, we can use trackBy function, which will be helpful for Angular to track the items which have been added or removed.
TrackBy function will take two arguments first is index and second is current item and we can return the unique identifier as a return argument.
Let’s add the below function into student.component.ts file and look now,
- trackByStudentID(index: number, student: any): string {
- return student.studentID;
- }
Updated student.component.ts code file,
- export class StudentListComponent {
- students: any[];
- constructor() {
- this.students = [{
- studentID: 1,
- studentName: 'Steve',
- gender: 'Male',
- age: 35,
- course: 'MCA'
- }, {
- studentID: 2,
- studentName: 'Bobby',
- gender: 'Male',
- age: 32,
- course: 'MBA'
- }, {
- studentID: 3,
- studentName: 'Rina',
- gender: 'Female',
- age: 45,
- course: 'B.Tech'
- }, {
- studentID: 4,
- studentName: 'Alex',
- gender: 'Female',
- age: 24,
- course: 'M.Tech'
- }, ];
- }
- getStudents(): void {
- this.students = [{
- studentID: 1,
- studentName: 'Steve',
- gender: 'Male',
- age: 35,
- course: 'MCA'
- }, {
- studentID: 2,
- studentName: 'Bobby',
- gender: 'Male',
- age: 32,
- course: 'MBA'
- }, {
- studentID: 3,
- studentName: 'Rina',
- gender: 'Female',
- age: 45,
- course: 'B.Tech'
- }, {
- studentID: 4,
- studentName: 'Alex',
- gender: 'Female',
- age: 24,
- course: 'M.Tech'
- }, {
- studentID: 5,
- studentName: 'Rahul',
- gender: 'Male',
- age: 26,
- course: 'MCA'
- }, ];
- }
- trackByStudentID(index: number, employee: any): string {
- return employee.studentID;
- }
- }
Let’s add trackBy function in student.component.html view file,
- <th>Odd Element</th>
- </tr>
- </thead>
- <tbody>
- <tr *ngFor="let s of students;let i=index;let f=first;let l=last;let ev=even;let od=odd;trackBy:trackByStudentID">
- <td>{{i}}</td>
- <td>{{s.studentID}}</td>
- <td>{{s.studentName}}</td>
- <td>{{s.gender}}</td>
- <td>{{s.age}}</td>
- <td>{{s.course}}</td>
- <td>{{f}}</td>
- <td>{{l}}</td>
- <td>{{ev}}</td>
- <td>{{od}}</td>
- </tr>
- </tbody>
- </table> <br /> <button (click)='getStudents()'>Get More Students</button>
Let’s look now at output and DOM elements in developer tools,
In the above example, on clicking "Get More Students" button, all the DOM elements are not affected, but only last one element has changed color of <tr> tag. That means Angular is now able to track all the items in the collection with the help of trackBy function.
Summary
This article described how to use ngFor built-in structural directive and its local variables and trackBy function with ngFor directive; why we need trackBy function; and how Angular tracks all the elements with the help of trackBy function.
To setup Angular project in your Visual Studio, download the "Quick Start Project" from the given link and use the contents from the downloaded folder into your project.
Write me in the comment box in case you want some help in setting up Angular into your Visual Studio.
Write me in the comment box in case you want some help in setting up Angular into your Visual Studio.
Learn Angular 5 from scratch with my Angular 5 Video Tutorial. YouTube link here - https://www.youtube.com/watch?v=QQTQb-OAMVQ
ReplyDeletehttps://www.youtube.com/watch?v=To_vaO4Rd6Q
Subscribe channel for upcoming Angular 5 tutorial videos. Keep watching, Keep Studying!
ReplyDelete