Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(signals): add signalState and selectSignal APIs #4007

Merged
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
15 changes: 15 additions & 0 deletions modules/signals/spec/helpers.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
import { Component } from '@angular/core';
import { TestBed } from '@angular/core/testing';

export function testEffects(testFn: (tick: () => void) => void): () => void {
@Component({ template: '', standalone: true })
class TestComponent {}

return () => {
const fixture = TestBed.configureTestingModule({
imports: [TestComponent],
}).createComponent(TestComponent);

TestBed.runInInjectionContext(() => testFn(() => fixture.detectChanges()));
};
}
5 changes: 0 additions & 5 deletions modules/signals/spec/index.spec.ts

This file was deleted.

208 changes: 208 additions & 0 deletions modules/signals/spec/select-signal.spec.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,208 @@
import { effect, isSignal, signal } from '@angular/core';
import { selectSignal } from '../src';
import { testEffects } from './helpers';

describe('selectSignal', () => {
timdeschryver marked this conversation as resolved.
Show resolved Hide resolved
it('creates a signal from provided projector function', () => {
const s1 = signal(1);
const s2 = selectSignal(() => s1() + 1);

expect(isSignal(s2)).toBe(true);
expect(s2()).toBe(2);

s1.set(2);

expect(s2()).toBe(3);
});

it('creates a signal from provided signals dictionary', () => {
const s1 = signal(1);
const s2 = signal(2);
const s3 = selectSignal({ s1, s2 });

expect(isSignal(s3)).toBe(true);
expect(s3()).toEqual({ s1: 1, s2: 2 });

s1.set(10);

expect(s3()).toEqual({ s1: 10, s2: 2 });

s1.set(100);
s2.set(20);

expect(s3()).toEqual({ s1: 100, s2: 20 });
});

it('creates a signal by combining provided signals', () => {
const s1 = signal(1);
const s2 = signal(2);
const s3 = selectSignal(s1, s2, (v1, v2) => v1 + v2);

expect(isSignal(s3)).toBe(true);
expect(s3()).toBe(3);

s1.set(10);

expect(s3()).toBe(12);

s1.set(100);
s2.set(20);

expect(s3()).toBe(120);
});

it(
'uses default equality function when custom one is not provided',
testEffects((tick) => {
const initialState = { x: { y: { z: 1 }, k: 2 }, l: 3 };
const state = signal(initialState);

const x = selectSignal(() => state().x);
const y = selectSignal(x, (x) => x.y);
const z = selectSignal(y, (y) => y.z);
const k = selectSignal(x, (x) => x.k);
const l = selectSignal(() => state().l);
const zPlusK = selectSignal(z, k, (z, k) => z + k);
const zWithL = selectSignal({ z, l });

let xEmitted = 0;
let yEmitted = 0;
let zEmitted = 0;
let zPlusKEmitted = 0;
let zWithLEmitted = 0;

effect(() => {
x();
xEmitted++;
});

effect(() => {
y();
yEmitted++;
});

effect(() => {
z();
zEmitted++;
});

effect(() => {
zPlusK();
zPlusKEmitted++;
});

effect(() => {
zWithL();
zWithLEmitted++;
});

expect(xEmitted).toBe(0);
expect(yEmitted).toBe(0);
expect(zEmitted).toBe(0);
expect(zPlusKEmitted).toBe(0);
expect(zWithLEmitted).toBe(0);

tick();

expect(xEmitted).toBe(1);
expect(yEmitted).toBe(1);
expect(zEmitted).toBe(1);
expect(zPlusKEmitted).toBe(1);
expect(zWithLEmitted).toBe(1);

state.update((state) => ({ ...state, l: 10 }));
tick();

expect(xEmitted).toBe(1);
expect(yEmitted).toBe(1);
expect(zEmitted).toBe(1);
expect(zPlusKEmitted).toBe(1);
expect(zWithLEmitted).toBe(2);

state.update((state) => ({ ...state, x: { ...state.x, k: 20 } }));
tick();

expect(xEmitted).toBe(2);
expect(yEmitted).toBe(1);
expect(zEmitted).toBe(1);
expect(zPlusKEmitted).toBe(2);
expect(zWithLEmitted).toBe(2);

state.update((state) => ({ ...state, x: { ...state.x, y: { z: 1 } } }));
tick();

expect(xEmitted).toBe(3);
expect(yEmitted).toBe(2);
expect(zEmitted).toBe(1);
expect(zPlusKEmitted).toBe(2);
expect(zWithLEmitted).toBe(2);

state.update((state) => ({ ...state, x: { ...state.x, y: { z: 10 } } }));
tick();

expect(xEmitted).toBe(4);
expect(yEmitted).toBe(3);
expect(zEmitted).toBe(2);
expect(zPlusKEmitted).toBe(3);
expect(zWithLEmitted).toBe(3);
})
);

it(
'uses custom equality function when provided',
testEffects((tick) => {
const state = signal([1, 2, 3]);
const numbers = selectSignal(() => state(), {
equal: (a, b) => a.length === b.length,
});
const first = selectSignal(state, (numbers) => numbers[0], {
equal: (a: number, b: number) => Math.round(a) === Math.round(b),
});

let numbersEmitted = 0;
let firstEmitted = 0;

effect(() => {
numbers();
numbersEmitted++;
});

effect(() => {
first();
firstEmitted++;
});

expect(numbersEmitted).toBe(0);
expect(firstEmitted).toBe(0);

tick();

expect(numbersEmitted).toBe(1);
expect(firstEmitted).toBe(1);

state.set([10, 20, 30]);
tick();

expect(numbersEmitted).toBe(1);
expect(firstEmitted).toBe(2);

state.set([10.1, 20.1, 30.1]);
tick();

expect(numbersEmitted).toBe(1);
expect(firstEmitted).toBe(2);

state.set([10.9, 20.9]);
tick();

expect(numbersEmitted).toBe(2);
expect(firstEmitted).toBe(3);

state.set([10.7, 20.7, 30.7]);
tick();

expect(numbersEmitted).toBe(3);
expect(firstEmitted).toBe(3);
})
);
});
168 changes: 168 additions & 0 deletions modules/signals/spec/signal-state.spec.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,168 @@
import { effect } from '@angular/core';
import { signalState } from '../src';
import { testEffects } from './helpers';

describe('signalState', () => {
const initialState = {
user: {
firstName: 'John',
lastName: 'Smith',
},
foo: 'bar',
numbers: [1, 2, 3],
ngrx: 'signals',
};

describe('$update', () => {
it('updates state via partial state object', () => {
const state = signalState(initialState);

state.$update({
user: { firstName: 'Johannes', lastName: 'Schmidt' },
foo: 'baz',
});

expect(state()).toEqual({
...initialState,
user: { firstName: 'Johannes', lastName: 'Schmidt' },
foo: 'baz',
});
});

it('updates state via updater function', () => {
const state = signalState(initialState);

state.$update((state) => ({
numbers: [...state.numbers, 4],
ngrx: 'rocks',
}));

expect(state()).toEqual({
...initialState,
numbers: [1, 2, 3, 4],
ngrx: 'rocks',
});
});

it('updates state via sequence of partial state objects and updater functions', () => {
const state = signalState(initialState);

state.$update(
{ user: { firstName: 'Johannes', lastName: 'Schmidt' } },
(state) => ({ numbers: [...state.numbers, 4], foo: 'baz' }),
(state) => ({ user: { ...state.user, firstName: 'Jovan' } }),
{ foo: 'foo' }
);

expect(state()).toEqual({
...initialState,
user: { firstName: 'Jovan', lastName: 'Schmidt' },
foo: 'foo',
numbers: [1, 2, 3, 4],
});
});

it('updates state immutably', () => {
const state = signalState(initialState);

state.$update({
foo: 'bar',
numbers: [3, 2, 1],
ngrx: 'rocks',
});

expect(state.user()).toBe(initialState.user);
expect(state.foo()).toBe(initialState.foo);
expect(state.numbers()).not.toBe(initialState.numbers);
expect(state.ngrx()).not.toBe(initialState.ngrx);
});
});

describe('nested signals', () => {
it('creates signals for nested state slices', () => {
const state = signalState(initialState);

expect(state()).toBe(initialState);
expect(state.user()).toBe(initialState.user);
expect(state.user.firstName()).toBe(initialState.user.firstName);
expect(state.foo()).toBe(initialState.foo);
expect(state.numbers()).toBe(initialState.numbers);
expect(state.ngrx()).toBe(initialState.ngrx);
});

it('does not modify props that are not state slices', () => {
const state = signalState(initialState);
(state as any).x = 1;
(state.user as any).x = 2;
(state.user.firstName as any).x = 3;

expect((state as any).x).toBe(1);
expect((state.user as any).x).toBe(2);
expect((state.user.firstName as any).x).toBe(3);

expect((state as any).y).toBe(undefined);
expect((state.user as any).y).toBe(undefined);
expect((state.user.firstName as any).y).toBe(undefined);
});

it(
'emits new values only for affected signals',
testEffects((tick) => {
const state = signalState(initialState);
let numbersEmitted = 0;
let userEmitted = 0;
let firstNameEmitted = 0;

effect(() => {
state.numbers();
numbersEmitted++;
});

effect(() => {
state.user();
userEmitted++;
});

effect(() => {
state.user.firstName();
firstNameEmitted++;
});

expect(numbersEmitted).toBe(0);
expect(userEmitted).toBe(0);
expect(firstNameEmitted).toBe(0);

tick();

expect(numbersEmitted).toBe(1);
expect(userEmitted).toBe(1);
expect(firstNameEmitted).toBe(1);

state.$update({ numbers: [1, 2, 3] });
tick();

expect(numbersEmitted).toBe(2);
expect(userEmitted).toBe(1);
expect(firstNameEmitted).toBe(1);

state.$update((state) => ({
user: { ...state.user, lastName: 'Schmidt' },
}));
tick();

expect(numbersEmitted).toBe(2);
expect(userEmitted).toBe(2);
expect(firstNameEmitted).toBe(1);

state.$update((state) => ({
user: { ...state.user, firstName: 'Johannes' },
}));
tick();

expect(numbersEmitted).toBe(2);
expect(userEmitted).toBe(3);
expect(firstNameEmitted).toBe(2);
})
);
});
});
Loading