Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/gluecodes/storecle
A neat app state management and action orchestration for React and Solid.
https://github.com/gluecodes/storecle
react reactjs redux solid solidjs state-management
Last synced: 2 months ago
JSON representation
A neat app state management and action orchestration for React and Solid.
- Host: GitHub
- URL: https://github.com/gluecodes/storecle
- Owner: gluecodes
- License: mit
- Created: 2021-07-13T16:58:01.000Z (over 3 years ago)
- Default Branch: master
- Last Pushed: 2022-04-26T10:19:59.000Z (over 2 years ago)
- Last Synced: 2024-11-01T23:52:10.453Z (2 months ago)
- Topics: react, reactjs, redux, solid, solidjs, state-management
- Language: JavaScript
- Homepage:
- Size: 626 KB
- Stars: 27
- Watchers: 2
- Forks: 1
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE.md
- Code of conduct: CODE_OF_CONDUCT.md
Awesome Lists containing this project
README
# Storecle
**@gluecodes/storecle**
[![NPM Version](https://img.shields.io/npm/v/@gluecodes/storecle-solid.svg?style=flat)](https://www.npmjs.com/package/@gluecodes/storecle-solid)
A neat uni-directional app state management for [React](https://reactjs.org/) and [Solid](https://www.solidjs.com/) (:heart:).
## Features
Storecle uses a simple mental model which lets you access app-wide actions and their results by using Context API.
It consists of 4 main building blocks i.e. Store, User Actions (actions triggered by a user), Data Suppliers (actions executed prior to rendering) and Reload Types (action re-trigger groups).
The actions are just functions which are implicitly bound to the Store and write their results by returning/resolving.
Then, their results are accessible by their own names.To improve the code re-usability, Data Suppliers use a middleware pattern. They are executed in the order you specify and pass a snapshot of Store from one to another, letting you split the logic into small, specified functions.
- It works with both [React](https://reactjs.org/) and [Solid](https://www.solidjs.com/) (it's framework agnostic to certain degree).
- It uses Context API and `useEffect` / `createEffect` to provide action re-triggers based on specified Store changes.
- It facilitates splitting the business logic into granual, re-usable functions by applying a middleware pattern.
- It simplifies naming and reduces noise by letting you access action results by their own names.
- It provides an elegant approach to actions feeding UI with incoming data (e.g. from Web Sockets).
- It is made to work with your IDE's code auto-completion.## Motivation
I :heart: Redux, but it leaves plenty of room to be misused. Hence, Storecle is my proposal to let developers rely less on self-discipline and more on tooling and self-restrictive design.
1. To provide an easy way of separating app-wide logic from views i.e.:
- No inline: data fetches, transformers, conditionals.
- No nested action dispatchers upon other action completion.
2. To facilitate the action re-usability and modularization.
3. To provide a gradual path for [React](https://reactjs.org/) developers willing to use [Solid](https://www.solidjs.com/).## Installation
React:
```bash
yarn add @gluecodes/storecle-react
```or
```bash
npm i @gluecodes/storecle-react
```Solid:
```bash
yarn add @gluecodes/storecle-solid
```or
```bash
npm i @gluecodes/storecle-solid
```It works along with either [React](https://reactjs.org/) or [Solid](https://www.solidjs.com/) that also needs to be installed in your app. For details, see their own documentations.
## Usage
This module exports 3 constructs that can be imported for a particular framework in different parts of your app.
```javascript
import { builtInActions, PageProvider, useAppContext } from '@gluecodes/storecle-react'
```or
```javascript
import { builtInActions, PageProvider, useAppContext } from '@gluecodes/storecle-solid'
```For the purpose of the example I used a Solid version.
Soon the official starter templates will be released. Using this library means following certain patterns which are explained below using a simple counter example.
### Mental Model
> See: [Code Sandbox](https://codesandbox.io/s/bold-carlos-tj18g?file=/src/App.js) example for React.
> See: [Code Sandbox](https://codesandbox.io/s/awesome-hertz-jdcgg?file=/src/App.jsx) example for Solid.
File tree:
```
.
├── actions
│ ├── dataSuppliers (#2)
│ │ └── dataSuppliers.js
│ ├── reloadTypes.js (#4)
│ └── userActions (#3)
│ └── userActions.js
├── index.jsx (#1)
├── Layout.jsx (#5)
└── partials (#6)
└── Counter
└── Counter.jsx
```#### 1. Page Container
Page provider wraps a given Layout around a single app context.
- `dataSupplierPipeline` - an array providing the order in which Data Suppliers are executed.
- `dataSuppliers` - an object containing Data Suppliers.
- `getLayout` - a function which returns the page Layout.
- `reloadTypes` - an object containing Reload Types.
- `userActions` - an object containing User Actions.
- `onError` - a function triggered when an error is thrown either in Data Suppliers or User Actions.`./index.jsx`
```javascript
import { PageProvider } from '@gluecodes/storecle-solid'import * as dataSuppliers from './actions/dataSuppliers/dataSuppliers'
import * as userActions from './actions/userActions/userActions'
import * as reloadTypes from './actions/reloadTypes'import Layout from './Layout.jsx'
export default () => (
Layout}
reloadTypes={reloadTypes}
userActions={userActions}
onError={(err) => {
console.error(err)
}}
/>
)
```#### 2. Data Suppliers
Data suppliers provide data prior to rendering. Note the early returns which demonstrate how to resolve cached data based on Reload Type.
- `builtInActions` - an object containing the following built-in User Actions:
- `onStoreChanged` - a function which receives a callback to be triggered when Store changes.
- `runUserActions` - a function which allows for executing multiple User Actions at once.
- `runDataSuppliers` - a function which receives a Reload Type name. Note that it's exposed to ease the integration with legacy apps. Don't call it manually as Data Suppliers are implicitly reloaded based on the provided Reload Types.
- Each Data Supplier passes two arguments: `resultOf` and `nameOf`.
- `resultOf` - a function providing a result of a given Data Supplier or User Action.
- `nameOf` - a function providing a name of either Data Supplier, User Action or Reload Type.
- Data Suppliers can be either sync or async and write to a central Store by returning/resolving.`./actions/dataSuppliers/dataSuppliers.js`
```javascript
import { builtInActions } from '@gluecodes/storecle-solid'
import { reFetchCounter } from '../reloadTypes'export function getCounter(resultOf, nameOf) {
const reloadType = resultOf(builtInActions.runDataSuppliers)
const shouldFetch = reloadType === 'full' || reloadType === nameOf(reFetchCounter)if (!shouldFetch) {
return resultOf(getCounter)
}return global.sessionStorage.getItem('appWideCounter') || 0
}export function getTexts(resultOf) {
if (resultOf(builtInActions.runDataSuppliers) !== 'full') {
return resultOf(getTexts)
}return {
Click: 'Click'
}
}
```#### 3. User Actions
Actions triggered by a user.
`./actions/userActions/userActions.js`
```javascript
export function incrementCounter(counter) {
const incrementedCounter = Number(counter) + 1global.sessionStorage.setItem('appWideCounter', incrementedCounter)
}
```#### 4. Reload Types
A way to tell the app to re-run Data Suppliers based on executed User Actions.
- A Reload Type groups User Actions together to tell the app to reload all Data Suppliers as a consequence of their execution.
- When any of its User Actions is triggered, the app sets the Reload Type name under built-in `runDataSuppliers` and reloads all Data Suppliers.
- Data Suppliers can benefit from caching by early returning their results based on Reload Type name.
- Each Reload Type is a function which passes `nameOf` and returns an array of User Action names.
- `nameOf` - a function providing a name of User Action.`./actions/reloadTypes.js`
```javascript
import { incrementCounter } from './userActions/userActions'export const reFetchCounter = (nameOf) => [nameOf(incrementCounter)]
```#### 5. Layout
Nothing else than the page layout.
`./Layout.jsx`
```jsx
import Counter from './partials/Counter/Counter.jsx'export default () => (
<>
>
)
```#### 6. Partials
Partials are self-contained pieces of UI which have access to app state via the app context.
- `useAppContext` - a function which returns an array of 3 items: `resultOf`, `action`, `nameOf`.
- `resultOf` - a function providing a result of a given Data Supplier or User Action.
- `action` - a function which triggers User Action.
- `nameOf` - a function providing a name of either Data Supplier or User Action.`./partials/Counter/Counter.jsx`
```jsx
import { useAppContext } from '@gluecodes/storecle-solid'import { getCounter, getTexts } from '../../actions/dataSuppliers/dataSuppliers'
import { incrementCounter } from '../../actions/userActions/userActions'export default () => {
const [resultOf, action] = useAppContext()return (
{
action(incrementCounter)(resultOf(getCounter))
}}
>
{resultOf(getTexts)?.Click}: {resultOf(getCounter)}
)
}
```## Documentation
WIP, so far there is only this `README.md` and a project `./test/env`. More docs will come with starter templates and CLI tooling.
## License
[MIT](https://github.com/gluecodes/storecle/blob/master/LICENSE.md)