Mapped Sub States

NGXS provides the ability to merge multiple dynamic selectors into one.

Let's look at the code below:

interface Animal {
  type: string;
  age: string;
  name: string;
}

@State<Animal[]>({
  name: 'animals',
  defaults: [
    { type: 'zebra', age: 'old', name: 'Ponny' },
    { type: 'panda', age: 'young', name: 'Jimmy' }
  ]
})
export class ZooState {
  static pandas(age: string) {
    return createSelector(
      [ZooState],
      (state: Animal[]) => {
        return state.filter(animal => animal.type === 'panda' && animal.age === age);
      }
    );
  }

  static zebras(age: string) {
    return createSelector(
      [ZooState],
      (state: Animal[]) => {
        return state.filter(animal => animal.type === 'zebra' && animal.age === age);
      }
    );
  }

  static pandasAndZebras(age: string) {
    return createSelector(
      [ZooState.pandas(age), ZooState.zebras(age)],
      (pandas: Animal[], zebras: Animal[]) => {
        return [pandas, zebras];
      }
    );
  }
}

This construct will merge 2 dynamic selectors and memoize the result.

Another example could be multiple Zoos in our application:

interface Animal {
  type: string;
  age: string;
  name: string;
}

interface ZooStateModel {
  [id: string]: {
    animals: Animal[];
    ageFilter: string;
  };
}

@State<ZooStateModel>({
  name: 'animals',
  defaults: {
    zoo1: {
      ageFilter: 'young',
      animals: [
        { type: 'zebra', age: 'old', name: 'Ponny' },
        { type: 'panda', age: 'young', name: 'Jimmy' }
      ]
    }
  }
})
export class ZooState {
  static getZooAnimals(zooName: string) {
    return createSelector(
      [ZooState],
      (state: ZooStateModel[]) => state[zooName].animals
    );
  }

  static pandas(zooName: string) {
    return createSelector(
      [ZooState.getAnimals(zooName)],
      (state: Animal[]) => {
        return state.filter(animal => animal.type === 'panda' && animal.age === 'young');
      }
    );
  }

  static pandasWithoutMemoize(zooName: string) {
    return createSelector(
      [ZooState],
      (state: ZooStateModel) => {
        return state[zooName].animals.filter(
          animal => animal.type === 'panda' && animal.age === 'young'
        );
      }
    );
  }
}

In that example merging is required to avoid unnecessary store events. When we subscribe to Zoo.pandasWithoutMemoize store will dispatch event whenever ZooState will change (even ZooState.ageFilter), but when subscribing to Zoo.pandas store will dispatch event only if result has been changed.

Last updated