Package Exports
- @skatejs/val
This package does not declare an exports field, so the exports above have been automatically detected and optimized by JSPM instead. If any package subpath is missing, it is recommended to post an issue to the original package (@skatejs/val) to support the "exports" field. If that is not possible, create a JSPM override to customize the exports field for this package.
Readme
Val
Better VDOM / DOM integration
The goal of this wrapper is to provide a consistent interface across all virtual DOM solutions that provide a hyperscript-style virtual DOM function, but also provide a default interface for creating real DOM. This includes, but is not limited to:
- React
- Preact
- Virtual DOM
- Hyperscript (any implementation)
- Real DOM
- ...
npm install @skatejs/val
Rationale
The problems these different implemenations face is that the only common thing is the function that you invoke and the arguments that it accepts, at a top level. However, they all behave differently with the arguments you give them.
For example, React will only set props on DOM elements that are in its whitelist. Preact will set props on DOM elements if they are in element
. There's problems with each of these.
The problem with React is that you can't pass complex data structures to DOM elements that have properties that aren't in their whitelist, which every web component would be subject to.
With Preact, it's mostly good. However, the assumption is that your custom element definition is defined prior to Preact creating the DOM element in its virtual DOM implementation. This will fail if your custom element definitions are loaded asynchronously, which is not uncommon when wanting to defer the loading of non-critical resources.
Theres other issues such as React not working at all with custom events.
Solution
The best solution I've come across so far is to create a wrapper that works for any of these. The wrapper enables several things:
- Ability to pass a custom element constructor as the node name (first argument)
- Explicit control over which attributes to set (via
attrs: {}
in the second argument) - Explicit control over which events are bound (via
events: {}
in the second argument) - Explicit control over which props are set (anything that isn't
attrs
orevents
in the second argument) - Everything else is just passed through and subject to the standard behaviour of whatever library you're using
Requirements
This assumes that whatever library you're wrapping has support for a ref
callback as a common way for us to get access to the raw DOM element that we need to use underneath the hood.
Usage
The usage is simple. You import the wrapper and invoke it with the only argument being the virtual DOM function that you want it to wrap.
React:
import { createElement } from 'react';
import val from '@skatejs/val';
export default val(createElement);
Preact:
import { h } from 'preact';
import val from '@skatejs/val';
export default val(h);
In your components, you'd then import your wrapped function instead of the one from the library.
/** @jsx h */
import h from 'your-wrapper';
import { PureComponent } from 'react';
import { render } from 'react-dom';
class WebComponent extends HTMLElement {}
class ReactComponent extends PureComponent {
render () {
return <WebComponent />;
}
}
render(<ReactComponent />, document.getElementById('root'));
Real DOM
Val ships with a default adapter that generates real DOM nodes. To do this, simply pass no arguments to the val()
function:
/** @jsx h*/
import val from '@skatejs/val';
const h = val();
// <div>test</div>
console.log(<div>test</div>.outerHTML);
Everything works as advertised, so you can still pass custom elements, attributes and events as you normally would and things just work.
Being able to do this is immensely useful for testing real DOM and web components. Apply liberally!
Attributes
Attributes are specified using the attrs
object.
import h from 'your-wrapper';
h('my-element', {
attrs: {
'my-attribute': 'some value'
}
});
Events and custom events
Events are bound using the events
object. This works for any events, including custom events.
import h from 'your-wrapper';
h('my-element', {
events: {
click () {},
customevent () {}
}
});
Properties
Properties are categorised as anything that is not attrs
or events
.
h('my-element', {
someProp: true
});
Custom element constructors
You can also use your web component constructor instead of the name that was passed to customElements.define()
.
// So we have the reference to pass to h().
class WebComponent extends HTMLElement {}
// It must be defined first.
customElements.define('web-component', WebComponent);
// Now we can use it.
h(WebComponent);