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 🙏

© 2024 – Pkg Stats / Ryan Hefner

vlow

v1.1.17

Published

A simple library for unidirectional dataflow architecture inspired by Reflux

Downloads

61

Readme

CI Release Version

Vlow

A simple library for unidirectional dataflow architecture inspired by Reflux



Installation

Using npm:

$ npm i vlow

In your project:

import Vlow from 'vlow';
// Exposes:
//  - Vlow.version
//  - Vlow.createActions
//  - Vlow.Store
//  - Vlow.Component
//  - Vlow.withVlow

Or... download the latest release from here and load the file in inside your project. For example:

<!-- Add this line to expose `window.vlow.default` -->
<!-- WARN: Make sure React is loaded before this line -->
<script src="vlow.min.js"></script>

Overview

Vlow uses actions to update one or more stores which updates the state of all components who are mapped to the store(s). In turn, a component triggers an action. Vlow can be used for keeping a global state.

There are three steps which need to be understood to use Vlow:

  1. Create Actions
  2. Create Stores
  3. Map store(s) to Components

Create actions

Actions can be created using the Vlow.createActions function:

// this example creates an add and remove action
const ItemActions = Vlow.createActions([
    'add',
    'remove',
]);

Invoking an action is as simple as calling the function, for example:

ItemsActions.add(item);

Create a store

A Vlow Store is an object which holds a global state which can be shared across components.

Creating a store can be done by creating a subclass of Vlow.Store and call the Store constructor with an actions object, for example:

class ItemStore extends Vlow.Store {
    constructor() {
        // Call super with the actions to which this store should
        // listen too.
        // Note: multiple actions instances are possible,
        //       for example super(Actions1, Actions2);
        super(ItemActions);

        // Create the initial state
        this.state = {
            items: []
        };
    }

    // Implement onAction functions for each action defined by actions.
    // It is not required to create functions for all actions but usually
    // this is what you want and Vlow logs a warning in `development` mode
    // when actions are not implemented.
    onAdd(item) {
        // Update the state like you would do with React.
        // Just like react the first argument can also be a function.
        //
        // An optional second argument can be used. When given it must be
        // a callback function() which will be called once when all
        // components which are mapped to the store are rendered.
        this.setState({items: [...this.state.items, item]});
    }

    onRemove(itemId) {
        this.setState({items: this.state.filter(i => i.id !== itemId)});
    }
}

Map stores to Components

Now that the actions and stores are created, it is time to map them to a component.

This can be done either by using Vlow.withVlow which is the preferred method as of version 1.1.0, or it can be done by extending the Vlow.Component class.

Using withVlow

By using withVlow the store will be mapped to the component props. The function withVlow requires a Vlow Store or an array with multiple stores and returns a new function which accepts a component you want to wrap.

Here are some valid examples:

// Just parse the store
withVlow(SomeStore)(MyComponent);

// Multiple stores
withVlow([SomeStore, SomeOtherStore])(MyComponent);

// Map only specific store keys
withVlow({
    store: ItemStore,
    keys: ['items'] // listen only to 'items' changes
})(MyComponent);

And this is an example of how withVlow can be used:

import {withVlow} from 'vlow';
import ItemStore from '../Stores/ItemStore';

const ItemComponent = ({items}) => (
    <ul>
        {items.map(i => <li key={i.id}>{i.text}</li>)}
    </ul>
)

export default withVlow(ItemStore)(ItemComponent);

Note: It is still ok to use PropTypes for checking the props from a store, for example:

ItemComponent.propTypes = {
   items: PropTypes.arrayOf(PropTypes.shape({
       id: PropTypes.number,
       text: PropTypes.string
   })).isRequired
};

Using Vlow Component

It is recommended to use withVlow instead of Vlow.Component but in some cases you might prefer to extend your component directly with a Vlow.Component instead of React.Component. The main difference is that the state from the store will be merged with your components state, instead of receiving the state by props.

Inside the constructor you should use the mapStore() or mapStores() function to map the state to the store.

Here is an example of how to use Vlow.Component:

class ItemComponent extends Vlow.Component {
    constructor(props) {
        super(props);
        // In case you want to set state in the constructor, make
        // sure to do this before calling mapStore() since you
        // otherwise would overwrite the state defined by the store.
        this.state = {
            some: 'state'
        };

        // Function mapStore() accepts a store in which case the
        // complete store state will be mapped to the components state.
        // There are two more options:
        //
        // - Using a keys filter in which case a component only listens
        //   to certain store changes.
        //
        //      this.mapStore({
        //          store: ItemStore,
        //          keys: ['items'] // listen only to 'items' changes,
        //                          // other store state will be ignored
        //      });
        //
        // - Using an altState function which allows you to modify state
        //   before it will be applied. (more info on altState() can be
        //   found later in this documentation)
        this.mapStore(ItemStore);

        // The state now looks like:
        //    {some: 'state', items: []}
        // It's ok to modify the state as long as you do not
        // overwrite `this.state` with a new Object, for example:
        this.state.hasItems = this.state.items.length > 0;
    }

    render() {
        return (
            <ul>
                {this.state.items.map(i => <li key={i.id}>{i.text}</li>)}
            </ul>
        );
    }
}

In case you need multiple stores, the function this.mapStores([]) can be used which accepts an Array of stores. Each store may be defined in a different way.

this.mapStores([
    StoreOne, {
        store: StoreTwo
    }, {
        store: StoreThree,
        keys: ['foo', 'status']
    }, {
        store: StoreFour,
        altState: storeState, state, props => storeState
    }
]);

Alternative super class

By default the Vlow.Component class extends the React.Component class but you might want Vlow.Component to extend your own custom class. This is possible by using Vlow.Component.extend().

// In this example we create a Vlow Component which extends
// React.PureComponent instead of the default React.Component
class MyComponent extends Vlow.Component.extend(React.PureComponent) {
    ...
}

Using altState

Sometimes you want to listen to state changes in a store but then do something with this state instead of just applying the state to a component. This can be done by using an altState(storeState, state, props) hook which will be triggered on state changes in the store but before the component state is changed. The altState function should return the state changes you want to make or null in case you don't want to update the state of the component.

function altState(storeState, state, props) {
    // This function should return the state
    // you want to apply on the component. The function can
    // also return `null` in which case the components state
    // will not be changed.
    if (props.status === 'error') {
        // the components state will not be changed
        return null;
    }
    // Return some alternative state for `this` component.
    // Other components still receive the `original` state
    // from the store.
    return {
        items: storeState.items.filter(i => i.age > state.minAge)
    };
}

// Use the altState function
const stores = {
    store: MyStore,
    altState,
};