In my opinion, a robust state management system should effectively adhere to the following criteria:
-
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 aView
should align with theView
's lifecycle. When the screen is mounted, theState
andController
managed it is created and evolves throughout theView
lifecycle, triggering UI layer re-rendering when it changes. When theView
is unmounted, the managingController
andState
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 emitState
as necessary. - Reusability: One of the advantages is the ease of reusing logic between different
Controllers
.
- Easier Dependency Injection (DI): We can create a network of interdependent
-
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. TheContext
API is a good example of this mechanism, as it allows access to data without having to pass props down to every child component. -
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 theState
object is greater than5
.
This library is created to address the concerns mentioned above.
When defining a Controller
for a React view or view group, adopting a subclass approach brings several advantages:
Controller
instances can seamlessly subscribe to and depend on each other, enabling internal triggers for state emissions.
Utilizing inheritance facilitates the extension of a Controller
, promoting the creation of more reusable code.
Creating, accessing, and modifying properties within a Controller
becomes a straightforward process.
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.
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.
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>,
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>,
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
.
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)
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)
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
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.
Please open an issue if you spot any problems or for discussions. I would be happy to receive feedback.