arrow left
Back to Developer Education

How to Lazy Load Components in Angular 12

How to Lazy Load Components in Angular 12

Building large applications has a significant impact on the application route loading time. As the application grows, its performance on the browser reduces significantly and lowers user experience. <!--more--> In this tutorial, I'll walk you through the concepts of lazy loading in Angular and how it can improve the application loading time on the browser.

Table of contents

Prerequisites

To follow this tutorial along, you need the following;

  • Have Angular CLI locally installed.
  • Some basic knowledge of Angular2+.
  • Basic knowledge of JavaScript/TypeScript.

Objectives

This tutorial aims to teach you everything you need to know about lazy loading, a concept that will change how you approach your coding techniques for extensive applications in Angular.

What is lazy loading?

Lazy loading is a technique used in programming to improve the performance of applications, especially web pages, and reduce their load time. It causes pages to load much faster initially, and the additional files are downloaded when needed by the user.

It's an asynchronous way of loading components in an Angular application. The needed components are loaded to the browser, while those whose routes have not been visited remain unloaded.

The most important reason lazy loading is used is to improve the application's general loading time. Therefore, it's essential for use in large applications such as e-commerce applications with extensive listings of products.

It's also vital that components are loaded once, so when a user visits a specific lazy loaded route, the component will be readily available when a user visits next time, hence the concept of Single Page Application (SPA).

Setting up the project

Now that we have gotten a better idea of why we need to lazy load our applications, we can now proceed and set up our project by running the following commands:

ng new lazyLoadingExample

The command above, ng new lazyLoadingExample installs our Angular application. It's important to remember that these commands are only made available for use when you have installed the Angular CLI (globally/to a specific directory).

Now that our application is ready let's move forward and set up our government project with different departments.

Our project will have three modules, the AuthModule, GovernmentModule, and the GovernmentFormsModule. The main objective is to load these three modules at different times depending on the route activated by the user.

Let's set up our three modules by running the following commands:

 cd lazyLoadingExample
 ng g module auth --routing
 ng g module government --routing
 ng g module government-forms --routing

The commands above set up three previously discussed modules, each with a routing module to set up our routes.

Next, add two components to the auth directory as shown below:

ng g c auth/register
ng g c auth/login

When the above commands are executed, RegisterComponent and LoginComponent components are added to the AuthModule.

Let's follow the above steps and add components for the government modules as shown below:

# add ministries and governor office components
ng g c government/ministry
ng g c government/governor
#add forms to book and contact government officials
ng g c governor-forms/contact-official
ng g c governor-forms/book-appointment

Setting up shared components

In the previous section, we set up our project components and modules. We also added the routing modules, which we will edit to help us implement lazy loading.

In this section, we build a reusable component layout to help us share the standard features of the application.

In most cases, you'll need to set up these routes within a given layout. For example, login/register forms may only have inputs without the navbar or footer, unlike the other components.

Let's proceed and setup two layout components within a shared module as shown below:

First, create a shared module within the app directory:

ng g module shared

Next, run the following commands to add layout components:

ng g c shared/main-layout
ng g c shared/auth-layout

Now that we have our layout module and components, edit the main layout component as shown below:

<!--/app/shared/MainLayoutComponent  -->
<div class="top-row">
  <div class="top-bar-content">
    <div class="row">
      <!--col-1--->
      <div class="col-md-4">
        <div class="logo-top">
          <a class="text-decoration-none" [routerLink]="['/']">
          <img src="assets/images/top-logo.png" alt=""></a>
        </div>
      </div>
      <!--col-2--->
      <div class="col-md-4">
        <div class="social-top text-center"> <a href="#"><i class="fab fa-facebook-f"></i></a>
          <a href="#"><i class="fab fa-twitter"></i></a>
          <a href="#"><i class="fab fa-instagram"></i></a>
        </div>
      </div>
      <!--col-3--->

      <div class="col-md-4 text-right">
        <div class="top-right-bar text-end">
          <a *ngIf="currentUser!=null" class="btntop">Logged in as {{currentUser.name}}</a>
          <a *ngIf="!currentUser" class="btntop " [routerLink]="['/auth/login']">Sign In</a>
          <a *ngIf="!currentUser" class="btntop" [routerLink]="['/auth/registe']">Register</a>
          <button *ngIf="currentUser!=null" class="btntop btn btn-link" (click)="logout()">Sign Out</button>
        </div>
      </div>
    </div>
  </div>
</div>

<div class="middle-header">
  <div class="top-bar-content">
    <div class="row">
      <div class="col-md-4"></div>
      <div class="col-md-4 text-center">
        <div class="logo">
          <a class="" routerLink="/"><img src="assets/images/update-logo.png" width="250" height="56" alt="logo"></a>
        </div>
      </div>
      <div class="col-md-4 text-right">
        <div class="text-end bar-middle-right">
          <ul>
            <li><a href="#"><i class="far fa-bell"></i></a>
            </li>
            <li><a href="#"><i class="fas fa-search"></i></a>
            </li>
            <li><a class="toggle-bar" id="topbarToggle" href="#"><img src="assets/images/menu-icon.png" alt=""></a>
            </li>
          </ul>
        </div>
      </div>
    </div>
  </div>
</div>

<div class="marquee">
  <marquee class="text-white p-2 font-weight-bolder">
   some texts goes here....
  </marquee>
</div>

<router-outlet></router-outlet>

<footer>
  <div class="footer-contents">
    <div class="row">
      <div class="col-md-2 content footer-logos">
        <a class="btn btn-link" [routerLink]="['/']">
          <img class="logos" src="assets/images/new-logo.png" alt="logo">
          <img class="social" src="assets/images/logo-underline.png" alt="logo">
        </a>
        <div class="social-media-links">
          <ul class="list-group list-group-horizontal">
            <li class="list-group-item bg-transparent border-0"><img src="assets/images/facebook.png" alt="fb"> </li>
            <li class="list-group-item bg-transparent border-0"><img src="assets/images/twitter.png" alt="fb"> </li>
            <li class="list-group-item bg-transparent border-0"><img src="assets/images/instagram.png" alt="fb"> </li>
          </ul>
        </div>
      </div>
      <div class="col-md-5 content">
        <h3 class="footer-link quick-links text-center">Quick Links</h3>
        <div class="row">
          <div class="col-md-6">
            <ul class="list-unstyled"></ul>
              <li class="list-item"><a [routerLink]="['Link']" class="text-decoration-none">Link2</a></li>
              <li class="list-item"><a [routerLink]="['/Link']" class="text-decoration-none">Link2</a></li>
              <li class="list-item"><a [routerLink]="['Link']" class="text-decoration-none">Link2</a></li>
              <li class="list-item"><a [routerLink]="['Link']" class="text-decoration-none">Link2</a></li>
              <li class="list-item"><a [routerLink]="['Link']" class="text-decoration-none">Link2</a></li>
            </ul>
          </div>
        </div>
      </div>
    
      <div class="col-md-2 content">
        <h3 class="footer-link contact-us">Contact Us</h3>
        <div class="contact-us-lists">
          <h4 class="call-us">Call Now</h4>
          <ul class="list-unstyled">
            <li class="list-item"><a class="text-decoration-none">+284xxxx</a></li>
            <li class="list-item"><a class="text-decoration-none">+264xxxx</a></li>
          </ul>
          <h4 class="call-us">Email Address</h4>
          <ul class="list-unstyled">
            <li class="list-item"><a class="text-decoration-none">[email protected]</a></li>
            <li class="list-item"><a class="text-decoration-none">[email protected]</a></li>
          </ul>
        </div>
      </div>
    </div>
    <div class="divider text-center">
      <hr/>
    </div>
    <div class="row footer-contents-bottom">
      <div class="col-md-6 copyright">
        <h3>COPYRIGHT 2021, ALL RIGHTS RESERVED</h3>
      </div>
      <div class="col-md-6 banner">
        <h3>Some footer</h3>
      </div>
    </div>
  </div>
</footer>

The template above has 3 main parts, the header, main body and the footer. The header section has the authentication logic which checks if the current visiting user is authenticated or not.

<a *ngIf="currentUser!=null" class="btntop">Logged in as {{currentUser.name}}</a>

The condition above checks if the currentUser object is null or has a value. It's important to note that this currentUser object is stored in the browser's local storage.

We then have the sign out button which is only made visible to the visiting user only if they are already authenticated as shown in the following conditions:

<button *ngIf="currentUser!=null" class="btntop btn btn-link" (click)="logout()">Sign Out</button>

Next after the header we have the <router-outlet></router-outlet> element. The router-outlet is a directive that's available from the @angular/router package and is used by the router to mark wherein a template, a matched component should be inserted.

We've only defined basic information about our application on the footer, which you customize to meet your requirements.

Next, define your auth layout following the concepts above!

Setting up lazy routes

Now that we have layouts, let's proceed and set up the authentication routes as shown below (this section assumes you set up the auth layout in the previous area).

...
const routes: Routes = [
  {
    path: '',
    component: AuthLayoutComponent,
    children: [
      {
        path: 'register',
        component: RegisterComponent,
      },
      {
        path: 'login',
        component: LoginComponent,
      },
    ]
  }
];

@NgModule({
  ...
})
export class AuthRoutingModule{ }

The module above setup has two routes, register and login, each with their respective components. These routes are defined as children of the AuthLayoutComponent component we had created earlier.

Next, import this AuthRoutingModule module into the main module as shown below:

...
const routes: Routes = [
  
  {
    path:'auth',
    loadChildren: () => import('./auth/auth.module').then(mod=>mod.AuthModule)
  },
];

@NgModule({
  ...
})
...

The above module defines the path, in this case, auth, and now instead of setting up the component, we load the AuthModule, which already has the components.

Resultant full paths for authentications are:

/auth/register
/auth/login

You'll notice that the path combines the main route auth and that of the children components register and login.

The concepts remain the same, and you can follow the above procedure to set up other lazy loading routes for the government and the government forms.

Conclusion

In this tutorial, we've discussed the concept of Angular lazy loading. You've seen how we can use this concept to only load components whose routes have been activated by the user.

Components must not be referenced anywhere else except its routing module. Therefore, we need to remove the component's reference from the app-module.ts file; otherwise, it will be eagerly loaded. All child routes must be stacked together and have the same route prefix.

Lazy loading is a great way to make your SPA lean and mean faster load times with a few minor changes. These enhance the user experience, and will make all the difference for your end-users!

Happy coding!


Peer Review Contributions by: Miller Juma

Published on: Nov 19, 2021
Updated on: Jul 15, 2024
CTA

Cloudzilla is FREE for React and Node.js projects

Deploy GitHub projects across every major cloud in under 3 minutes. No credit card required.
Get Started for Free