react-model ·

The State management library for React
🎉 Support Hooks Api
👬 Fully TypeScript Support
📦 gzip bundle < 2KB with microbundle
⚙️ Middlewares Pipline ( redux-devtools support ... )
Table of Contents
Quick Start
Next.js + react-model work around
install package
npm install react-model
Core Concept
Model Register
react-model keep the state and actions in a global store. So you need to register them before using.
model/index.model.ts
import { Model } from 'react-model'
import Home from '../model/home.model'
import Shared from '../model/shared.model'
const models = { Home, Shared }
export const { getInitialState, useStore, getState } = Model(models)
export type Models = typeof models
⇧ back to top
useStore
The functional component in React 16.7.0-alpha.2 can use Hooks to connect the global store.
import React from 'react'
import { useStore } from '../index.model'
export default () => {
const [state, actions] = useStore('Home')
const [sharedState, sharedActions] = useStore('Shared')
return (
<div>
Home model value: {JSON.stringify(state)}
Shared model value: {JSON.stringify(sharedState)}
<button onClick={e => actions.increment(33)}>home increment</button>
<button onClick={e => sharedActions.increment(20)}>
shared increment
</button>
<button onClick={e => actions.get()}>fake request</button>
<button onClick={e => actions.openLight()}>fake nested call</button>
</div>
)
}
⇧ back to top
Model
Every model have their own state and actions.
const initialState = {
counter: 0,
light: false,
response: {} as {
code: number
message: string
}
}
type StateType = typeof initialState
type ActionsParamType = {
increment: number
openLight: undefined
get: undefined
}
const Model: ModelType<StateType, ActionsParamType> = {
actions: {
increment: async (state, _, params) => {
return {
counter: state.counter + (params || 1)
}
},
openLight: async (state, actions) => {
await actions.increment(1)
await actions.get()
actions.get()
await actions.increment(1)
await actions.increment(1)
await actions.increment(1)
return { light: !state.light }
},
get: async () => {
await new Promise((resolve, reject) =>
setTimeout(() => {
resolve()
}, 3000)
)
return {
response: {
code: 200,
message: `${new Date().toLocaleString()} open light success`
}
}
}
},
state: initialState
}
export default Model
⇧ back to top
getState
Key Point: State variable not updating in useEffect callback
To solve it, we provide a way to get the current state of model: getState
Hint: The state returned should only be used as readonly
import { useStore, getState } from '../model/index.model'
const BasicHook = () => {
const [state, actions] = useStore('Counter')
useEffect(() => {
console.log('some mounted actions from BasicHooks')
return () =>
console.log(
`Basic Hooks unmounted, current Counter state: ${JSON.stringify(
getState('Counter')
)}`
)
}, [])
return (
<>
<div>state: {JSON.stringify(state)}</div>
</>
)
}
⇧ back to top
Advance Concept
immutable Actions
The actions use immer produce API to modify the Store. You can return a producer in action.
TypeScript Example
const Model: ModelType<StateType, ActionsParamType> = {
actions: {
increment: async (s, _, params) => {
return (state: typeof s) => {
state.counter += params || 1
}
},
decrease: (s, _, params) => s => {
s.counter += params || 1
}
}
}
JavaScript Example
const Model = {
actions: {
increment: async (s, _, params) => {
return state => {
state.counter += params || 1
}
}
}
}
⇧ back to top
SSR with Next.js
shared.model.ts
const initialState = {
counter: 0
}
const Model: ModelType<StateType, ActionsParamType> = {
actions: {
increment: (state, _, params) => {
return {
counter: state.counter + (params || 1)
}
}
},
asyncState: async () => {
await waitFor(4000)
return { counter: 500 }
},
state: initialState
}
_app.tsx
import { models, getInitialState, Models } from '../model/index.model'
let persistModel: any
interface ModelsProps {
initialModels: Models
persistModel: Models
}
const MyApp = (props: ModelsProps) => {
if (!(process as any).browser) {
persistModel = Model(models, props.initialModels)
} else {
persistModel = props.persistModel || Model(models, props.initialModels)
}
const { Component, pageProps, router } = props
return (
<Container>
<Component {...pageProps} />
</Container>
)
}
MyApp.getInitialProps = async (context: NextAppContext) => {
if (!(process as any).browser) {
const initialModels = await getInitialState()
return { initialModels }
} else {
return { persistModel }
}
}
hooks/index.tsx
import { useStore, getState } from '../index.model'
export default () => {
const [state, actions] = useStore('Home')
const [sharedState, sharedActions] = useStore('Shared')
return (
<div>
Home model value: {JSON.stringify(state)}
Shared model value: {JSON.stringify(sharedState)}
<button
onClick={e => {
actions.increment(33)
}}
>
</div>
)
}
⇧ back to top
Middleware
We always want to try catch all the actions, add common request params, connect Redux devtools and so on. We Provide the middleware pattern for developer to register their own Middleware to satisfy the specific requirement.
const tryCatch: Middleware<{}> = (context, restMiddlewares) => {
const { next } = context
next(restMiddlewares).catch((e: any) => console.log(e))
}
let actionMiddlewares = [
tryCatch,
getNewState,
setNewState,
stateUpdater,
communicator,
devToolsListener
]
const consumerAction = (action: Action) => async (params: any) => {
const context: Context = {
modelName,
setState,
actionName: action.name,
next: () => {},
newState: null,
params,
consumerActions,
action
}
applyMiddlewares(actionMiddlewares, context)
}
export { ... , actionMiddlewares}
⚙️ You can override the actionMiddlewares and insert your middleware to specific position
⇧ back to top
Other Concept required by Class Component ( Not First Class, ONLY SUPPORT ON CSR, Welcome to PR )
Provider
The global state standalone can not effect the react class components, we need to provide the state to react root component.
import { PureComponent } from 'react'
import { Provider } from 'react-model'
class App extends PureComponent {
render() {
return (
<Provider>
<Counter />
</Provider>
)
}
}
⇧ back to top
connect
We can use the Provider state with connect.
Javascript decorator version
import React, { PureComponent } from 'react'
import { Provider, connect } from 'react-model'
const mapProps = ({ light, counter }) => ({
lightStatus: light ? 'open' : 'close',
counter
})
@connect(
'Home',
mapProps
)
export default class JSCounter extends PureComponent {
render() {
const { state, actions } = this.props
return (
<>
<div>states - {JSON.stringify(state)}</div>
<button onClick={e => actions.increment(5)}>increment</button>
<button onClick={e => actions.openLight()}>Light Switch</button>
</>
)
}
}
TypeScript Version
import React, { PureComponent } from 'react'
import { Provider, connect } from 'react-model'
import { StateType, ActionType } from '../model/home.model'
const mapProps = ({ light, counter, response }: StateType) => ({
lightStatus: light ? 'open' : 'close',
counter,
response
})
type RType = ReturnType<typeof mapProps>
class TSCounter extends PureComponent<
{ state: RType } & { actions: ActionType }
> {
render() {
const { state, actions } = this.props
return (
<>
<div>TS Counter</div>
<div>states - {JSON.stringify(state)}</div>
<button onClick={e => actions.increment(3)}>increment</button>
<button onClick={e => actions.openLight()}>Light Switch</button>
<button onClick={e => actions.get()}>Get Response</button>
<div>message: {JSON.stringify(state.response)}</div>
</>
)
}
}
export default connect(
'Home',
mapProps
)(TSCounter)
⇧ back to top