Уважаемые коллеги, представляю вашему вниманию и на ваше осуждение контейнер для управления состоянием React приложения xstore. Он определенно является таким маленьким детским велосипедом рядом с большим и сверкающим мотоциклом Redux. Все мы программисты JavaScript являемся такой большой и не сбавляющей обороты фабрикой по производству велосипедов.


Для более менее просто начинающих или начинающих свое знакомство с React JavaScript программистов Redux может показаться несколько сложной штукой, которая иногда непонятно как работает и к которой сложно "законнектиться", хочется чего-то попроще, чего-то похожего на данный маленький велосипед.


Давайте рассмотрим его поближе.


Установка


npm install --save xstore

Использование


Для начала нам нужно добавить в хранилище "обработчики".
(Пример обработчика представлен в блоке ниже)


index.js


import React from 'react'
import ReactDOM from 'react-dom'
import Store from 'xstore'
import App from './components/App'

import user from './store_handlers/user'
import dictionary from './store_handlers/dictionary'

// Имя обработчика не должно содержать символ "_".
// В данном случае мы имеем два обработчика: user и dictionary.
Store.addHandlers({
  user,
  dictionary
});

ReactDOM.render(
  <App/>,
  document.getElementById('root')
);

Далее идет пример обработчика "user".
В нем содержится редьюсер "init", который нужен, чтобы определить начальное состояние хранилища "user".


./store_handlers/user.js


import axios from 'axios'

const DEFAULT_STATE = {
  name: 'user',
  status: 'alive'
}

/**
 ===============
 Reducers
 ===============
*/

// Будет автоматически вызван для инициализации начального состояния.
// Если данный редьюсер не существует, начальное состояние будет пустым объектом.
// Вызывайте этот редьюсер, чтобы сбросить состояние до начального.
// Для вызова используйте this.props.dispatch('USER_INIT').
const init = () => {
  return DEFAULT_STATE;
}

// this.props.dispatch('USER_CHANGED', {name: 'NewName'})
const changed = (state, data) => {
  return {
    ...state,
    ...data
  }
}

/**
 ===============
 Actions
 ===============
*/

// this.props.doAction('USER_CHANGE', {data})
const change = ({dispatch}, data) => {
  // {dispatch, doAction, getState, state}
  dispatch('USER_CHANGED', data);
}

// this.props.doAction('USER_LOAD', {id: userId})
const load = ({dispatch}, data) => {
  // {dispatch, doAction, getState, state}
  axios.get('/api/load.php', data)
    .then(({data}) => {
      dispatch('USER_CHANGED', data);
    });
}

// Этот объект конечно же должен быть обязательно такого вида
export default {
  actions: {
    load,
    change
    // remove, save, add_item, remove_this_extra_item, .....
  },
  reducers: {
    init,
    changed
   // removed, saved, item_added, this_extra_item_removed, .....
  }
} 

Далее пример того, как подключить компонент к хранилищу.


./components/ComponentToConnect/index.js


import React from 'react'
import Store from 'xstore'

class ComponentToConnect extends React.Component {
  render() {
     // Свойства user и dictionary будут получены из хранилища.
     let {user, dictionary} = this.props;
     return (
       <div className="some-component">
            ....
       </div>
     )
  }

  loadUser(userId) {
     // Вызов экшена в компоненте.
     this.props.doAction('USER_LOAD', {id: userId});
  }

  setUser(userData) {
     // Вызов редьюсера в компоненте.
     // Но лучше так не делать и вызывать только экшены.
     this.props.dispatch('USER_CHANGED', userData);
  }
}

// Непосредственно подключение к хранилищу.
const params = {
  has: 'user, dictionary'
}
export default Store.connect(ComponentToConnect, params);

Возможные опции "params" для подключения:


{
  // Названия хранилищ, к которым будет подключен компонент:
  has: 'user, dictionary',
  // или по-другому:
  has: ['user', 'dictionary'],

  // Если необходимы только некоторые поля из хранилища:
  has: 'user:name|status, dictionary:userStatuses',
  // или
  has: ['user:name|status', 'dictionary:userStatuses'],

  // Компонент будет ждать содержимое данных хранилищ, и только тогда отрисуется.
  shouldHave: 'user,dictionary',
  // или
  shouldHave: ['user', 'dictionary'],

  // Чтобы извлечь данные из нескольких хранилищ на верхний уровень, установите в true.
  // В результате компонент получит пропсы "name", "status", "userStatuses" вместо "user" и "dictionary"
  flat: true,

  // Нужен, чтобы добавить префикс к извлеченным данным, работает только если flat = true.
  // В результате компонент получит пропсы "user_name", "user_status", "dictionary_userStatuses"
  withPrefix: true,

  // Вы можете добавить обработчики непосредственно здесь.
  // Если в этом списке содержатся все необходимые, параметр "has" можно не передавать.
  handlers: {
    user,
    dictionary
  }
}

Список публичных методов хранилища:


import Store from 'xstore'

// Возвращает клонированный объект содержащий в себе данные всех хранилищ:
let state = Store.getState();

// Возвращает клонированный объект содержащий в себе данные хранилища "user":
let userState = Store.getState('user');

// Возвращает поле "name" из хранилища "user":
let userName = Store.getState('user.name');

// Возвращает поле с индексом 0 из поля "items" из хранилища "user":
let someItem = Store.getState('user.items.0');

// Добавление обработчиков:
Store.addHandlers({
  user: userHandler,
  dictionary: dictionaryHandler
})

// Вызов экшена "load" хранилища "user":
// Название экшена будет приведено в нижний регистр, так что не зависит от регистра.
Store.doAction('USER_LOAD', {id: userId});

// Вызов редьюсера "loaded" хранилища "user":
Store.dispatch('USER_LOADED', data);

// Подписка компонента на изменения хранилища:
Store.connect(Component, params);

А теперь о том как это работает:


Метод "connect" создает новый HOC класс XStoreConnect, который скрывает в себе всю логику по взаимодействию компонента и хранилища. Данный класс подписывается на изменения хранилища и, когда там происходят какие-то изменения, им вызывается метод setState защищенный от вызова извне (например через this.refs.xStoreConnect.setState(...)), после чего данный компонент перерисовывается, тем самым обновляя пропсы в обёрнутом компоненте.
Прямое изменение состояния компонента-обёртки this.refs.xStoreConnect.state = something тоже ни к чему не приведет, данный класс умеет находить внедренные данные и удалять их.


Код компонента-обёртки
// .... Здесь еще много функционала хранилища

const LOCAL_OBJECT_CHECKER = {};
const connect = (ComponentToConnect, connectProps) => {
    let ready = false;
    let {
        has,
        handlers,
        shouldHave: shouldHaveString,
        flat,
        withPrefix
    } = connectProps;

    if (!has && handlers instanceof Object) {
        has = Object.keys(handlers);
    }
    let shouldHave = [];
    if (typeof shouldHaveString == 'string') {
        shouldHaveString = shouldHaveString.split(',');
        for (let item of shouldHaveString) {
            if (item) {
                shouldHave.push(item.trim());
            }
        }
    }
    let doUnsubscribe,
        doCleanState,
        stateItemsQuantity;
    return class XStoreConnect extends React.Component {
        constructor() {
            super();
            const updater = (state) => {
                stateItemsQuantity = Object.keys(state).length;
                if (ready) {
                    this.setState(state, LOCAL_OBJECT_CHECKER);
                } else {
                    this.state = state;
                }
            }
            doUnsubscribe = () => {
                unsubscribe(updater);
            }
            doCleanState = () => {
                cleanStateFromInjectedItems(updater, this.state);
            }
            subscribe(updater, {has, handlers, flat, withPrefix});
        }

        setState(state, localObjectChecker) {
            if (state instanceof Object && localObjectChecker === LOCAL_OBJECT_CHECKER) {
                super.setState(state);
            }
        }

        componentWillMount() {
            ready = true;
        }

        componentWillUnmount() {
            ready = false;
            doUnsubscribe();        
        }

        render() {
            let {props, state} = this;
            let newStateKeysQuantity = Object.keys(state).length;
            if (stateItemsQuantity != newStateKeysQuantity) {
                doCleanState();
            }
            for (let item of shouldHave) {
                if (state[item] === undefined) {
                    return null;
                }
            }
            let componentProps = {
                ...props,
                ...state,
                doAction,
                dispatch
            };
            return <ComponentToConnect {...componentProps}/>
        }
    }
}

Генерация файлов обработчиков из командной строки:


npm install -g xstore
xstore create-handler filename

Также можно прописать в "scripts" в "package.json":


{
  scripts: {
    "create-handler": "node node_modules/xstore/bin/exec.js"
  }
}
npm run create-handler filename

Эта команда создаст файл filename.js (если такого не существует) с шаблонным кодом обработчика.


Вот и всё, совсем просто не так ли? А теперь можете пинать. Буду рад советам и разумной критике, уважаемые коллеги.

Комментарии (7)


  1. vasIvas
    06.11.2017 18:25
    +1

    user.items.0 — это не нормально. Очень нехорошо обращаться чем-либо в объекте не через точечную нотацию.


    1. bushstas Автор
      06.11.2017 20:07

      Спасибо за совет, не нормально так же и user.items или только user.items.0?


      1. timiskhakov
        06.11.2017 21:31

        Думаю, vasIvas имел ввиду закон Деметры. С user.items все нормально.


      1. vasIvas
        06.11.2017 21:33

        ненормально работать с объектом через строковые запросы. Поэтому user.items это не нормально, а уж user.items.0 просто безумно. Почему Вы отвергаете классику getState().user.items?


  1. Tuerer
    06.11.2017 19:44

    теперь крути мемоизированные селекторы, мидлвари и в нпм) шучу)


  1. redyuf
    06.11.2017 20:53

    Для реального использования не годится, разве что в образовательных целях.
    'user, dictionary' — строковая типизация.
    getState('user.name'); — строковые селекторы, что будет с путями при рефакторинге стейта?
    Store.addHandlers — как быть, если нужна копия стора с другими путями в стейте?

    Redux сложный? По-моему, как раз одно из самых простых решений (если сравнить реализацию с тем же mobx), наивная реализация пишется строк в 100.

    State-tree контейнеров десятки в js, зачем еще один? Вот mobx-state-tree неплох, разве он такой сложный, что надо писать свой велосипед?

    Попробуйте избавиться от синглтонов, в редуксе как раз все через фабрики создается: createStore и т.д.

    Попробуйте использовать типы, без типов рефакторинг приложения — как хождение по минному полю. Сделайте так, что б при изменении в объекте user свойства name на firstName, typescript или flow подсвечивали ошибку в селекторе. В том же mst типы выводятся.


  1. AlexzundeR
    07.11.2017 09:46

    Мне кажется вызов action'ов должен быть более декларативным. Было бы здорово, если бы в props прокидывались действия из store, и их вызов осуществлялся через точечную нотацию, вместо указания строковых констант в достаточно абстрактном методе doAction.