Redux Selectors - Computed State

Computed state properties are functions on stored state which provide alternate, filtered, or aggregate views. They are useful because they allow the amount of data stored in state to be kept to a minimum.

The redux library angular-redux/store does not explicitly implement computed state in the same way as other libraries, but achieves it via @select() and @select$() decorators (ref: The Select Pattern).

The problem is, the code for these selectors is not centralized. Examples given in docs show each component or service implementing the computation code. However, it would be preferable to defined the logic once on the store.

For example, the visibleFiles property is used on both the page component and the search component, so it would be useful to define the calculation in one place.

This can be done by creating a Computed class and injecting it into components and services as needed. Note that some 'properties' are implemented as methods since we need to pass in the page parameter. The result is a higher-order function similar to the pattern for Middleware.

This is possible, since a Selector can be a function - from the source

export type FunctionSelector<RootState, S> = ((s: RootState) => S);
export type Selector<RootState, S> =
  | PropertySelector
  | PathSelector
  | FunctionSelector<RootState, S>;

Computed class

@Injectable()
export class Computed {

  constructor(private ngRedux: NgRedux<IAppState>) {}

  visibleFiles$(page): Observable<IFileInfo[]> {
    return this.ngRedux.select<IFileInfo[]>(this.visibleFiles(page));
  }

  private visibleFiles = (page) => (state) => {
    const files = state.pages[page].files || [];
    const numVisible = state.pages[page].numVisible || 0;
    return files.slice(0, numVisible);
  }
  ...
}

Usage

  private getResults(searchTerm) {
    this.computed.visibleFiles$(this.page)
      .take(1)
      .map(files => ...
  }

Last Updated: 9/3/2018, 8:47:06 PM