Skip to content
This repository has been archived by the owner on Jan 4, 2018. It is now read-only.

skatejs/renderer-react

Repository files navigation

This has been moved to the skatejs monorepo!

skatejs/renderer-react

SkateJS renderer for React.

Install

npm install @skatejs/renderer-react react react-dom skatejs

Usage

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);

Wrapping React components with Web Components

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.

Using Flow to share prop types

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);

Future plans

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;
}

About

React renderer for SkateJS.

Resources

License

Stars

Watchers

Forks

Packages

No packages published