Angular — Facade Design Pattern and How It Can Improve Performance

What Facade Design Pattern is — and how it improves the performance of your Angular application.

V — D
Bits and Pieces

--

Introduction

As an Angular developer, you may have come across the need to simplify complex code and provide a simplified interface to other parts of the application. This is where the Facade Design Pattern comes into play. In this blog post, we will discuss the Facade Design Pattern, its implementation in Angular, and how it can improve the performance of your application.

What is the Facade Design Pattern?

The Facade Design Pattern is a structural design pattern that provides a simplified interface to a complex system of classes, functions, and APIs. It allows you to encapsulate a group of complex subsystems and expose a simplified interface to the client code.

In simple terms, a Facade is like a wrapper around a complex system that provides a simple interface for clients to interact with. The Facade hides the complexity of the system and exposes a simple API for clients to use.

Implementation in Angular

In Angular, we can implement the Facade Design Pattern using Services. Services are the backbone of Angular applications and are used to share data, logic, and functionality across multiple components. Services can also be used to encapsulate complex subsystems and provide a simplified interface to the client code.

Let’s take a look at an example of how we can implement the Facade Design Pattern using Services in Angular.

Example 1

Suppose we have a complex system of classes and functions that are responsible for handling authentication in our Angular application. The authentication system consists of the following classes and functions:

  • AuthService: Responsible for handling user authentication
  • UserService: Responsible for managing user data
  • TokenService: Responsible for managing user tokens
  • JwtHelperService: Responsible for decoding JWT tokens

Our application has multiple components that require access to these classes and functions. To simplify the code and provide a simplified interface to the client code, we can create a Facade Service called AuthFacadeService.

The AuthFacadeService will encapsulate the complex authentication subsystem and provide a simplified interface to the client code. The AuthFacadeService will have the following methods:

  • login(): Handles user login
  • logout(): Handles user logout
  • getUser(): Retrieves user data
  • isAuthenticated(): Checks if the user is authenticated

Here’s how the AuthFacadeService can be implemented:

import { Injectable } from '@angular/core';
import { AuthService } from './auth.service';
import { UserService } from './user.service';
import { TokenService } from './token.service';
import { JwtHelperService } from './jwt-helper.service';

@Injectable({
providedIn: 'root'
})
export class AuthFacadeService {

constructor(
private authService: AuthService,
private userService: UserService,
private tokenService: TokenService,
private jwtHelperService: JwtHelperService
) { }

login(username: string, password: string) {
// Call AuthService to handle user authentication
const token = this.authService.login(username, password);

// Save token to TokenService
this.tokenService.saveToken(token);
}

logout() {
// Call AuthService to handle user logout
this.authService.logout();

// Remove token from TokenService
this.tokenService.removeToken();
}

getUser() {
// Call UserService to retrieve user data
return this.userService.getUser();
}

isAuthenticated(): boolean {
// Check if the user is authenticated using the JWT token
const token = this.tokenService.getToken();
return token && !this.jwtHelperService.isTokenExpired(token);
}
}

In the above code, we have created a Facade Service called AuthFacadeService that encapsulates the complex authentication subsystem. The AuthFacadeService provides a simplified interface to the client code with methods like login(), logout(), and getUser(). These methods internally use the AuthService to perform the necessary operations.

Example 2

Let’s take an example of an e-commerce application that has multiple services for handling different parts of the application, such as authentication, shopping cart, and payment. These services are tightly coupled and dependent on each other, making it difficult to maintain and improve performance. To solve this problem, we can create a facade service that exposes a simplified interface to these services.

First, let’s create the services that we want to simplify:

@Injectable()
export class AuthService {
// Authentication logic
}

@Injectable()
export class CartService {
// Shopping cart logic
}

@Injectable()
export class PaymentService {
// Payment logic
}

Now, let’s create the facade service that exposes a simplified interface to these services:

@Injectable()
export class FacadeService {
constructor(
private authService: AuthService,
private cartService: CartService,
private paymentService: PaymentService
) {}

// Simplified interface to authentication
login(username: string, password: string) {
return this.authService.login(username, password);
}

// Simplified interface to shopping cart
addToCart(item: Item) {
return this.cartService.addToCart(item);
}

removeFromCart(item: Item) {
return this.cartService.removeFromCart(item);
}

// Simplified interface to payment
pay() {
return this.paymentService.pay();
}
}

As you can see, the FacadeService provides a simplified interface to the complex system of services. Instead of having to interact with each service separately, the client can interact with the FacadeService and perform all the necessary actions.

Now, let’s take a look at how we can use this FacadeService in our components:

@Component({
selector: 'app-login',
template: `
<form (ngSubmit)="onSubmit()">
<input type="text" [(ngModel)]="username">
<input type="password" [(ngModel)]="password">
<button type="submit">Login</button>
</form>
`
})
export class LoginComponent {
username: string;
password: string;

constructor(private facadeService: FacadeService) {}

onSubmit() {
this.facadeService.login(this.username, this.password)
.subscribe(() => {
// Redirect to dashboard
}, (error) => {
// Display error message
});
}
}

@Component({
selector: 'app-cart',
template: `
<div *ngFor="let item of cartItems">
{{ item.name }}
<button (cllick)="addToCart(item)">Add to cart</button>
<div>
`
export class CartComponent {
username: string;
password: string;

constructor(private facadeService: FacadeService) {}

addToCart(item: Item) {
this.facadeService.addToCart(item)
.subscribe(() => {
// Add to cart
}, (error) => {
// Display error message
});
}
}

In the above code, we can clearly see that facade service is taking care of all the functionality related to login, adding to cart or remove from cart etc. By creating a simple interface that hides the complexity of the underlying system, we can improve the readability and testability of our code.

I hope this article has provided you with a good understanding of the Facade pattern and how to use it in Angular applications.

Overall, the facade design pattern is a powerful tool for improving the performance of your Angular applications by simplifying complex operations. By abstracting away the complexity of the underlying systems, the facade design pattern can make it easier for developers to write clean, readable code that is easier to maintain and update over time.

💡 You could consider isolating your Facade code into its own module, then use Bit to reuse the factory logic across different components and applications. With Bit you can easily version, test, and publish — and then install, import, and use your Facade in other parts of your application.

Learn more here:

If you’re looking to improve the performance of your Angular applications, I highly recommend giving the facade design pattern a try. By simplifying complex operations and creating a clean, easy-to-use interface for client code, you can create more efficient, effective applications that deliver better results for your users.

By using the examples and implementation details outlined in this article, you can start implementing the Facade pattern in your Angular applications today.

Thanks for reading

Build Angular Apps with reusable components, just like Lego

Bit’s open-source tool help 250,000+ devs to build apps with components.

Turn any UI, feature, or page into a reusable component — and share it across your applications. It’s easier to collaborate and build faster.

Learn more

Split apps into components to make app development easier, and enjoy the best experience for the workflows you want:

Micro-Frontends

Design System

Code-Sharing and reuse

Monorepo

Learn more:

--

--

Enthusiastic JS developer, Angular, React, Redux, functional programming, NodeJS, Crypto, Block-chain