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-scope-utils

v1.1.1

Published

Utilities for Redux scope patterns

Downloads

14

Readme

redux-scope-utils

Utilities for Redux Scope Patterns

👷‍Docs Work In Progress!

✅ TODOS:

  • Proof all doc code
  • Proof use of italics and code
  • Unit test modules
  • Document modules
  • Document alternate patterns

Redux Scope Utils is a set of functions to promote reusability of actions, reducers and selectors in a redux application. They can be wrote in a generic manner, yet used across an application, applying only to a specific part of the state tree, the scope.

scopedReducer(reducer, scope)

Only actions with a matching scope will be passed to the reducer.

scopedAction(actionCreator, scope)

Adds scope to an action so that it can pass the scoped reducer's test.

scopedSelector(selector, scope)

Uses scope to traverse the state tree. The selector should be relative to a the reducer.

In Depth

While there are many ways to write reducers, this library assumes your state will be constructed by using combineReducers from the redux package. The combineReducers function can be used to create deeply nested application states (objects containing objects). At the end of this tree will be a final reducer, and it's path is the scope.

/* The `/` separated path to the reducer is the `scope`
{
  menu: {
    order: scopedReducer(menuReducer, 'menu/order/')
  }
}

To explain the concept of a scope we need to build an application state with multiple of the same reducers. Lets start with a simple reducer based off the counter example from the Redux docs:

const counterReducer = (state = 0, action) => {
  switch (action.type) {
    case 'INCREMENT':
      return state + 1;
    case 'DECREMENT':
      return state - 1;
    default:
      return state;
  }
};

The counterReducer's initial state is just 0. It can handle INCREMENT and DECREMENT actions to change the state by one.

The Problem

The next example will use combineReducers to add 2 instances of the counterReducer reducer. When we dispatch an action we can see a problem with this design. Both reducers will increment since they both handle the INCREMENT action type.

const rootReducer = combineReducers({
  likes: counterReducer,
  followers: counterReducer
});
// Produces an initial state:
// { likes: 0,  followers: 0 }
// ...

store.dispatch({ type: 'INCREMENT' });
// Produces a new state:
// { likes: 1, followers: 1 }

How would you approach this problem? Write separate likes and followers reducers, each handling their own action types (INCREMENT_LIKES, INCREMENT_FOLLOWERS, ...)?

Scoped Reducers

The goal of this library is redux logic reuse and we will use "scoped" reducers to do so. A scoped reducer will handle a generic action type like INCREMENT, but only if the action has the proper scope property.

When an action is dispatched, the store's rootReducer will be called with the current state and the dispatched action. The example rootReducer is created with combineReducers, which will call each child reducer with the state and the action.

The scopedReducer(reducer, scope) function from this library will act as a gateway to our counterReducer. The reducer will only be called if a matching scope is included with the action.

import { createStore, combineReducers } from 'redux';
import { createScopedReducer } from 'redux-scope-utils';

const rootReducer = combineReducers({
  likes: scopedReducer(counterReducer, 'likes'),
  followers: scopedReducer(counterReducer, 'followers')
});

const store = createStore(rootReducer);

store.dispatch({ type: 'INCREMENT' });
// Nothing changes
// { likes: 0, followers: 0 }

store.dispatch({ type: 'INCREMENT', meta: { scope: 'likes' } });
// likes changed!
// { likes: 1, followers: 0 }

Scoped Actions

A scoped action is a redux action, which contains info related to it's "scope"

Lets refactor our counter so it has action creators (functions which return action objects).

const INCREMENT = 'INCREMENT';
const DECREMENT = 'DECREMENT';

// Increase counter value by one
export const incCounter = () => ({ type: INCREMENT });

// Lower counter value by one
export const decCounter = () => ({ type: DECREMENT });

export const counterReducer = (state = 0, action) => {
  switch (action.type) {
    case INCREMENT:
      return state + 1;
    case DECREMENT:
      return state - 1;
    default:
      return state;
  }
};

Then for each scoped reducer, you can create matching scoped actions creators. When they are called, they will contain the scope information.

import { scopedAction } from 'redux-scope-utils';
// Import action creators from your reusable module
import { incCounter, decCounter } from 'example/counter';

// Creates a new action creator which includes the `scope` of `likes`
const upvote = scopedAction(incCounter, 'likes');
const downvote = scopedAction(decCounter, 'likes');

// Somewhere else... dispatch the action with included scope
import { upvote } from 'store/likes';

store.dispatch(upvote());
/* Creates an action:
 * {
 *   type: 'INCREMENT',
 *   meta: { scope: 'likes' }
 * }
 */

Scoped Selectors

The final part of this is retrieving data from state. To get data from our scoped reducer, we will use a scoped selector. A selector is a function which takes state and returns a subset of the state or derives a new value.

The counterReducer is so simple it barely needs a selector. It's state is simply a number,.

const getCount = state => state;
// ...
const getLikes = scopedSelector(getCount, 'likes');
const getFollowers = scopedSelector(getCount, 'followers');

A reducer with more complex shape will have more complex selectors. For example a reducer for a shopping cart may maintain a list of items in the cart and if it has been submitted.

The selectors for a scoped reducer should be relative to their reducer. The scope will be used to traverse the state tree to the reducers node and apply the selector from there.

// Example "cartReducer" state
// {
//   submitted: false,
//   items: [
//     { name: 'Milk', price: 315 },
//     { name: 'Cookies', price: 225 }
//   ]
// }

const getCartSubmitted = state => state.submitted;
const getCartItems = state => state.items;
const getCartTotal = state => getCartItems(state).reduce(
  (acc, curr) => acc + curr.price,
  0
)

See scopedSelector documentation for more examples.

Why meta?

An action in Redux is not a strictly conformed object. The only requirement is that it contains a type property. This library follows the pattern specified for Flux Standard Actions. It specifies that an action must contain a type property, and optionally a payload, error or meta property.

The Flux Standard Actions documentation describes meta as:

The optional meta property MAY be any type of value. It is intended for any extra information that is not part of the payload.

In our case the "extra information" is our scope!

References