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

https://github.com/guillotinaweb/ngx-state-traverser


https://github.com/guillotinaweb/ngx-state-traverser

Last synced: 1 day ago
JSON representation

Awesome Lists containing this project

README

          

# ngx-state-traverser

[![Build Status](https://travis-ci.com/guillotinaweb/ngx-state-traverser.svg?branch=master)](https://travis-ci.com/guillotinaweb/ngx-state-traverser)

ngx-state-traverser brings [angular-traversal](https://github.com/guillotinaweb/angular-traversal) power to (ngrx)[https://ngrx.io/].

## Principle

Most of modern apps uses routing.

Routes are declared locally (like `/posts/{{id}}`), they are mapped to components (`PostComponent`), those components get the necessary information from the route and its parameters (like `/posts/2` is parsed as `posts` + `2` so we deduce the id 2 is a post id) in order to build the appropriate backend endpoint (probably something quite close to `https://my-backend.net/posts/2`). The backend response might contain an attribute like `relatedPost: "/posts/10"`), so in the template we can display a link like ``.

Routes are not flexible (all the posts are supposed to be accessed with the `/posts` prefix), but even more annoyingly, they implies we know in advanced what will be the returned resource for a given link in order to render it properly.

That's the opposite of web. When navigating on the web, any link might lead to any kind of resource (maybe another page? maybe a file? we don't know, and the browser does not need to know in advanced), at the time we click and then load the resource, depending on its type, the browser will behave accordingly.

Traversing replaces routing by implementing the same principle. Instead of mapping routes to components, we map resource types to component, and when we load the resource (whatever it is), we use the appropriate component to render it.

It also allows to map several views for a given type (like the `@@edit` view for a post will use a different component than the default view).

See [angular-traversal](https://github.com/guillotinaweb/angular-traversal) for more details.

ngx-state-traverser stores each traversed context in a the TraverserState.
It contains the current context and a `collection` of all the previously traversed pathes.

## Type mapping

We declare some views for our data type (`view` is the default view name):

```typescript
traverser.addView('view', 'post', PostComponent);
traverser.addView('edit', 'post', EditPostComponent);
traverser.addView('view', 'user', UserProfileComponent);
```

And we connect traversing to our state:
```typescript
this.store.dispatch({ type: TraverserActions.Types.Watch});
```

Then the app is able to expose the following pathes:

- http://localhost:4200/introduction
- http://localhost:4200/introduction/@@edit
- http://localhost:4200/2019/07/new-post
- http://localhost:4200/eric

(assuming the backend exposes /introduction, 2019/07/new-post, and /eric as posts and user profile)

The `` will load the current context according the current path, and instanciate the component corresponding to the context type (and view).

## Navigate

Any template might contain some links like:

```html



```

Note: here and later on, the provided link can be either a full path (`/2019/07/new-post`), a relative path (`../eric`), or a backend URL (`https://my-backend.net/introduction`).

In code, we can navigate by dispatching an action:

```typescript
this.store.dispatch(new TraverserActions.Traverse('../eric'))
```

## Get the current context

The component gets the `context` from the state.

We can be the raw context (`any`) using the `getContext` selector:

```typescript
context = this.store.pipe(select(TraverserSelectors.getContext));
```

But the `TraverserContext` function allows to get a typed context:

```typescript
context = TraverserSelectors.TraverserContext(this.store);
```

## State-first resolver

Our angular-traversal resolver is triggered everytime we traverse to a given path.
A typical implementation would just make the corresponding backend call to get the object.

As ngx-state-traverser stores all the traversed objects in TraverserState, we could use it as a cache ervider.

To do so, we just need to put the `@StateResolver` decorator on our `resolve` method. Example:

```ts
@StateResolver({
maxAge: 60 * 1000,
})
resolve(path: string, view: string, queryString?: string): Observable {
const headers = new HttpHeaders()
.append('Accept', 'application/json')
.append('Content-Type', 'application/json');
return this.http.get(this.backend + path, { headers });
}
```

This decorator will return the requested object from TraverserState (if iterists).

`maxAge` parameter allows to force backedn call if the stored object is older than `maxAge`. It is optional, if not defined, stored objects are systematically used.

Important: `@StateResolver` will only work if we inject a TraverserState store (named `store`) in our `Resolver` service. The `StateFirst` interface will make sure it is the case (er will require `store` to be public).

## Accessing other resources

Sometimes the current context does not contain all the information we need.

Let's say we need the list of all the posts from the current month when displaying a given post.

We can get the folder content from the state:

```typescript
folder = TraverserSelectors.TraverseTo(this.store, '..');
```

Assuming the current context path is `/2019/07/10`, it retrieves the content of `/2019/07`.

All the traversed resources are stored in the state, but if the requested resource is not yet in the state, the `TraverserResource()` function will load it from backend.

As the state acts as a cache system, it can be cleaned on demand:

```typescript
this.store.dispatch(new TraverserActions.CleanTraverserResources(['../eric', '/2019/*']))
```

And it can be updated:

```typescript
this.store.dispatch(new TraverserActions.UpdateTraverserResource({
path: '/2019/07/03',
changes: {
title: "New title"
}
}));
```

## Tiles

`angular-traversal` allows to manage the main page view.

Since version 1.3.0, it also allows to define small blocks within the current page (named "tiles").

See the [angular-traversal documentation](https://github.com/guillotinaweb/angular-traversal#tiles) for more details.

We can load a context in a tile by doing:

```typescript
this.store.dispatch(new TraverserActions.LoadTile({tile: 'details', path}));
```

and remove it by doing:

```typescript
this.store.dispatch(new TraverserActions.EmptyTile({tile: 'details'}));
```

## NgRX state

Due to a bug in NgRX 9.0, we need to set `strictStateImmutability` and `strictActionImmutability` to false.

```typescript
StoreModule.forRoot({}, { runtimeChecks: {
strictStateImmutability: false,
strictActionImmutability: false,
}}),
```