Skip to content

Overview

Signals are not new and have been around for a long time. They are also known as signals or observables.

Many popular JavaScript frameworks now include signals as part of their core library. Each of the implementations have their own unique features and APIs. Signals.dart is a port of the Preact signals library and is designed to be as close to the original API as possible in the core API.

Signals in preact started off by being implemented with dependencies tracked using a set but was later changed to use a linked list. The linked list implementation is more performant by taking advantage of signal boosting and is the implementation used in Signals.dart.

There is also a DartPad playground with some of the core methods that you can use to experiment!

Minimal Updates

An advantage with signals is the computation you get to save. If you never read a signal it never gets computed. That means that if you have a chain of computed values and never read the value of the last one then none of the callbacks would be called.

import 'package:signals/signals.dart';
final a = signal(0);
final b = computed(() => a.value + 1);
final c = computed(() => b.value + 1);
final d = computed(() => c.value + 1);
// if you never read `d` then none of the callbacks will be called
// All the callbacks will be called
print(d.value); // 3
// None of the callbacks will be called because the
// value is cached at each node
print(d.value); // 3

Signals also are a pull based state management library unlike most push based systems. This means that just because you update a signal value it does not mean that it will propagate (i.e. notifyListeners) to its targets.

Computed is also a special signal that keeps track of its dependencies and caches its value so it will only recompute on read and when the dependencies change. This can be pretty extensive and you can have a chain of computed signals and each of them are optimizing for the minimal amount of updates.

import 'package:signals/signals.dart';
final a = signal(0);
final b = signal(0);
final c = computed(() => a.value + b.value);
final d = computed(() => c.value + 1);
final e = computed(() => d.value + 1);
// All the callbacks will be called
print(e.value); // 2
// None of the callbacks will be called because the
// value is cached at each node
print(e.value); // 2
// Only the callbacks that need to be updated
// will be called
b.value = 1;
print(e.value); // 3

Further reading