Angular Lazy-loading

Lazy loading is a technology of angular that allows you to load JavaScript components when a specific route is activated. It improves application load time speed by splitting the application into many bundles. When the user navigates by the app, bundles are loaded as needed.

Lazy loading helps to keep the bundle size small, which helps reduce load times. We must use the class decorator to create an Angular module @NgModule, and the decorator uses a metadata object that defines the module.

The main properties are:

  • import: Components of this module are used with Array with other modules.
  • Declarations: It receives an array of the components.
  • Export: Defines an array of components, directives, and pipes used by other modules.
  • Provider: Declares services that are available to the entire application if it is a root module.

Example of a native root module:

import { BrowserModule } from '@angular/platform browser;  

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

import { AppComponent} from './app.component';  

import { AppRoutingModule } from './app-routing.module';  

  

@NgModule({  

  Announcement: [  

    AppComponent  

  ],  

  import: [  

    browser module,  

    app routing module  

  ],  

  Providers: [], //Os service declarados and Propried providers do modulo root ficum disponivis m toda a applicacao  

  bootstrap: [AppComponent] //define some or module root um bootstrap  

})  

export class AppModule { }

In the above sample code, we have converted the AppModule class into an Angular module using the @NgModule decorator.

Lazy Loading Basics

This section introduces the basic process of lazy-loaded routing configuration.

In the lazy load Angular modules, use load children in your AppRoutingModule root configuration as follows.

AppRoutingModule (excerpt)  

content_copyconst routes: Routes = [  

  {  

    path: 'item',  

    loadChildren: () => import('./items/items.module').then(m => m.ItemModule)  

  }  

];

Add a route for the component in the lazy-loaded routing module.

Routing module for the lazy-loaded module

content_copyconst routes: Routes = [  

  {  

    path: '',  

    component: ItemsComponent  

  }  

];

Continue the below sections of the page for step-by-step instructions on lazy loading modules.

Please follow the below steps to implement Lazy Loading in an Angular Project:

Step 1:

ng new angular-lazy-loading

Create a new Angular project

Step 2:

ng g m lazy-loading-routing

The purpose of independent routing is to handle all components with Angular lazy-loading modules.

ng g m lazy-loading-routing
Angular Lazy-loading

Step 3:

Create a component called lazy-demo within the lazy-loading module,

ng g c lazy demode 
Angular Lazy-loading

Step 4:

Adding a link to the header will implement lazy loading.

app.component.html

<li class="nav-item">  

   <a class="nav-link" [routerLink]="['/lazy-loading']">  

      slow loading  

   </a>  

</li>

Step 5:

Here we will lazily load the module using load children.

app-routing.module.ts

{  

   path: 'lazy-loading',  

   load children: () => import('./lazy-loading/lazy-loading.module')  

   .then(m => m.LazyLoadingModule)  

},

Step 6:

Set the route in lazy-loading-routing.module.ts.

lazy-loading-routing.module.ts

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

import {RouterModule, Routes} from '@angular/router';  

import { LazyDemoComponent } from './lazy-demo/lazy-demo.component';  

const route: route = [  

 { path: '', component: LazyDemoComponent }  

];  

 @NgModule({  

    import: [routerModule. for child (route)],  

    export: [routermodule]  

})  

export class LazyLoadingRoutingModule { }

If above import option gives error, then try this,

import { LazyDemoComponent } from "E:/Pranam/Git Repository Projects/Angular-Lazy-Loading-Example/src/app/lazy-demo/lazy-demo.component";

Run the application.

Go to the terminal and execute the “ng serve” command.

Angular Lazy-loading

Now go to the browser https://localhost:4200.

Angular Lazy-loading

You will see that main.js is being served upon refreshing the browser. The Lazy Loading module is loaded only when the root /lazy-loading is hit.

Validate lazy loading

Run the following command to generate the build,

Angular Lazy-loading
npm run build

The above image verifies that a separate section is generated for the lazy loading module.

There you will see a separate file for modules that use lazy loading.

Angular Lazy-loading

There are two steps to install the Lazy Loaded Feature Module:

  • Build feature module with CLI using –route flag.
  • Configure routes.

Set Up an App

If you don’t have an application, you can follow the below steps to create an app with CLI. If you already have an app, go the configured route. Enter the command below where customer-app is the name of the app:

content_copy new client-app --routing

It creates an application named client-app and a file called the routing flag app-routing.module.ts, one of the important files to set up lazy loading. Navigate to the project by issuing the cd client-app command.

Create a Module with Routing

The path to load the client feature module is the client as it is specified with the –route option:

content_copyng new customer-app --routing

It creates an application called client-app, and the –routing flag creates a file called app-routing.module.ts, which is one of the files needed to set up lazy loading for your feature module. Navigate to the project by issuing the cd client-app command.

The –routing option used Angular/CLI version 8.1 or higher.

It defines a new lazy-loadable convenience module, CustomersModule in the Customer.module.ts file, and the routing module CustomersRoutingModule in the customers-routing.module.ts file.

content_copyconst routes: Routes = [  

  {  

    path: 'customers',  

    loadChildren: () => import('./customers/customers.module').then(m => m.CustomersModule)  

  }  

];

String-based lazy loading

The string syntax for the loadChildren route was deprecated in the import() syntax. You can choose to use string-based lazy loading (loadChildren: ‘./path/to/module#Module’) to include lazy-loaded routes in your tsconfig file, including lazy-loading compilation.

By default, the CLI will create strict file inclusion projects with import() syntax.

Add another Feature Module

Use the same command to create a lazy-loaded module with routing and its stub component.

content_copyng generate module orders --route orders --module app. module  

It creates a new folder called Orders which contains the Order module and the Order Routing module, and the new OrderComponent source files.

Set up the UI

src/app/app-routing.module.ts  

content_copyconst routes: Routes = [  

  {  

    path: 'customers',  

    loadChildren: () => import('./customers/customers.module').then(m => m.CustomersModule)  

  },  

  {  

    path: 'orders',  

    loadChildren: () => import('./orders/orders.module').then(m => m.OrdersModule)  

  }  

];

You can type the URL in the address bar, and navigation UI is easy and common for the user.

src/app/app.component.html  

content_copy<h1>  

  {{title}}  

</h1>  

<button routerLink="/customers">Customers</button>  

<button routerLink="/orders">Orders</button>  

<button routerLink="">Home</button>  

<router-outlet></router-outlet>

To see your application in the browser so far, enter the below-given command in the terminal window:

content_copying serve

Go to localhost:4200, where you can see “customer-app” and three buttons.

Angular Lazy-loading

The buttons work because the CLI adds routes to the feature module in the Routes array of app.modules.ts.

Import And Route Configuration

The CLI adds each feature module to the route map at the application level. In the app-routing.module.ts file, update the route array:

src/app/app-routing.module.ts  

content_copyconst routes: Routes = [  

  {  

    path: 'customers',  

    loadChildren: () => import('./customers/customers.module').then(m => m.CustomersModule)  

  },  

  {  

    path: 'orders',  

    loadChildren: () => import('./orders/orders.module').then(m => m.OrdersModule)  

  },  

  {  

    path: '',  

    redirectTo: '',  

    pathMatch: 'full'  

  }  

];

The first two routes to the Customer module and Order module.

If you are using CLI and then the following steps mentioned in the page.

src/app/customers/customers.module.ts  

import { CommonModule } from '@angular/common';  

import { CustomersRoutingModule } from './customers-routing.module';  

content_copyimport { NgModule } from '@angular/core';  

import { CustomersComponent } from './customers.component';  

@NgModule({  

  imports: [  

    CommonModule,  

    customersRoutingModule  

  ],  

  declarations: [customersComponent]  

})  

export class customersModule { }

The Customers.module.ts file imports the customers-routing.module.ts and Customers.component.ts files. The CustomersRoutingModule is listed in the @NgModule import array and gives the Customers module access to its routing module. The CustomersComponent declaration is in the Array, which means that the CustomersComponent belongs to the CustomersModule.

The feature-specific route definition file imports its feature components defined in the customer-routing.module.ts file and in the JavaScript import statements.

src/app/customers/customers-routing.module.ts  

content_copyimport { NgModule } from '@angular/core';  

import { Routes, RouterModule } from '@angular/router';  

  

import { CustomersComponent } from './customers.component';  

  

  

const routes: Routes = [  

  {  

    path: '',  

    component: CustomersComponent  

  }  

];  

  

@NgModule({  

  imports: [RouterModule.forChild(routes)],  

  exports: [RouterModule]  

})  

export class CustomersRoutingModule { }

The routing module of other feature modules is configured similarly.

src/app/orders/orders-routing.module.ts (excerpt)  

content_copyimport { OrdersComponent } from './orders.component';  

  

const routes: Routes = [  

  {  

    path: '',  

    component: OrdersComponent  

  }  

];

forRoot() and forChild()

You can see that the CLI adds RouterModule.forRoot(routes) to the AppRoutingModule import array.

It specifies that Angular AppRoutingModule is a routing module and forRoot() specifies a routing module. It configures all the routes you pass, gives you access to the router’s instructions, and registers the router service.

The CLI also adds RouterModule.forChild to feature routing modules. The forRoot() method takes care of the router’s global injector configuration. There is no injector configuration in its forChild() method. It uses instructions like Router Outlet and RouterLink.

Preloading

The Preloading improves UX by loading parts of the application in the background. You can preload module or component data.

Preloading Module

Preloading modules improve UX by loading parts of your application in the background, so users don’t wait for elements to download when they activate root.

To enable preloading of the lazy loaded modules, import the PreloadAllModules token from the router.

AppRoutingModule (excerpt)  

content_copyimport { PreloadAllModules } from '@angular/router';

In the AppRoutingModule, specify the preloading strategy in forRoot().

AppRoutingModule (excerpt)  

content_copyRouterModule.forRoot(  

  appRoutes,  

  {  

    preloadingStrategy: PreloadAllModules  

  }  

)

Preloading Component Data

You can use a resolver to preload component data. The resolvers improve UX by blocking the page until the necessary data can display on the page.

Resolvers

With the CLI, create a resolver service. The command to generate a service is below:

content_copyng generate service

Implement the Resolve interface provided by the @angular/router package in the newly created service:

Resolver service (excerpt)  

content_copyimport { Resolve } from '@angular/router';  

  

...  

  

/* An interface that represents your data model */  

export interface Crisis {  

  id: number;  

  name: string;  

}  

  

export class CrisisDetailResolverService implements Resolve {  

  resolve(route: ActivatedRouteSnapshot, state: RouterStateSnapshot): Observable {  

    // your logic goes here  

  }  

}

Import the resolver into the module’s routing module.

Feature module’s routing module

content_copyimport { CrisisDetailResolverService } from './crisis-detail-resolver.service';

Add a resolve object to the component’s route configuration.

content_copy{  

  path: '/your-path',  

  component: YourComponent,  

  resolve: {  

    crisis: CrisisDetailResolverService  

  }  

}

In the component’s constructor, inject an Activated Route class that represents the current route.

Component’s constructor

content_copyimport { ActivatedRoute } from '@angular/router';  

  

@Component({ ... })  

class YourComponent {  

  constructor(private route: ActivatedRoute) {}  

}

Use the injected instance of the Activated Route class to access data associated with a given route.

Component’s ngOnInit lifecycle hook

content_copyimport { ActivatedRoute } from '@angular/router';  

  

@Component({ ... })  

class YourComponent {  

  constructor(private route: ActivatedRoute) {}  

  

  ngOnInit() {  

    this.route.data  

      .subscribe(data => {  

        const crisis: Crisis = data.crisis;  

        // ...  

      });  

  }  

}

Pros of Lazy Loading

  • Reduces application start-up time.
  • The application consumes less memory due to on-demand loading.
  • Unnecessary requests to the server are avoided.

Cons of Lazy Loading

  • Adding extra lines of code to existing ones to implement lazy load the code a bit.
  • Additional code when integrating with JavaScript
  • User experience can affect. For example, backtracking is not possible if the page structure is not optimal.

Comments

Leave a Reply

Your email address will not be published. Required fields are marked *