Skip to content

A clean state management library for React/React Native. Effectively separate logic and UI

License

Notifications You must be signed in to change notification settings

SilentCatD/react-state-view-controller

Repository files navigation

React State-View-Controller

NPM version codecov npm bundle size Build License

Overview

In my opinion, a robust state management system should effectively adhere to the following criteria:

  1. Effective Separation of Logic and UI

    These two aspects should not be tightly coupled. While it may be acceptable for a single or simple component, when dealing with complex components, screens, or features (comprising multiple screens), the controlling logic should be separated. This separation brings several benefits:

    • Easier Dependency Injection (DI): We can create a network of interdependent Controller with clearly defined coupling relationships, adhering to the Component Principle.
    • State Lifecycle: A State/Model managing a View should align with the View's lifecycle. When the screen is mounted, the State and Controller managed it is created and evolves throughout the View lifecycle, triggering UI layer re-rendering when it changes. When the View is unmounted, the managing Controller and State should be disposed of, cleaning up any unnecessary resources.
    • Abstraction: It hides the complexity of logic, allowing the UI to trigger a logic flow by simply calling a function in the Controller, which may involve complex interactions, operations through APIs, or interactions with other Controllers, and emit State as necessary.
    • Reusability: One of the advantages is the ease of reusing logic between different Controllers.
  2. Dependency Injection (DI)

    A strong state management system should provide an effective mechanism for dependency injection. The Controller should be easily accessible for any of the children within its scope, especially those currently managed by it. The Context API is a good example of this mechanism, as it allows access to data without having to pass props down to every child component.

  3. Scoped Re-render Trigger and Re-render Filter

    Regardless of how effectively the framework optimizes this process (e.g., comparing props in React), the trigger for the rebuild process should be optimized. It's better if we have more control over where and when this process occurs, even though it's just a trigger for UI re-rendering.

    We should be able to specify where re-rendering may occur and set conditions for it. For example, we might have a component that only re-renders when the a property of the State object is greater than 5.

Usage

This library is created to address the concerns mentioned above.

Controller

Overview

When defining a Controller for a React view or view group, adopting a subclass approach brings several advantages:

1. Inter-Controller Subscription

Controller instances can seamlessly subscribe to and depend on each other, enabling internal triggers for state emissions.

2. Inheritance and Extensibility

Utilizing inheritance facilitates the extension of a Controller, promoting the creation of more reusable code.

3. Property Manipulation

Creating, accessing, and modifying properties within a Controller becomes a straightforward process.

State Management

The core of the state management revolves around the State object, responsible for holding data crucial for UI rendering. The UI utilizes the State object during its rendering process.

Base Controller Class

Explore some of the available functions in the base Controller class. The provided functionality eliminates the need for redundant code, unless customization is desired.

abstract class Controller<T> {
  // The initial state of the UI
  constructor(initialState: T)

  // The observable to subscribe to if necessary
  // Will fire new states and notifying listeners.
  public get observable(): Observable<T>

  // The current state that the controller is maintaining
  public get state(): T

  // Emit a new state and trigger all listeners.
  // Will be merged with existed state
  protected emit(state: Partial<T>)

  // Override if necessary to clean up any resources.
  // Will be called when attached view unmounted
  public async dispose(): Promise<void>
}

A Controller will directly interact with the State object to mutate it, indirectly causing UI re-rendering.

To create a new Controller, you need to extend the Controller class, which provides methods for state mutation and notifying listeners. In the example below, the CounterController has {count: 0} as its initialState. It updates it through methods like increaseCounter or decreaseCounter using emit(newState).

import { Controller } from 'react-state-view-controller'

type CounterState = {
  count: number
}

class CounterController extends Controller<CounterState> {
  constructor() {
    super({ count: 0 })
  }
  increaseCounter() {
    // Use `emit` to update new state.
    this.emit({ count: this.state.count + 1 })
  }
  decreaseCounter() {
    this.emit({ count: this.state.count - 1 })
  }
}

Do note that the newState object must be different from the old State. Otherwise, the Controller will just skip it. This optimization avoids unnecessary state updates.

When using emit, the {...this.state} is not needed to copy other contents of old state. Object passed in the emit(newState) function will be merge with the existed current state.

One of many common patterns is to handle all of the necessary logic to fetch data from an API in the Controller, then emit the data from within the Controller. For example:

import { Controller } from 'react-state-view-controller'

type CounterState = {
  loading: boolean
  count: number
}

class CounterController extends Controller<CounterState> {
  ///
  async fetchCounter() {
    this.emit({ loading: true })

    // some function to fetch data
    const newCounter = await fetchDataFromSource()

    // emit new state with changed data
    this.emit({ loading: false, count: newCounter })
  }
  ///
}

As you can see, we don't need to worry about UI-related code here in this async operation. It is the UI's responsibility to subscribe to changes in the State object and render accordingly.

In the UI, we can then check for the loading property of the State object and render a LoadingScreen if necessary.

Provider and Dependency Injection (DI)

A Controller manages states for a group of child components, and so it must be provided to them for further interaction. This is similar to the Context API. In fact, this library uses the Context API internally for DI.

To provide a group of child components with a Controller, we can use ControllerProvider:

<ControllerProvider source={() => new CounterController()}>
  <CounterComponent />
  <ButtonComponent />
</ControllerProvider>

The source parameter take in a function to create a Controller and keep it with useRef throughout the component's life-span. When the component unmounted, a clean up function will be called automatically, trigger the dispose function defined inside the Controller class, allowing resources clean up when a Controller is not needed anymore.

Another way to provide a Controller to children components would be using ControllerProvider but pass to the source param an instance of Controller instead of a create function:

// existed isntance else where, you would have to manage the references and cleanup yourself though
<ControllerProvider source={counterControllerInstance}>
  <CounterComponent />
  <ButtonComponent />
</ControllerProvider>

But be aware that Controller provided this way won't be kept with useRef like the first approach, clean up function also won't be called when this ControllerProvider unmounted.

The CounterComponent and ButtonComponent will now have access to the CounterController.

Inside the ButtonComponent or CounterComponent, you can get the provided CounterController instance through the useProvider hook:

import { useProvider } from 'react-state-view-controller'

const ButtonComponent = () => {
  // pass in the type of class
  const controller = useProvider(CounterController)
  return (
    <div>
      <button onClick={() => controller.increaseCounter()}>Increase</button>
      <button onClick={() => controller.decreaseCounter()}>Decrease</button>
    </div>
  )
}

You can interact with the provided CounterController as needed. Please note that this will get the nearest provided CounterController, and if it can't find one (e.g., if a CounterController is not provided to this scope), an error will be thrown.

This hook should not cause re-render when new State from CounterController is emitted.

The Controller instance passed to ControllerProvider will have it's name extracted and used as a resource-key for later query, this mean if you provide a class with the name A, you have to use the exact type to query it with useProvider hook.

But for cases that one type of Controller may has many implemmentations, subclass-ing each other, and we only want the dependent/client of this controller know it's base type, we can use the ctor param in ProviderController to overwrite the query type.

// query type will now become `TestController`
// else where:
// useProvider(TestController)
<ControllerProvider ctor={TestController} source={()=> SubclassOfTestController()}>
  <Screen />
</ControllerProvider>,

Context hell

Now, we can observe that a wrapper like this indents the file slightly. However, when multiple ControllerProvider components are present, the file may appear disorganized.

For instance, you might encounter a structure like this:

<ControllerProvider create={() => new CounterController()}>
  <ControllerProvider create={() => new CounterController2()}>
    <ControllerProvider create={() => new CounterController3()}>
      <ControllerProvider create={() => new CounterController4()}>
        <App />
      </ControllerProvider>,
    </ControllerProvider>,
  </ControllerProvider>,
</ControllerProvider>,

In such cases, the MultiProvider component can be utilized to reduce indentation:

import { MultiProvider } from 'react-state-view-controller';

<MultiProvider
  providers={[
    <ControllerProvider create={() => new CounterController()} />,
    <ControllerProvider create={() => new CounterController2()} />,
    <ControllerProvider create={() => new CounterController3()} />,
    <ControllerProvider create={() => new CounterController4()} />,
  ]}
>
  <App />
</MultiProvider>,

Both representations are equivalent. The nested components wrapped each other, granting access to the above provided resources if needed.

In situations where a single ControllerProvider requires access to already provided Controller to depend on it, consider separating it into a distinct component:

import { useProvider } from 'react-state-view-controller'

const DependentProvider = ({ children }) => {
  // We can get the CounterController here because it's provided is above this component.
  const controller = useProvider(CounterController)

  return <ControllerProvider create={() => new DependentController(controller)}>{children}</ControllerProvider>
}

Then this component can then be used as follows:

<MultiProvider
  providers={[
    <ControllerProvider create={() => new CounterController()} />,
    <DependentProvider />,
    <ControllerProvider create={() => new CounterController3()} />,
    <ControllerProvider create={() => new CounterController4()} />,
  ]}
>
  <App />
</MultiProvider>,

useAutoDispose

While ControllerProvider provides an effective way of DI, there are times that this is unecessary, as we just simply want to separate logic of a small view to a controller, dependent child components won't go that deeply nested, but still want to benefit from the auto-cleanup feature.

In this case, we can use the useAutoDispose hook:

const controller = useAutoDispose(() => new CounterController())

This controller instance will be kept with useRef, making it persist between re-render and will auto call the dispose function when current component is unmounted.

Controller created this way can also be feed into the source param of ProviderController mentioned above, specifically the second approach, which the source param take in an instance. Because the responsibility of persisting the controller instance and auto cleaning up feature is handled already by useAutoDispose.

Builder

To trigger the re-render process when new State is emitted from Controller within the same scope, you can use the Builder component.

<Builder
  // specify the class type to query
  source={CounterController}
  buildWhen={(prev, curr) => {
    // Optional: If this function is provided and returns `false`, the re-render trigger will be skipped.
    // We are provided with the previous state - the state that the component is using for rendering,
    // and the new state, which will potentially be used for rendering if we return true or omit
    // this function entirely.
  }}
>
  {(state: number, controller: CounterController) => {
    // render content based on state
    return <View></View>
  }}
</Builder>

This component can also take in directly an instance of Controller as the source instead of a type.

<Builder
  // passing an instance
  source={counterInstance}
  buildWhen={(prev, curr) => true}
>
  {(state: number) => {
    // render content based on state
    return <View></View>
  }}
</Builder>

There are hooks for all of this as well, in case you don't need to scope the re-render, but rather need to render a whole component, or you just prefer hook.

import { useBuilder } from 'react-state-view-controller'

// buildWhen is also provided
// type
const [state, controller] = useBuilder(CounterController, (prev, curr) => true)
// or instance
const state = useBuilder(counterInstance)

Selector

Usually, we don't need to watch for changes in the whole State, but rather just a portion of it, Selector can be used to do state-filtering.

// only re-render when state.count5 changed
<Selector source={MultiCounterController} selector={(state) => state.count5}>
  {(value, controller) => {
    // render content based on selected value
    return <View></View>
  }}
</Selector>

It also support direct controller instance variant

<Selector source={multiCounterInstance} selector={(state) => state.count5}>
  {(value) => {
    // render content based on selected value
    return <View></View>
  }}
</Selector>

Alternatively, we can use the useSelector hook

import { useSelector } from 'react-state-view-controller'

// only trigger re-render when `state.count5` changed
const [value, controller] = useSelector(MultiCounterController, (state) => state.count5)
// or the equivalent
const value = useSelector(multiCounterInstance, (state) => state.count5)

Listener

For triggering actions on the UI without causing a re-render, you can use this Component:

<Listener
  // specify the type
  source={CounterController}
  // callback when new state emitted
  listener={(state: number, controller: CounterController) => console.log(state)}
  // for state filter, similar with Builder
  listenWhen={(prev: number, current: number) => true}
>
  <View></View>
</Listener>

You will be provided with the listener callback to be called when the State changes. There is also listenWhen, similar to Builder.buildWhen, to filter changes in State to listen to as needed.

The controller instance variant:

<Listener
  source={counterInstance}
  listener={(state: number) => console.log(state)}
  listenWhen={(prev: number, current: number) => true}
>
  <View></View>
</Listener>

The hooks for this feature:

import { useListener } from 'react-state-view-controller'

const controller = useListener(
  CounterController,
  (state: number, controller: CounterController) => console.log(state), // callback when state changed
  (prev, curr) => true, // callback filter
)
// or
const controller = useListener(
  counterInstance,
  (state) => console.log(state), // callback when state changed
  (prev, curr) => true, // callback filter
)

Note that this hook is not intended by default to cause re-render, it just simply triggers callback

For other types

Of course, a view Controller is not the only thing that we need to DI in typical production app, we also have services, repositories, models,... or other types of data.

To acommondate these, you can check out the react-scoped-provider library, which is fully compatible with this library, in fact, the useProvider hook is just re-import and re-export from it.

Conclusion

Example

Please open an issue if you spot any problems or for discussions. I would be happy to receive feedback.