Ecosyste.ms: Awesome

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

Awesome Lists | Featured Topics | Projects

https://github.com/ArnaudRinquin/redux-reroute

Location reducer and routing helpers for redux.
https://github.com/ArnaudRinquin/redux-reroute

Last synced: 2 months ago
JSON representation

Location reducer and routing helpers for redux.

Awesome Lists containing this project

README

        

# redux-reroute

[![Build Status](https://travis-ci.org/ArnaudRinquin/redux-reroute.svg)](https://travis-ci.org/ArnaudRinquin/redux-reroute)
[![Dependency Status](https://david-dm.org/ArnaudRinquin/redux-reroute.svg)](https://david-dm.org/ArnaudRinquin/redux-reroute)
[![Commitizen friendly](https://img.shields.io/badge/commitizen-friendly-brightgreen.svg)](http://commitizen.github.io/cz-cli/)

Location reducer and routing helpers for [`redux`](https://github.com/rackt/redux).

## Rationale

**Context** We are using an application state based rendering flow (redux)

**Questions**
* How to deal with the location changes?
* Should the URL be the result of the app state? - Or -
* Should the application state be the result of the URL?
* Where do I put my non-location based routing logic (ex: authentication)?
* How to implement deep linking?

**Choices made by `redux-reroute`**

* The application state remains the only source of truth
* The location, or url, is only a part of the application state
* It is up to the developer to build the decision tree resulting in the intended UI
* `redux-reroute` = minimal base + optional helpers
* Based on idiomatic `redux` principles (action > reducer > app state > render loop)
* There is not concept of `ViewContainer`, `Router` or `Route` components
* Determine which component to render by using a component `selector` (`(appState) => Component`, as defined by [`reselect`](https://github.com/rackt/reselect)), anywhere in your application

## Features

1. reduce the location into the application state, providing:
1. matched url pattern
1. url parameters (within the path or query string)
1. optionally use provided helpers to generate your component `selector(s)` (as defined by [`reselect`](https://github.com/rackt/reselect))

## The minimal example ([source](./examples/minimalist/index.js))

In this example, we demonstrate the base principles:

* declare some route patterns (using [`url-pattern`](https://github.com/snd/url-pattern) syntaxe)
* connect `reroute` to the store
* decide which component to show depending on app state
* pick url params from app state
* navigate using regular links

Note: helpers provided by `reroute` remove the need for most of the boilerplate shown in this example.

```js
import React, { Component } from 'react';
import { render } from 'react-dom';
import { Provider, connect } from 'react-redux';
import { createStore, combineReducers } from 'redux';

import { connectToStore, location } from 'redux-reroute';

// Regular example actions and reducers
const increment = (by = 1) => ({type: 'INCREMENT', by});
const decrement = (by = 1) => ({type: 'DECREMENT', by});

const counter = (state = 0, {type, by}) => {
switch (type) {
case 'INCREMENT':
return state + by;
case 'DECREMENT':
return state - by;
default:
return state;
}
}

// redux reducers and store creation
const rootReducer = combineReducers({
location, // the `reroute` location reducer
counter // your own reducers
});

const store = createStore(rootReducer);

// Define your route templates
// the object keys aren't used internally, they are just for latter reference
// `connectToStore` actually transforms it to an array of string.
const routes = {
home: '/',
buttons: '/path/to/buttons(/by/:by)',
total: '/path/to/total'
};

// Connect the `reroute` location handler to the store
const disconnect = connectToStore(store, routes);

// Regular application top level components connect to redux
// Only to get access to `actions` and bits of the app state
@connect((state) => ({by: state.location.urlParams.by}), { increment, decrement })
class ButtonsPage extends Component {
render() {
const by = parseInt(this.props.by || 1);
return (


Increment
Decrement
By {by}


);
}
}

@connect(({counter}) => ({counter}))
class TotalPage extends Component {
render() {
const { counter } = this.props;
return (


Total: {counter}



);
}
}

// This component is responsible for picking and rendering the right component
// It connects to get the `location.matchedRoute` value of the app state
//
// Note: this is not necessary when using a component selector, this is only
// meant to demonstrate the `reroute` principles.
@connect(state => ({ matchedRoute: state.location.matchedRoute }))
class ComponentSwitch extends Component {
render() {
const {matchedRoute} = this.props;

// Simply render the right component based on `matchedRoute`
switch (matchedRoute) {
case routes.home: return

Home page

case routes.buttons: return
case routes.total: return
default: return
unknown route

}
}
}

// The top level component, just a list of links and the component switch
// Note how we only use regular `` elements for navigation
class App extends Component {
render() {
return (


Links


Home

Increment-Decrement Buttons

Increment-Decrement Buttons by 2

Increment-Decrement Buttons by 5

Total

Unknown

View





);
}
}

render(, document.getElementById('app-container'));
```

## API

No comprehensive API doc for now, have a look at the examples.

However, here are the bits of code provided by `reroute`:

* `connectToStore(store, routes)`, must-call, to dispatch location related actions
* `location` reducer, filling the app state with `matchedRoute` and `urlParams`
* `Link` component to generate links from route patterns and url parameters
* `createComponentSelector` helper to create component selector
* `noMatchRouteSelector` helper to generate a selector returning whether a route is matched

### `createComponentSelector(routeToComponentCreators, [locationSelector])`

This helper is meant to ease route-pattern mapping to components.

```js
// [usual dependencies]
import { createComponentSelector, connectToStore, NO_MATCH } from `reroute`;

//
const routes = {
home: '/',
users: '/users',
user: '/users/:userId'
};

connectToStore(store, routes);

// Create a component selector that will use `matchedRoute` to
// pick the right component
const componentSelector = createComponentSelector({
[routes.home]: () => ,
[routes.users]: () => ,
[routes.user]: (urlParams) => ,
[NO_MATCH]: () =>
});

// Connect your component using create componentSelector
@connect(componentSelector)
class App extends React.Component {
render() {
// `this.props.component` is the component you should render
const { component } = this.props;
return

{component}

}
}
```

If the logic of your app routing is more complex than just mapping URL
to component (like authentication), you should still use this helper to
create component selectors and combine them using [`reselect`](https://github.com/rackt/reselect) as in [this
example](./examples/authentication/).

## Recipes / Examples

* [more generic route to component routing](./examples/basic/)
* [protect some views behing authentication](./examples/authentication/)

### Moving `location` data within the application state tree

In most case, the `location` reducer will be used right in the root reducer so
the `location` data is there, at the root.

However, you might want to move it somewhere else. _No problem_.

The only side effect is that the various helpers won't find the `location` data where they expect it to be. All you need to do is to pass them the optional `location` selector.

```js

import { location } from 'redux-reroute';
import { combineReducer } from 'redux';
import { myReducer } from './reducers/mine';
import { myOtherReducer } from './reducers/other';

// manually building an `stuff` reducer containing the `location` data under `path`
const stuff = (state = {}, action) {
return {
path: location(state.path, action)
};
}

const rootReducer = combineReducer({
myReducer,
myOtherReducer,
stuff
});

// in this context, the `location` data will accessible this way:
// location = store.getState().stuff.path

// Just create the location selector pointing to the right place
const locationSelector = (state) => state.stuff.path

// Example for `createComponentSelector`
const componentSelector = createComponentSelector({
// your regular routePattern > component settings
[routes.home]:
[routes.users]:
}, locationSelector); // <- the last option parameter is the locationSelector
```

## TODO

* More examples
* `#`-free path handling
* asynchronous location change, allowing things like loading data
* tests, once the API is a little bit stabilized

## Why not `redux-router`?

We found out than the routing provided by `redux-router` does not let you integrate non-location based routing easily.

You will have to work against it to inject your non-location routing logic, by using either:

* more routing logic in the component `Router` outputs, splitting the routing logic into location vs. non-location code.
* redirecting using `onEnter`, creating some weird non-idiomatic loops.
* re-wrapped and replace the reducer `react-redux` uses
* ???

With `reroute`, you don't have to work around anythings. You build the routing by composing location and non-location related application state.

![reroute-vs-redux-router](./images/reroute-vs-redux-router.png)