Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/mrozbarry/hyperapp-actionpack
Make hyperapp actions easy to manage, compose, and execute.
https://github.com/mrozbarry/hyperapp-actionpack
es6 functional hyperapp javascript module
Last synced: 3 months ago
JSON representation
Make hyperapp actions easy to manage, compose, and execute.
- Host: GitHub
- URL: https://github.com/mrozbarry/hyperapp-actionpack
- Owner: mrozbarry
- Created: 2024-01-31T14:29:59.000Z (11 months ago)
- Default Branch: master
- Last Pushed: 2024-05-21T13:51:45.000Z (8 months ago)
- Last Synced: 2024-05-22T12:47:03.970Z (8 months ago)
- Topics: es6, functional, hyperapp, javascript, module
- Language: JavaScript
- Homepage: https://github.com/mrozbarry/hyperapp-actionPack
- Size: 20.5 KB
- Stars: 0
- Watchers: 1
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
Awesome Lists containing this project
README
# Hyperapp Action Pack
This is a proof of concept of how to build [hyperapp](https://github.com/jorgebucaran/hyperapp) actions in a container that you can manage globally.
See [composable-state](https://github.com/mrozbarry/composable-state) for documentation on state updates.
## Install
```sh
npm install --save hyperapp-actionPack
```## Example
### Simplest
How to wire your actions into your app.
```js
import ActionPack from 'hyperapp-actionPack';
import { app, h, text } from 'hyperapp';
import { select, replace } from 'composable-state';const actions = new ActionPack();
actions.declare('++', (props) => composable.select('counter', composable.replace(old => old + 1)));
actions.declare('--', (props) => composable.select('counter', composable.replace(old => old - 1)));app({
init: {
counter: 0,
},view: (state) => {
return h('div', { style: { display: 'flex', justifyContent: 'center', alignItems: 'center' } }, [
h('button', { type: 'button', onclick: actions.act('--') }, text('-1')),h('div', {}, text(`${state.counter}`)),
h('button', { type: 'button', onclick: actions.act('++') }, text('+1')),
]);
},
});
```### Conditional updates based on state
Get access to the global state to decide how to properly update the state.
```js
import ActionPack from 'hyperapp-actionPack';
import { select, replace, collect } from 'composable-state';const actions = new ActionPack();
actions.declare('play-sound', (props, state) => (
state.enabled
? select('audioSrc', props.audioSrc)
: collect([])
));
```### Composable exposed in stateMutators
As a convenience, all composable-state mutators are available as a third parameter to your state-mutator method.
```js
import ActionPack from 'hyperapp-actionPack';const actions = new ActionPack();
actions.declare('++', (props, _state, { select, replace }) => select('counter', replace(old => old + 1)));
```### With effects
And of course, your actions can schedule side-effects, too.
```js
import ActionPack from 'hyperapp-actionPack';const actions = new ActionPack();
const effectFx = (dispatch, props) => {
console.log(`Hello ${props.name} from my side-effect`);
};
const effect = (props) => [effectFx, props];actions.declare('runMyEffect', (props, _state, { collect }) => [
collect([]),
effect({ name: 'world' }),
]));
```### Chaining actions
Chaining actions together has never been easier
```js
import ActionPack from 'hyperapp-actionPack';const actions = new ActionPack();
actions.declare('step1', (props, _state, { collect }) => [
collect([]),
actions.andThen('step2', props),
]);actions.declare('step2', (props, _state, { collect }) => collect([]));
```### Debugging
For now, debugging is strictly console/devtools based.
To turn it on, just pass the global `console` object into the constructor.
In the future, I may write proper debug/devtools adapter using the console api.```js
import ActionPack from 'hyperapp-actionPack';const actions = new ActionPack(console);
actions.declare('++', (props, _state, { select, replace }) => (
select('counter', replace(old => old + 1))
);
```### It sets up a singleton
If you want to split up your actions across multiple file, you can do-so easily with the handle singleton static method.
**actions1.js**
```js
import ActionPack from 'hyperapp-actionPack'const actions = ActionPack.singleton();
actions.declare('foo', (_props, _state, { select, replace }) => select('name', replace('foo')));
```**actions2.js**
```js
import ActionPack from 'hyperapp-actionPack'const actions = ActionPack.singleton();
actions.declare('bar', (_props, _state, { select, replace }) => select('name', replace('bar')));
```**app.js**
```js
import ActionPack from 'hyperapp-actionPack';
import { app, h, text } from 'hyperapp';const actions = ActionPack.singleton();
app({
init: { name: null },view: (state) => h('div', {}, [
h('div', {}, `Hello${state.name ? `, ${state.name}` : ''}`),
h('button', { type: 'button', onclick: actions.act('action1.foo') }, text('Foo')),
h('button', { type: 'button', onclick: actions.act('action2.bar') }, text('Bar')),
]),
});
```