Angular Signals
To update values in a component angular use Zonejs to detect changes in the component and update the view. But zonejs is not always the best solution, for example when you have a lot of data to update in a component, it can slow down the application.
This is why since angular 16 we have the option to use signals to update the view.
Signal
With signal we can now explicitly tell angular to update the view when we want to.
import { signal } from '@angular/core';
@Component({
selector: 'app-signals',
templateUrl: './signals.component.html',
styleUrls: ['./signals.component.css']
})
export class SignalsComponent {
counter = signal<number>(0);
increment() {
this.counter.update(value => value + 1);
}
decrement() {
this.counter.update(value => value - 1);
}
}
And so the view will be updated only when we call the update
method.
<p>Counter: {{ counter() }}</p>
<button (click)="increment()">Increment</button>
<button (click)="decrement()">Decrement</button>
Set/Update/Mutate
We can also use the set
, update
and mutate
methods to update the value of the signal.
- Set: Set the value of the signal
- Update: Update the value of the signal
.update(oldValue => oldValue + 1)
- Mutate: Mutate has been removed since angular 17 To update value on mutable objects like arrays or objects
import { signal } from '@angular/core';
@Component({
selector: 'app-signals',
templateUrl: './signals.component.html',
styleUrls: ['./signals.component.css']
})
export class SignalsComponent {
counter = signal<number>(0);
itemas = signal<string[]>([]);
increment() {
// this.counter.set(counter() + 1); <= you can also use counter() to get the value
// You can do it anywhere in the component no only in the template
this.counter.update(value => value + 1);
// for example
this.items.mutate(items => [...items, 'item']).push();
}
decrement() {
this.counter.update(value => value - 1);
}
reset() {
this.counter.set(0);
}
}
Computed
We can also use computed signals to compute values from other signals.
import { signal } from '@angular/core';
@Component({
selector: 'app-signals',
templateUrl: './signals.component.html',
styleUrls: ['./signals.component.css']
})
export class SignalsComponent {
counter = signal<number>(0);
double = computed(() => this.counter() * 2);
increment() {
this.counter.update(value => value + 1);
}
decrement() {
this.counter.update(value => value - 1);
}
}
<p>Counter: {{ counter() }}</p>
<p>Double: {{ double() }}</p>
<button (click)="increment()">Increment</button>
<button (click)="decrement()">Decrement</button>
Angular will memoize the computed value and only update the view when the value changes.
Effect
We can also use effects to run side effects when a signal changes. This is used to run code when signals change.
import { signal, effect } from '@angular/core';
@Component({
selector: 'app-signals',
templateUrl: './signals.component.html',
styleUrls: ['./signals.component.css']
})
export class SignalsComponent {
counter = signal<number>(0);
double = computed(() => this.counter() * 2);
increment() {
this.counter.update(value => value + 1);
}
decrement() {
this.counter.update(value => value - 1);
}
constructor() {
effect(() => {
console.log('Counter changed:', this.counter());
});
}
}
Input Signals
We can also use input signals to pass signals to child components.
import { signal } from '@angular/core';
@Component({
selector: 'app-signals',
templateUrl: './signals.component.html',
styleUrls: ['./signals.component.css']
})
export class SignalsComponent {
name = input<number>(0); // <= input signal
lastName = input.required<string>(''); // <= required input signal
age = input(0, { alias: 'userAge' }); // <= input signal with alias, default name is the variable name
// This allows users to bind to your input using [userAge], while inside your component you can access the input values using this.age.
}
Last words
Since angular 18 we should now use input signals instead of @Input to pass data to child components. As mentionned Here