TypeScript Class Decorators

Class decorators in TypeScript are a special kind of decorator that can be applied to a class declaration. They are used to modify or extend the behavior of the class at design time. A class decorator is a function that is applied to the constructor of the class.

Class decorators are often used in frameworks like Angular to add functionality such as dependency injection or logging, without modifying the original class directly.

 

What is a Class Decorator?

A class decorator is a function that takes the target class constructor as an argument. It can modify the class's prototype, add properties or methods, or even replace the class with another class.

Class decorators are prefixed with the @ symbol, followed by the decorator name, and are applied to the class itself.

 

Enabling Class Decorators in TypeScript

Before using decorators in TypeScript, you must enable the experimentalDecorators option in the tsconfig.json file.

Example: Enabling Class Decorators

{
  "compilerOptions": {
    "experimentalDecorators": true
  }
}
  • The experimentalDecorators option allows the use of decorators, which are not yet fully standardized in JavaScript.

Syntax of Class Decorators

A class decorator is a function that takes the target class constructor as an argument.

function MyClassDecorator(constructor: Function) {
  console.log("Class decorated:", constructor);
}

Here, MyClassDecorator is a decorator function that logs the class constructor to the console. The decorator will be applied to the class when it is instantiated.

 

Applying Class Decorators

To apply a class decorator, simply place the decorator before the class declaration.

Example: Applying a Class Decorator

function LogClass(target: Function) {
  console.log(`Class ${target.name} has been decorated.`);
}

@LogClass
class Person {
  constructor(public name: string, public age: number) {}
}

const person = new Person('John', 25);
// Output: Class Person has been decorated.
  • The LogClass decorator is applied to the Person class. When an instance of Person is created, the decorator logs a message to the console indicating that the class has been decorated.

 

Modifying the Class Behavior

Class decorators can modify the class by adding new methods, properties, or modifying its constructor. Decorators can even replace the class constructor with a new one.

Example: Modifying the Class Constructor

function AddMethod(target: Function) {
  target.prototype.sayHello = function() {
    console.log("Hello from the decorator!");
  };
}

@AddMethod
class Person {
  constructor(public name: string, public age: number) {}
}

const person = new Person('John', 25);
person.sayHello();  // Output: Hello from the decorator!
  • In this example, the AddMethod decorator adds a new method sayHello to the class prototype, making it available on all instances of the Person class.

 

Replacing the Class Constructor

A class decorator can also replace the class constructor with a new one. This can be useful if you want to modify the instantiation logic of a class.

Example: Replacing the Constructor

function ReplaceConstructor(target: Function) {
  const newConstructor = function(this: any) {
    console.log("New constructor called.");
    this.name = "Default Name";
    this.age = 30;
  };
  
  newConstructor.prototype = target.prototype;
  return newConstructor;
}

@ReplaceConstructor
class Person {
  constructor(public name: string, public age: number) {}
}

const person = new Person('John', 25);
console.log(person.name);  // Output: Default Name
console.log(person.age);   // Output: 30
  • Here, the ReplaceConstructor decorator replaces the Person class constructor with a new one that assigns default values to name and age. Even though the Person class is instantiated with arguments, the new constructor is called.

 

Class Decorator with Parameters

You can also pass parameters to class decorators by using a decorator factory. A decorator factory is a function that returns a decorator function.

Example: Class Decorator with Parameters

function Logger(message: string) {
  return function(target: Function) {
    console.log(`${message} - Class ${target.name}`);
  };
}

@Logger("Logging")
class Person {
  constructor(public name: string, public age: number) {}
}

const person = new Person('John', 25);
// Output: Logging - Class Person
  • In this example, the Logger decorator factory allows you to pass a message parameter to customize the log message.

 

Using Class Decorators for Dependency Injection (DI)

In frameworks like Angular, class decorators are used for dependency injection. The @Injectable() decorator, for example, marks a class as available for DI, and the framework automatically resolves and injects the dependencies when the class is instantiated.

Example: Dependency Injection with Class Decorators (Simulated)

function Injectable(target: Function) {
  // Simulating dependency injection by storing the class in a registry
  const container = (window as any).container || (window as any).container = {};
  container[target.name] = target;
}

@Injectable
class Service {
  constructor(public name: string) {}
}

const service = new (window as any).container.Service("My Service");
console.log(service.name);  // Output: My Service
  • The @Injectable decorator marks the Service class as injectable, and we simulate a simple DI container to resolve and instantiate the class with its dependencies.

 

Summary

  • Class decorators in TypeScript are used to modify or extend the behavior of a class.
  • They are prefixed with @ and can modify the class prototype, add methods, or even replace the class constructor.
  • Decorators are enabled by setting experimentalDecorators: true in the tsconfig.json file.
  • Class decorators can be used to add functionality like logging, dependency injection, or even modify the class structure.