npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

redux-model-utils

v1.0.1

Published

Utilities and patterns for models using redux.js

Downloads

39

Readme

npm install --save redux-model-utils

File size: 2k (min/gzip).

Redux model utilities

You have in your hands a set of model-building utilities for Redux apps. Like Redux itself, this library works very well whether or not you use React for rendering.

To use this library, you'll need to build your models using the patterns described below. These patterns, in effect, ARE the library.

Before jumping into the full documentation, let's take a look at what your code will look like when you use this library.

Model-building patterns

This library takes Redux and wraps it in a thin, almost object-oriented facade. Each feature in your app can be represented by a single object (or "model"). Each model has its own private methods (i.e., action creators), and its own private data (i.e., its slice of the store). Views interact with the store by invoking these methods, and retrieve data either by using React-Redux's connect utility, or by subscribing to a slice of the store.

Behind the scenes, the methods dispatch actions and use reducers to reconcile the model data. (It's all Redux, in the end.) But from a developer's perspective, this is a private, internal implementation detail. To use one of these models, you will never use the dispatch method, nor mapStateToProps, nor will you need to create action types.

Consider a simple model to track user data. The following is pseudo-code, just to illustrate where we're going:

pseudocode-only.js
// private data:
let userID = 0,
    preferences = {
      lang: 'en',
      shirtSize: 'L'
    };

// public methods
export function loadUser(id) {}
export function saveUser() {}
export function updateLanguagePref(newLang) {}

// public accessors
export const language = () => preferences.lang;
export const shirtSize = () => preferences.shirtSize;

Here's the representation using Redux Model Utils:

user-model.js
import {modelBuilder} from 'redux-model-utils';

// the modelBuilder tool takes one parameter, an object with four keys:
//  {name, initialState, actionMap, selectors}
// the insteresting bits are "actionMap" and "selectors"
//
const model = modelBuilder({

  name: 'userModel',

  // the store data for this model
  initialState: {
    userID: 0,
    preferences: {
      lang:      'en',
      shirtSize: 'L'
    }
  },

  // the model's "methods". these are described declaratively, and transformed into
  // callable functions on "model.actions". for example: model.actions.loadUser(88).
  // the actions are typically passed to connected components as a prop (see below)
  //
  actionMap: {

    // this describes a function on model.actions.loadUser()
    loadUser: {
      // the function (i.e., action creator) takes one parameter
      params: 'id',

      // this reducer is atomic; it's only invoked when this action is dispatched.
      // for now, we'll just set the user's ID. (async actions are covered later.)
      reducer: (state, {id}) => ({...state, userID: id})
    },

    // more methods, on model.actions.saveUser(), etc
    saveUser: {/* ... */},
    updateLanguagePref: {/* ... */}
  },

  // the following "selectors" are passed as props to your connected components.
  // selectors can expose data in raw form (directly from state) or they can
  // calculate data
  //
  selectors: {

    // your connected component gets calculated values on
    //   "this.props.language" and "this.props.shirtSize"
    language:  state => state.preferences.lang,
    shirtSize: state => state.preferences.shirtSize,

    // the following prop gives connected components direct access to the model's methods
    // e.g., "this.props.userActions.loadUser(99)"
    userActions: () => model.actions
  }
});
export default model;
user-view.js
import userModel from './models/user-model.js';

@connect(userModel.mapStateToProps)
class UserView extends React.Component {
  render() {
    // we now have access to the model's selectors as props
    //
    const {language, shirtSize, userActions} = this.props;
    
    // the model's methods have been exposed as props.userActions
    const {loadUser, saveUser} = userActions;
    
    // ...
  }
}

export default UserView;

Why?

This not only removes most of the Redux boilerplate, but it also gives you:

  • Atomic reducers. Each action has a single-purpose reducer. There are no switch statements. (You can share action types if you like, so you can have multiple reducers handle the same action.)

  • Encapsulation. The model has everything it needs to manage a single feature in your app. There are no separate mapStateToProps calls in container components, no separate action-type files.

  • Declarative API. Defining a new action is simple, and all happens in one place.

There are also tools for async actions, private actions, shared actions, and more. Read the full docs on actions.

Easy connection to React components

If you're using this library in a React app, you're in luck. No need to write mapStateToProps or mapDispatchToProps. It's all done for you when you connect a model to a component.

// here, we create a connected component around the user model above
//
import userModel from './models/user-model.js';

class UserSettingsPage extends Component {
  render() {
    let {language, shirtSize, userActions} = this.props;
    // ...

    return <button onClick={() => userActions.saveUser()}>Save</button>
  }
}
export default connect(userModel.reactSelectors)(UserSettingsPage);

Details in the React docs.

Change-notification (subscriptions)

Built on top of Redux's subscription tool, this subscription will only alert you when a property changes. Use this for cross-cutting concerns. (In non-React apps, this will be your main rendering trigger.)

let unsubscribe = userModel.subscribe(userModel.selectors.shirtSize, (newShirtSize, oldShirtSize) => {
    // callback is called once at initialization time, and then only
    // when the property changes
    console.log(`Font size changed from ${oldFontSize} to ${newFontSize}`);
});

Direct read-only access to the model state

Here's a snippet from a model which shows how you can peek into the properties of any model, not just those which you connect to your components. Caveat follows the code.

let initialState = {
      token:     null,
      pollTimer: null
    },
    // selectors are also converted into read-only properties
    // on the model's "data" object
    selectors = {
      token:     state => state.token,
      pollTimer: state => state.pollTimer
    },
    uiModel = modelBuilder({/*... */});

// ... then later, in your view ...
let token = uiModel.data.token;

This means you don't need to connect every model to every component. Peek into any model you like, and you'll get the latest values from the master Redux store.

Caveat: This is convenient, but not always a best practice. In React apps, you'll typically use mergeReactSelectors if your views need to respond to changes across multiple models. And in non-React apps, you'll typically use subscribe to track changes. Use with caution, as this feature may be deprecated soon.

Learn more