Scoping Stores

Creating complex React components like widgets can make state hard to maintain. Especially when there are many nested components that need to pass state back and forth. TrebleGSM has a feature called Scoped Stores that helps alleviate this headache.

Imagine having a data grid component that displays rows and has search functionality. The data grid has a main component made up of a table header row, data rows, and search inputs. Each one of those are separate components that live within the main component. They all need to share state and pass it between each other. If only local state is used, then prop drilling and lifting up state create a component full of spaghetti code 🍝. If the state is put into the App's global state then you have a bunch of state that is only used by one component bloating your App's global state and causing more complexities 😬. With a Scoped Store we can set up a Store within a component that will only allow that component and it's nested components to subscribe to it. Allowing the component to be what it was meant to be... a component! 😎

Create Scoped Store

To create a Scoped Store in our component a new instance of React Context will need to be created.

Treble uses React's native Context API to manage global state under the hood. This means each instance of the Treble container has to utilize a unique instance of React Context. When no custom Context is passed to Treble it utilizes its default Context instance. Scoped Treble components need a custom Context to work correctly.

Create a Store.js file in the desired component folder structure. You can learn more about creating a Treble Store by going here. createStore takes an optional object as a second parameter. The context property will need to be added to this object. Example below.

const ScopedStore = createStore([], { context: /***/ })

You can create a new Context instance by importing the useScopedTreble hook and assigning it to a new variable called Context.

import {useScopedTreble} from 'treble-gsm'

const Context = useScopedTreble();

Pass the new Context variable to the createStore option property context.

import {useScopedTreble} from 'treble-gsm'

const Context = useScopedTreble();

const ScopedStore = createStore([], { context: Context });

The Store is now a Scoped Store! 😸

Create Scoped useTreble Hook

Now that a Scoped Store is set up, a custom useTreble hook must be created. This hook will only subscribe to this Store.

Import the useTreble hook into Store.js.

import {useTreble} from 'treble-gsm';

Assign useTreble to a new custom hook variable and pass the scopedContext variable to it.

const useScopedStore = useTreble(scopedContext);

You can call the new hook anything you want, BUT it must start with use. See Building Your Own Hooks

You will then need to export the custom Hook from your Store.js file so it is accessible to the rest of the component. Final Store.js should look like below.

Connect Component to Store

In order to connect a component to the Store, it will need to be wrapped by withTreble. The withTreble function takes an options object parameter. You will need to add a store property and pass the Store to it. Once this is set up the wrapped component and its children components will have access to the Scoped Store.

import {Store} from './scoped-store';

function SomeComponent(){
    return(
      <>
      </>
    )
}
const SomeComp = withTreble(SomeComponent, {store: Store});
export default SomeComp;

Subscribing to Multiple Stores

If your component is utilizing a Scoped Store there still might be a scenario where your component needs to subscribe to the App's global Store as well. All that needs to be done in this scenario is to import the default useTreble hook and destructure the state objects.

import {useTreble} from 'treble-gsm';
import {Store, useScopedStore} from './scoped-store';

function SomeComponent(){

    const [{someState}] = useTreble();
    const [{someScopedState}] = useScopedStore();
    return(
      <>
      </>
    )
}
const SomeComp = withTreble(SomeComponent, {store: Store});
export default SomeComp;

Updating Multiple Stores

This can easily be accomplished by giving the useTreble Store object an unique name.

import {useTreble} from 'treble-gsm';
import {Store, useScopedStore} from './scoped-store';

function SomeComponent(){

    const [{someState}, Store] = useTreble();
    const [{someScopedState}, ScopedStore] = useScopedStore();
    return(
      <>
      </>
    )
}
const SomeComp = withTreble(SomeComponent, {store: Store});
export default SomeComp;

Now your component has a Scoped Store and you can subscribe and update this state like you would with the App's gobal Store. Can you feel the power?! ⚡⚡⚡

Want to Support this Project?