Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/akmjenkins/ngx-stateful

A StatefulComponent for Angular
https://github.com/akmjenkins/ngx-stateful

angular component ngx-stateful state state-management stateful stateful-components stateless-co

Last synced: about 1 month ago
JSON representation

A StatefulComponent for Angular

Awesome Lists containing this project

README

        

# ngx-stateful

[![npm version](https://img.shields.io/npm/v/ngx-stateful.svg)](https://npmjs.org/package/ngx-stateful)
[![Coverage Status](https://coveralls.io/repos/github/akmjenkins/ngx-stateful/badge.svg)](https://coveralls.io/github/akmjenkins/ngx-stateful)
[![Build Status](https://travis-ci.com/akmjenkins/ngx-stateful.svg)](https://travis-ci.com/akmjenkins/ngx-stateful)
[![Bundle Phobia](https://badgen.net/bundlephobia/minzip/ngx-stateful)](https://bundlephobia.com/result?p=ngx-stateful)

## Sample App

[Simple Stackblitz App](https://stackblitz.com/edit/angular-ivy-k47m7k) (it's not a "ToDo") to show you how **ngx-stateful** works.

## Why Stateful?

First of all, you probably don't need this package, it's completely trivial to implement yourself using a [StateSubject](https://www.bennadel.com/blog/3522-creating-a-simple-setstate-store-using-an-rxjs-behaviorsubject-in-angular-6-1-10.htm). It looks like [ngrx has tried to do this](https://ngrx.io/guide/component-store) but they've failed to [KISS](https://en.wikipedia.org/wiki/KISS_principle). Plus (unpopular opinion ahead) dependency injection in Angular is not the way to go. Not because DI is inherently bad, but because most developers just don't use it for it's intended purpose. Instead of relying on abstract dependencies and "programming to the interface", it's used simply to share state around various services in the application in weird ways that are hard to debug.

Secondly, just because your application uses state (spoiler alert: every application has state), doesn't mean it needs all the boilerplate of a [redux implementation](https://ngrx.io/guide/store).

After working on an (unncessary) large scale angular application with way too many components with devs who had a variety of skill levels, I realized how unclear the concept of `State` is to a lot of developers, even though they may be using redux to manage their state.

I blame OOP principles (yeah, I said it). When components are backed by instances of classes, often developers tend to put instance variables on their components without fully realizing that, by doing so, **they are causing that component to become a StatefulComponent** (see [StatefulWidget](https://api.flutter.dev/flutter/widgets/StatefulWidget-class.html) if you know anything about [Flutter](https://api.flutter.dev/index.html)).

Aside - this is also the reason why I'm such a big fan of [React](https://reactjs.org) moving to functional components - not because react is better, but just because it tries to force developers to critically think about state before using it (typing [`useState`](https://reactjs.org/docs/hooks-reference.html#usestate) comes with more forethought than `this.someString = 'fred'`, IMO.)

Back to Angular components with instance variables - these components then **own something**, and when state is scattered up and down your component tree (without it being clearly identifiable - does this component class have instance variables? How are they used?) then it just becomes a nightmare to onboard yourself into a large codebase.

So let's start making it obnoxiusly clear to all of your team members when a component in Angular owns state - it must extend `StatefulComponent`, and whenever something about that state changes, it must call `setState`.

It provides some neat helper methods that make it easy to discern when and why state is being updated without looking through your component file for `this.someProp = X` and then checking your template file to see if `someProp` is being bound to something in the template.

## Aren't you just trying to make Angular like React?

**YES**... kind of, but that's because it **is** like React. React, Vue, Angular, and Flutter are all effectively the same - they render a tree of components, some of which own state (instance members? computed properties?) and the tree needs to update (render) when things change. All of these frameworks can be written in the exact same style, and they probably should be, it makes moving between applications written in in various frameworks tremendously easy. Unidirectional data flow and the composable component pattern of building user interface applications have easily won the day, so we, as developers, should embrace it fully rather than trying to invent new state management patterns each time a new framework shows up - looking at you [BLOC](https://medium.com/@aaron.chu/flutter-state-management-bloc-pattern-9cd6011c699).

## How to use

```js
import { StatefulComponent } from 'ngx-stateful';

interface MyState {
fetching: boolean;
data?: string[];
error: false;
}

@Component({
selector:'app',
template:`










`
})
export class MyComponent extends StatefulComponent implements ngOnInit {

public fetching$ = this.selectKey('fetching');
public error$ = this.selectKey('error');
public data$ = this.selectKey('data');

constructor() {
super({
fetching:false,
error:false
})
}

async ngOnInit() {
this.setState({fetching:true});

try {
this.setState({data:await fetchSomeData()});
} catch(err) {
this.stateState({error:true});
}

this.setState({fetching:false});
}

}
```

## API

Any component class that extends this must call `super()` with the state:

```js
class MyComponent extends StatefulComponent {
constructor() {
super({...your state})
}
}
```

- `public state: T`

The current **IMMUTABLE** state of the component.


- `public state$: Observable`

An observable of your components state, think [async pipe](https://angular.io/api/common/AsyncPipe) in your template.


- `public setState(state: StateSetter)`

Sets the state of the component by either patching it or using the function signature.

```js
private startFetching() {
// patch function signature
this.setState({fetching:true});
}

private stopFetching() {
// callback function signature
this.setState(state => ({...state,fetching:false}));
}
```


- `public select(selector: (state: T) => R): Observable`

Retrieve state with a selector (see [reselect](https://github.com/reduxjs/reselect)).

```js
public fetching$ = this.select(state => state,({fetching}) => fetching);
```

- `public selectKey(key: string): Observable`

Retrieve state by string key (when your state is an object).
```js
public fetching$ = this.selectKey('fetching');
```


- `*public onChange(selector: (state: T) => R): Observable<[R,R]>`

Returns on observable that emits only when a certain piece of the state changes.


- `*public onKeyChange(key: string): Observable<[R,R]>`

Same deal as [selectKey](#selectkey)

*Note: `onChange` and `onKeyChange` **DO NOT** emit if the piece of state being selected has never changed. See [pairwise](https://www.learnrxjs.io/learn-rxjs/operators/combination/pairwise).