elem-vdom-vanilla-experiment
v0.0.18
Published
Simple tool to build UIs
Downloads
3
Maintainers
Readme
elem-vdom-vanilla-experiment
Simple and idiotic lib to build UI components. It's a component library promoting functional composition with the full expressiveness of JavaScript and support for all existing JavaScript libraries. elem-vdom-vanilla-experiment is just a quick and dirty experiment to avoid string templates, string concat and manual mutations when modifying the DOM. elem-vdom-vanilla-experiment
is written in using ES2015 syntax.
This branch is an experiment to only use pure DOM API. No virtual DOM technique here
Install
with npm do :
npm install elem-vdom-vanilla-experiment --save
Build
If you want to build the project yourself, you need to install Node 4
and do the following :
git clone https://github.com/mathieuancelin/elem-vdom.git elem-vdom-vanilla-experiment
cd elem-vdom-vanilla-experiment
git fetch --all
git checkout puredom
npm install
npm run build
Examples
If you want to run examples, just clone the project then :
git clone https://github.com/mathieuancelin/elem-vdom.git elem-vdom-vanilla-experiment
cd elem-vdom-vanilla-experiment
git fetch --all
git checkout puredom
npm install
npm start
open http://localhost:8080/examples
API
Elem.el(name, attributes, children)
: Create a representation of an HTML element. Children can be a string/number/boolean, anElem.el
, an array ofElem.el
or a__asHtml
object.Elem.svg(name, attributes, children)
: Create a representation of a simple SVG elementElem.render(elem, container)
: render an element to a container in the DOMElem.renderToString(elem)
: render an element as an HTML stringElem.renderToJson(elem)
: render an element as JSON objectElem.findDOMNode(ref)
: return an actual DOM node based on an elementref
Elem.stylesheet(obj)
: create an extendable set of CSS inline stylesElem.nbsp(times)
: creates a<span>
containing one or more
Elem.text(value)
: creates a<span>value</span>
Elem.predicate(predicate, elem)
: return element if predicate is true or undefined if false. Predicate can be a function
A few other APIs are also available but none are mandatory to use :
Elem.Perf
: performance measurement tools (used in the examples to craft the Perf monitor)Elem.Store
: tools to create flux like store (heavily inspired by Redux)Elem.Devtools
: tools for DX
Dependencies
elem-vdom-vanilla-experiment depends on babel-runtime
and error-stack-parser
but these dependencies are bundled in the distribution package.
How can I use elem-vdom ?
First imports elem.js
in your page. Then you will be able to build your first node
const MyAwesomeNode = Elem.el('h1', 'Hello World!');
Elem.render(MyAwesomeNode, '#container');
Of course, you can build much more complicated nodes
const node = Elem.el('div', { className: 'col-md-6' }, [
Elem.el('h3', 'Hello World!'),
Elem.el('p', { style: { backgroundColor: 'red' } }, "Lorem ipsum ....")
]);
Elem.render(node, '#container');
As you construct the node tree with functions and arrays, it is pretty easy to map and filter model objects to render your components easily.
Attributes use camel case shaped keys, so something like backgroundColor
will be rendered as background-color
. Also, you can notice that the class
attribute is named className
. Also, you can provide an object for className
value with boolean as values. Every key with a false value will not be rendered.
const shouldDisplayDarkBackground = true;
const shouldDisplayBrightBackground = !shouldDisplayDarkBackground;
Elem.el('div', {
className: {
withBackground: true,
darkBackground: shouldDisplayDarkBackground,
brighBackground: shouldDisplayBrightBackground
}
}, 'Hello');
will produce
<div class="with-background dark-background">Hello</div>
As children are just nodes in an array, it is really easy to add or remove elements from your UI. You can also pass undefined elements or functions that can return undefined to not render nodes.
If you want to provide a child as HTML value, just pass an object like {__asHtml: ' ;-)'}
.
You can also attach callback to event on elements like
function saySomething() {
alert("Something !");
}
const node = Elem.el('div', { className: 'col-md-6' }, [
Elem.el('h3', 'Hello World!'),
Elem.el('button',
{
className: ['btn', 'btn-primary'],
onclick: saySomething
},
'Say something'
),
Elem.el('p', { style: { backgroundColor: 'red' } }, "Lorem ipsum ....")
]);
Elem.render(node, '#container');
And no, the output WILL NOT BE
<div class="col-md-6">
<h3>Hello World</h3>
<button class="btn btn-primary" onclick="saySomething">Say Something</button>
<p style="background-color: red;">Lorem ipsum ....</p>
</div>
but the following with an event listener on the root element of the component listening to click
events on the button.
<div class="col-md-6">
<h3>Hello World</h3>
<button class="btn btn-primary">Say Something</button>
<p style="background-color: red;">Lorem ipsum ....</p>
</div>
Supported events are
wheel scroll touchcancel touchend touchmove touchstart click doubleclick
drag dragend dragenter dragexit dragleave dragover dragstart drop
change input submit focus blur keydown keypress keyup copy cut paste
SVG
You can also simply use SVG with elem-vdom-vanilla-experiment, using the dedicated API :
function svg() {
return Elem.svg('svg', { xmlns: Elem.svgNS, version: "1.1", width: "300", height: "200" }, [
Elem.svg('title', 'Simple SVG pict'),
Elem.svg('desc', "A rectangle, a line and a circle"),
Elem.svg('rect', { width: 100, height: 80, x: 0, y: 70, fill: "green" }),
Elem.svg('line', { x1: "5", y1: "5", x2: "250", y2: "95", stroke: "red" }),
Elem.svg('circle', { cx: "90", cy: "80", r: "50", fill: "blue" }),
Elem.svg('text', { x: "180", y: "60" }, 'A text')
]);
}
Elem.render(svg, document.getElementById("svg"));
I just want functions everywhere man ...
Pretty easy actually, Elem is made for that :-)
function DateField() {
return Elem.el('div', [
Elem.el('h1', moment().format(this.props.format))
]);
}
function TimeField() {
return Elem.el('div', [
Elem.el('h2', moment().format(this.props.format))
]);
}
function GraphicalClock() {
const hoursRotation = 'rotate(' + (30 * moment().hours()) + (moment().minutes() / 2) + ')';
const minutesRotation = 'rotate(' + (6 * moment().minutes()) + (moment().seconds() / 10) + ')';
const secondsRotation = 'rotate(' + (6 * moment().seconds()) + ')';
return Elem.el('div', { className: "clock", style: { width: `${this.props.width}px`, height: `${this.props.height}px` } }, [
Elem.svg('svg', { xmlns: Elem.svgNS, version: "1.1", viewBox: "0 0 100 100"}, [
Elem.svg('g', { transform: "translate(50,50)" }, [
Elem.svg('circle', { className: "clock-face", r: "48", fill: 'white', stroke: '#333' }),
Elem.svg('line', { className: "hour", y1: "2", y2: "-20", transform: hoursRotation }),
Elem.svg('line', { className: "minute", y1: "4", y2: "-30", transform: minutesRotation }),
Elem.svg('g', { transform: secondsRotation }, [
Elem.svg('line', { className: "second", y1: "10", y2: "-38" }),
Elem.svg('line', { className: "second-counterweight", y1: "10", y2: "2" })
])
])
])
]);
}
function Clock() {
if (this.context.interval === null) {
this.context.interval = setInterval(this.redraw, 1000);
}
return Elem.el('div', { style: { display: 'flex' } }, [
Elem.el('div', { style: { display: 'flex', flexDirection: 'column' } }, [
Elem.el(DateField, { format: 'DD/MM/YYYY' }),
Elem.el(TimeField, { format: 'HH:mm:ss' })
]),
Elem.el(GraphicalClock, { width: 120, height: 120 })
]);
}
Elem.render(Clock, container);
The Elem.render
context
Stateless functions as components is the key pattern of elem-vdom-vanilla-experiment
. In fact, it's better if your components are dumb and just need some props to work, the application data being provided by an external store. But sometimes you need more.
Each time you call Elem.render
with a component function as first parameter, an application context will be created and passed to the component function and all the sub component function. The context is available as this
or as a function parameter. The context will also be attached to any event handler (using this
) attached by elem-vdom-vanilla-experiment
.
function MyComponent(ctx, props) { // app context as parameter
this. .... // app context as this
}
The context is unique for the root component function (and all the sub functions) and will be common to the whole component tree. This context contains everything needed to trigger a redraw of the root component function (and the whole sub tree), and different ways to attach data to the tree. These ways are :
props
state
context
The props
are just properties passed to components (functions). These properties can change for each render, but are considered immutable during the calls. Each component function have its own props
, and these are not shared with the whole sub tree.
The state
is an object that will contains mutable data that represent internal state of the whole component (created with Elem.render
). You can read data directly on the object but to mutate it, you have to call setState(diff)
or replaceState(newState)
. Each call to these function will trigger a redraw of the root component and its sub tree.
The context
is a simple object that you can manipulate the way you want. It's a good way to provide common services to a component tree.
The API of the Elem.render
context is the following :
props
: props of the current component functionrefs
: refs of DOM nodes inside the current render treestate
: mutable state of the current render treerefresh()
orredraw()
: rerender the current function at the same placesetState(diff)
: mutate the state with diff and trigger a refreshreplaceState(newState)
: mutate the whole state and trigger a refreshgetDOMNode()
: return root DOM node of the current render treecontext
: a multipurpose object shared by all the components in the treewithInitialState(function)
: define an initial state value, the arg function will be called oncewithDefaultProps(function)
: define default props values, the arg function will be called oncewithInitialContext(function)
: define an context, the arg function will be called once
If you want a component (a function) to get its own state, you just have to assign a key to it.
Elem.el(MyButton, { key: "button1" })
Then the this.state
passed to the function MyButton
will be its own. You just have to be careful to assign the same key to the same component each time you render the tree so elem-vdom-vanilla-experiment
can track down the right sub state and pass it to the component.
You can also specify an initialState to the sub-state of a component
function MyButton() {
this.withInitialState({ done: false }); // only set once
...
}
it work also with the main render state
function MyButton() {
this.withInitialState({ done: false }); // only set once
...
}
Elem.render(MyButton, container);
But, I like jsx syntax, how can I use it ?
If you use babel, add jsxPragma=Elem.jsx to options in your .babelrc or babel-loader.
see
https://github.com/mathieuancelin/elem-vdom/blob/master/.babelrc#L5
or
https://github.com/mathieuancelin/elem-vdom/blob/master/webpack.config.js#L53
then you can write stuff like :
import Elem from 'elem-vdom-vanilla-experiment';
function Bordered() {
return (
<div style={{ borderStyle: 'solid', borderColor: this.props.color || 'black', borderWidth: this.props.width || '1px'}}>
{this.props.children}
</div>
);
}
function Child() {
return <small>Just a child</small>;
}
function Parent() {
return (
<div>
<h1>Hello World!</h1>
<Bordered color="blue">
<Child />
</Bordered>
</div>
);
}
Elem.render(Parent, document.getElementById('app'));
Really, there is no 'actual' component API ?
Actually, there is one
const Clicker = Elem.createComponent({
name: 'Clicker',
init() {
// do whatever you want here
},
getInitialState() {
return {
count: 0
};
},
getDefaultProps() {
return {
text: 'You clicked'
};
},
getRootContext() {
return {
clickerService: ClickerService.create(API_KEY)
};
},
render() {
const click = () => {
this.context.clickerService.click();
this.setState({ state: this.state.count + 1 });
};
return (
<div>
<span>{this.props.text} : {this.state.count}</span>
<button type="button" onClick={click}>Click</button>
</div>
);
}
});
Elem.render(Clicker, container);
// or
Elem.render(<div><Clicker text="Vous avez cliqué" /></div>, container);
you can also use module syntax
export const name = 'Clicker';
export function init() {
// do whatever you want here
}
export function getInitialState() {
return {
count: 0
};
}
export function getDefaultProps() {
return {
text: 'You clicked'
};
}
export function getRootContext() {
return {
clickerService: ClickerService.create(API_KEY)
};
}
function click() {
this.context.clickerService.click();
this.setState({ state: this.state.count + 1 });
}
export function render() {
return (
<div>
<span>{this.props.text} : {this.state.count}</span>
<button type="button" onClick={click}>Click</button>
</div>
);
}
to use it
import * as ClickerComponent from './clicker';
const Clicker = Elem.createComponent(ClickerComponent);
Elem.render(Elem.createComponent(ClickerComponent), container);
// or
Elem.render(<div><Clicker text="Vous avez cliqué" /></div>, container);
But, how can I get an actual DOM node from inside my component ?
That's pretty easy, you just have to use refs. Refs give you access to any node inside your component that has been marked with a ref
parameter.
function MyComponent() {
const clickMe => () {
console.log(Elem.findDOMNode(this.refs.myInputText).value);
}
return Elem.el('div', [
Elem.el('input', { type: 'text', ref: 'myInputText', value: 'Hello World!' }, []),
Elem.el('button',
{ type: 'button', className: 'btn btn-primary', onclick: clickme },
'Click me !!!')
]);
}
Elem.render(MyComponent, container);
But you can't render that stuff server side (universal apps), right ?
Actually you can and it's pretty easy.
First you can use Elem.renderToString
on any Elem.el
node you want.
But you can also do the same on components, let's write a funny clock component;
export default function Clock() {
const seconds = (moment().seconds() % 60) * 6;
const minutes = (moment().minutes() % 60) * 6;
const hours = (moment().hours() % 12) * 30;
return Elem.el('div', { className: 'circle'}, [
Elem.el('div', { className: 'hour',
style: { transform: `rotate(${this.state.hours}deg)` }}, ''),
Elem.el('div', { className: 'minute',
style: { transform: `rotate(${this.state.minutes}deg)` }}, ''),
Elem.el('div', { className: 'second',
style: { transform: `rotate(${this.state.seconds}deg)` }}, ''),
Elem.el('span', { className: 'centered' },
`${moment().hours()} h ${moment().minutes()} m ${moment().seconds()} s`)
]);
}
Now we can instanciate it on the server side, and render it as an HTML string :
const express = require('express');
const app = express();
const Clock = require('./clock');
app.get('/clock.html', (req, res) => {
res.send(Elem.renderToString(Clock));
});
const server = app.listen(3000, () => {
const host = server.address().address;
const port = server.address().port;
console.log(`Clock app listening at http://${host}:${port}`);
});
on the client side, you just have to re-render the component at the same div dans Elem while re-attach itself generated DOM.
What about webcomponents ?
Just use Elem.registerWebComponent(name-with-a-dash, component)
and use it like an average HTML element
<my-awesomecomponent></my-awesomecomponent>
How can I test my components
We recommand to write your tests with mocha
and chai
with a headless browser named jsdom
. You will need to use Node 4
instead of node
because jsdom
requires it. First, install everything :
npm install --save-dev mocha chai jsdom simulant
then create a folder for your tests and create a main file that will run all the tests
const env = require('elem-vdom-vanilla-experiment/lib/test/env');
env.setupEnv();
const tests = [
require('./myfirsttest.js'),
require('./mysecondtest.js')
];
you need to do so, because the setupEnv
function need to be called before the first loading of Elem
. This will setup everything you need to create global window
and document
objects based on working with jsdom
. If you want to use a real browser or something like phantomjs
, you don't need to call setupEnv
, just use an HTML document with a <div id="app"></div>
.
Then in you test file, you can test the behavior of your components :
import { expect } from 'chai';
import Elem from 'elem-vdom-vanilla-experiment';
import DOM from 'elem-vdom-vanilla-experiment/lib/test/dom';
import { it } from 'elem-vdom-vanilla-experiment/lib/test/desc';
import Clicker from '../components/clicker';
describe('my awsome cliker component', () => {
it('can be clicked', () => {
DOM.renderComponent(Clicker);
expect(DOM.html('#label')).to.be.equal('0');
DOM.click('button');
expect(DOM.html('#label')).to.be.equal('1');
DOM.click('button');
expect(DOM.html('#label')).to.be.equal('2');
});
});
Here we use a custom it
function that is written like :
export function it(what, block) {
global.it(what, done => {
DOM.cleanup();
block(DOM, expect);
DOM.cleanup();
done();
});
}
so if you need to write an async test and use the done
function when you need it, don't use that, but don't forget to call DOM.cleanup()
to be sure to work on a clean page.
The DOM API is just a bunch of helpers to find nodes in the current document. If you're comfortable with those API, you are not forced to use the DOM API. Just mount your component in the #app node and don't forget to unmount everything between each test. The API is the following :
appHtml()
: return '#app' innerHTMLhtml(of)
: return innerHTML of the selected nodechildren(of)
: return children of the selected node as an arraynode(of)
: return selected nodenodes(of)
: return selected nodes as an arrayrenderComponent(component)
: render anElem
component in the '#app' nodecleanup()
: empty the '#app' nodeon(selector).simulate('keyup', { charCode: 42 })
: fire an event on the selected nodechange(selector, value)
: fire a change event on the selected inputclick(selector)
: click on the selected nodeselect(selector)
: return a fluent API to assert on selected nodeid()
get()
: return the id of the selected nodeshouldBe(value)
: assert on the id of the selected node
className()
get()
: return the className of the selected nodeshouldBe(value)
: assert on the className of the selected node
html()
get()
: return the innerHTML of the selected nodeshouldBe(value)
: assert on the innerHTML of the selected node
children()
get()
: return the children of the selected nodeshouldBe(value)
: assert on the children of the selected nodecount()
get()
: return the number of children of the selected nodeshouldBe(value)
: assert on the number of children of the selected node
click()
: fire a click event on the selected nodechange(value)
: fire a change event on the selected nodeshouldExist()
: assert if the selected node existsshouldNotExist()
: assert if the selected node does not existsshouldBe(value)
: assert if the innerHTML of the parent of the selected node is equals to the value (string or el)simulate(name, event)
: simulate any event on the selected node
About Elem.predicate
You can use Elem.predicate
to render element if a predicate is true
const somethingTrue = true;
function somethingFalse() {
return false;
}
Elem.render(Elem.el('ul', [
Elem.predicate(somethingTrue, Elem.el('li', '1')),
Elem.el('li', '2'),
Elem.predicate(somethingFalse, Elem.el('li', '3')),
Elem.el('li', '4')
]), container);
will produce :
<ul>
<li>1</li>
<li>2</li>
<li>4</li>
</ul>
About Elem.stylesheet
The Elem.stylesheet
function allows you to create something like
var Style = Elem.stylesheet({
circle: {
borderRadius: '50%',
width: '120px',
height: '120px',
marginLeft: '10px',
marginTop: '10px',
background: 'white',
border: '3px solid #61b2a7',
position: 'relative',
},
circleCentered: {
position: 'absolute',
top: '55%',
left: '0px',
width: '120px',
textAlign: 'center',
fontFamily: "'Montserrat',sans-serif",
textShadow: '1px 1px 1px rgba(34, 34, 34, 0.5)'
},
circleSecond: {
width: '0',
height: '0',
position: 'absolute',
top: '50%',
left: '50%',
margin: '-40% -1px 0 0',
padding: '40% 1px 0',
background: '#61b2a9',
'-webkit-transform-origin': '50% 100%',
'-ms-transform-origin': '50% 100%',
transformOrigin: '50% 100%'
},
circleMinute: {
width: '0',
height: '0',
position: 'absolute',
top: '50%',
left: '50%',
margin: '-40% -3px 0',
padding: '40% 3px 0',
borderRadius: '3px',
background: '#61b2a7',
'-webkit-transform-origin': '50% 100%',
'-ms-transform-origin': '50% 100%',
transformOrigin: '50% 100%'
},
circleHour: {
width: '0',
height: '0',
position: 'absolute',
top: '50%',
left: '50%',
margin: '-25% -4px 0',
padding: '25% 4px 0',
borderRadius: '3px',
background: '#61b2a7',
'-webkit-transform-origin': '50% 100%',
'-ms-transform-origin': '50% 100%',
transformOrigin: '50% 100%'
},
circleAfter: {
position: 'absolute',
top: '50%',
left: '50%',
width: '12px',
height: '12px',
margin: '-6px 0 0 -6px',
background: '#fff',
borderRadius: '6px',
content: "",
background: '#61b2a7',
display: 'block'
}
});
And use parts for inline styles of your component. Every stylesheet and stylesheet element owns an extend
function to customize your styles on demand. You can also specify an extend
member in a stylesheet element for declarative customization.
let Style = Elem.stylesheet({
withBackground: {
backgroundColor: 'yellow',
borderStyle: 'solid',
borderWidth: '1px',
borderColor: 'black'
}
});
let CustomBackground = Style.withBackground.extend({
backgroundColor: 'red',
borderRadius: '5px',
borderColor: 'blue'
});
let CustomStyle = Elem.stylesheet({
customBackground: {
extend: Style.withBackground,
backgroundColor: 'red',
borderRadius: '5px',
borderColor: 'blue'
}
});
You can also use it as an actual stylesheet for your pages, just call mount()
on a stylesheet object to mount it in the DOM. You can call unmount()
to remove it.
About Elem.Perf
Elem.Perf
is a performance measurement tool available through
const Perf = Elem.Perf;
// or
const Perf = require('elem-vdom-vanilla-experiment/lib/devtools/perfs');
// or
import * as Perf from 'elem-vdom-vanilla-experiment/lib/devtools/perfs';
The API is the following :
start
: enable performance measurementstop
: disable performance measurementmarkStart(name)
: mark the start of a measuremarkStop(name)
: mark the stop of a measuremark(name, function)
: mark the start and stop of a measure around a block of codemeasures
: return all collected measurescollectMeasures(clear)
: return all collected measures and maybe clear the measures storeprintMeasures
: print collected measures and clear the measures store
You can use the Perf API to build great perf analysis tools like here : https://github.com/mathieuancelin/elem-vdom/blob/master/src/devtools/perfmonitor.js
You can use it with
const Monitor = require('elem-vdom-vanilla-experiment/lib/devtools/perfmonitor');
// or
import * as Monitor from 'elem-vdom-vanilla-experiment/lib/devtools/perfmonitor';
function PerfMonitor() {
return Elem.el(Monitor);
}
Elem.render(PerfMonitor, container);
About Elem.Store
Just a bunch of tools to create stores (heavily inspired by Redux) available through
const Store = Elem.Store;
// or
const Store = require('elem-vdom-vanilla-experiment/lib/store');
// or
import * as Store from 'elem-vdom-vanilla-experiment/lib/store';
The API is the following :
createStore(reducer, initialState)
: Create a store from one or more reducershandleActions(actions, initialState)
: same than createStore with helpers to avoid switch casewithInitialState(initialState)
: builder forhandleActions
enrichCreateStoreWith(plugins...)
: return a newcreateStore
function with enhanced dispatchPlugins
: Plugins to enhance stores withenrichCreateStoreWith
.Provider
: Component that will feed the context with the store and some actionsSelector
: Component that will be use to wrap sub components to pass them state parts and actions
const INCREMENT_COUNTERS = 'INCREMENT_COUNTERS';
const DECREMENT_COUNTERS = 'DECREMENT_COUNTERS';
function increments() {
return {
type: INCREMENT_COUNTERS
};
}
function decrements() {
return {
type: DECREMENT_COUNTERS
};
}
const counters = Store.withInitialState({ value1: 0, value2: 0 }).handleActions({
[INCREMENT_COUNTERS]: (state) => ({ value1: state.value1 + 1, value2: state.value2 + 2 }),
[DECREMENT_COUNTERS]: (state) => ({ value1: state.value1 - 1, value2: state.value2 - 2 })
});
function anotherWayToDefineCountersAsAReducer(state = 0, action) {
switch (action.type) {
case INCREMENT_COUNTERS:
return { value1: state.value1 + 1, value2: state.value2 + 2 };
case DECREMENT_COUNTERS:
return { value1: state.value1 - 1, value2: state.value2 - 2 };
default:
return state;
}
}
let store = Store.createStore({ counters });
function CounterSelector1(state) {
return {
counter: state.counters.value1 // state.counters because we target the counters reducer
};
}
function CounterSelector2(state) {
return {
counter: state.counters.value2 // state.counters because we target the counters reducer
};
}
function CountLine() {
// here this.props.counter come from the store, throught the selector
// here this.prop.perform come from actions on the Store.Selector
return <p onClick={this.props.perform}>{this.props.name} : {this.props.counter + ''}</p>;
}
function Counter() {
return (
<Store.Provider store={store} actions={{ increments, decrements }} render={ () =>
<div>
<Store.Selector
selector={CounterSelector1}
actions={{ perform: increments }}
name="count1"
render={CountLine} />
<Store.Selector
selector={CounterSelector2}
actions={{ perform: decrements }}
name="count2"
render={CountLine} />
<button type="button" onClick={this.context.actions.increments}>+1</button>
<button type="button" onClick={this.context.actions.decrements}>-1</button>
</div>
} />
);
}
Elem.render(Counter, container);
If you want to play with enhanced store, write something like the following
const INCREMENT_COUNTERS = 'INCREMENT_COUNTERS';
const DECREMENT_COUNTERS = 'DECREMENT_COUNTERS';
function increments() {
return {
type: INCREMENT_COUNTERS
};
}
function decrements() {
return {
type: DECREMENT_COUNTERS
};
}
const counters = Store.withInitialState({ value1: 0, value2: 0 }).handleActions({
[INCREMENT_COUNTERS]: (state) => ({ value1: state.value1 + 1, value2: state.value2 + 2 }),
[DECREMENT_COUNTERS]: (state) => ({ value1: state.value1 - 1, value2: state.value2 - 2 })
});
let createStore = Store.enrichCreateStoreWith(Store.Plugins.Logger, ...);
let store = createStore({ counters });
...
The Logger plugin will be plugged into the dispatch
function of the store and will be executed before the actual dispatch. This allows you to control the actions, pass it to the dispatch, or stop dispatch, or dispatch later.
You can require the plugins from
const Plugins = Elem.Store.Plugins;
// or
const Devtools = require('elem-vdom-vanilla-experiment/lib/storeplugins');
// or
import * as Devtools from 'elem-vdom-vanilla-experiment/lib/storeplugins';
A simple example is action logging
const logger = store => next => action => {
console.log('Dispatching', action);
return next(action);
};
You can check out the code of existing plugins at https://github.com/mathieuancelin/elem-vdom/blob/master/src/storeplugins.js
About Elem.Devtools
Just a bunch of tools for DX available through
const Devtools = Elem.Devtools;
// or
const Devtools = require('elem-vdom-vanilla-experiment/lib/devtools');
// or
import * as Devtools from 'elem-vdom-vanilla-experiment/lib/devtools';
The API is the following :
Elem.Devtools.Perf
: the perf API as explained in About Elem.PerfElem.Devtools.PerfMonitor
: the perf monitor as explained in About Elem.Perf. Also available atelem-vdom-vanilla-experiment/lib/devtools/perfmonitor
Elem.Devtools.Redbox(error)
: A component that will display a JavaScript Error in a red box. Also available atelem-vdom-vanilla-experiment/lib/devtools/redbox
Elem.Devtools.ErrorMonitor(function)
: A function to wrap a function that can throw errors. If so, the Redbox is displayed instead of the wrapped function return. Also available atelem-vdom-vanilla-experiment/lib/devtools/errormonitor
function App() {
return <h1>Hello {this.props.who}!</h1>;
}
function DebugApp() {
return Elem.el(Elem.Devtools.ErrorMonitor(App), { who: 'World' });
}
Elem.render(DebugApp, container);
Elem.Devtools.Inspector
: An inspector element to browse component tree created by elem, inspect props and state, updating state live, trigger redraw, etc ... Just use it withElem.render(Elem.Devtools.Inspector, '#inspector', { __inspectorSilent: true })
. The '__inspectorSilent' is very important at the root of the tree that include the inspector to avoid the inspector to auto refresh itself.
Elem.Devtools.InspectorAPI
: API to get internal data about the rendered Elem trees. You can use the API to build tools like the Inpector.start()
: start inspectionstop()
: stop inspectionisEnabled()
: is inspection startedgetExposedStateAndProps()
: get the current internal datacleanup()
: cleanup current internal datasubscribe(listener)
: subscribe notification of changes of the internal dataephemeralSubscribe(listener)
: subscribe notification of changes of the internal data, but one shot