I’m working on a React Flux project where async interaction is an ongoing thing and I’m still a bit wonky when it comes to the react lifecycle.
In my statefull components I have a getStateFromStores function, if there is no state in store, this returns a false bool, otherwise it returns the state.
As I have it right now the cycle is as following:
I have one bound function (ES6 react class) in my components called getState(), this simply invokes getStateFromStores and checks that the return is not false, and if not, invokes setState() with the state.
Since notifications of state change can come from different sources, I like to declare the logic in one place (getState)
State change can come from the following:
ComponentDidMount (should I move this to initialState?) This is the first thing that happens in the components lifecycle. (And unless state already exists in our store, we get false in return)
ComponentWillReceiveProps If state is changed in parent component, we notify the component that it should fetch new state with the new props.
Store change listener When store has new state, getState is invoked as the callback (although a bit problematic, because emit can happened for reasons other than as a consequence of the given component’s call – so I might implement an onStoreChange as the callback who in turn calls getState)
Im toying around with a state prop called fetching which I can use to control the flow, but is this problematic? – because every time I change state it triggers a render (?) — could I use static for this?
I imagine something of the following; DidMount triggers getState(), getState sets fetching to true, and now all other invoke-tries will fail because of the bool check.
Then I’d implement a resetState, that brings back the state to default (fetching false) and then getState can be triggered again.
I’d abstract all this functionality in a “Higher-Order component”
Thanks!
What I've found working really well for me in most cases is to have a root "controller component". Let's call it
Controller
for now. It's basically a React component just as any other – but its responsibility differs a bit.I see it as solely delegating. It's only responsibility is to get data (state) from the stores and pass it down (as props) to its child components and let them do all the work.
This is done by setting the data from the stores as the
Controller
's state. Any changes in the stores are reported back to (only) theController
, which updates its state. Since the state is passed down as props to the children, a change in a store will trigger all child components using that data to re-render (if they have to).I try to pass down only relevant data to the child components, but don't worry to much about them getting to much data (or about some extra re-rendering).
In practice the
Controller
would look something like this: