This has been moved to the skatejs monorepo!
SkateJS renderer for React.
npm install @skatejs/renderer-react react react-dom skatejs
The simple use case is if you're using React as a rendering layer.
import { props, withComponent } from 'skatejs';
import withReact from '@skatejs/renderer-react';
import React from 'react';
class WcHello extends withReact(withComponent()) {
// The `children` prop is auto-defined in the `withReact` mixin and outputs a <slot />
static props = {
yell: props.boolean
}
renderCallback({ children, yell }) {
return <div>Hello, {yell ? <strong>{children}</strong> : children}!</div>;
}
}
customElements.define('wc-hello', WcHello);
Being able to wrap a React component with a web component is extremely powerful because you can author everything in React - whether it be existing or new components - and use them in any other stack that works with the DOM such as Vue, Angular, Preact, Inferno, CycleJS and more.
This isn't specific to React, we have other renderers and the same rule applies, React is simply the largest and most popular option at the moment, so it carries more weight to make an example with it.
Here we take the previous example and author a standalone React component from it.
import { props, withComponent } from 'skatejs';
import withReact from '@skatejs/renderer-react';
import React from 'react';
class ReactHello extends React.Component {
render() {
const { children, yell } = this.props;
return (
<div>Hello, {yell ? <strong>{children}</strong> : children}!</div>
);
}
}
class WcHello extends withReact(withComponent()) {
static props = {
yell: props.boolean
}
renderCallback({ props }) {
return (
<ReactHello {...props} />
);
}
}
customElements.define('wc-hello', WcHello);
For either example, you can now just write HTML:
<wc-hello yell>World</wc-hello>
It's important to note that it's a best practice to provide an attribute API, so we must specify props
that will auto-link props to attributes. This is also required because the component needs to know which props cause a re-render.
This can be automated as described in the next section.
If you're using Flow, you can share prop type definitions for both components using this Babel plugin.
The example above can be rewritten to share Flow types for their props.
// @flow
import { props, withComponent } from 'skatejs';
import withReact from '@skatejs/renderer-react';
import React from 'react';
type Props = {
yell: boolean;
};
class ReactHello extends React.Component {
props: Props;
render() {
const { children, yell } = this.props;
return (
<div>Hello, {yell ? <strong>{children}</strong> : children}!</div>
);
}
}
class WcHello extends withReact(withComponent()) {
props: Props;
renderCallback() {
return (
<ReactHello {...this.props} />
);
}
}
customElements.define('wc-hello', WcHello);
One of the major use cases of this renderer would be to wrap React components. Therefore it would make sense to provide a conventional implementation of renderCallback
, instead of having to specify it every time. Maybe something like the following:
class WcHello extends withReact(withComponent()) {
props: Props;
static reactComponent = ReactHello;
}