preactement
TypeScript icon, indicating that this package has built-in type declarations

1.8.5 • Public • Published

preactement

Sometimes it's useful to let the DOM render our components when needed. Custom Elements are great at this. They provide various methods that can inform you when an element is "connected" or "disconnected" from the DOM.

This package (only 2KB GZipped) provides the ability to use an HTML custom element as the root for your components. In addition, it allows the use of async code resolution if your custom element isn't immediately used, which is a great strategy for reducing code weight. The exported function can also be used for hydration from SSR in Node.

It's also a great way for you to integrate Preact into other server side frameworks that might render your HTML.

This package supports Preact. If you're using React, go to reactement for more info.

Getting Started

Install with Yarn:

$ yarn add preactement

Install with NPM:

$ npm i preactement

Using define()

preactement exports one function, define(). This allows us to register a custom element via a provided key, and provide the component we'd like to render within. It can also generate a custom element with props ready for hydration if run on the server.

The first argument must be a valid custom element string, e.g hyphenated. If you do not provide this, a prefix of component- will be applied to your element name.

In the browser

In order to register and render a component, you'll need to call define() with your chosen component, e.g:

import { define } from 'preactement';
import { HeroBanner } from './heroBanner';

/*[...]*/

define('hero-banner', () => HeroBanner);

This registers <hero-banner> as a custom element. When that element exists on the page, preactement will render our component.

If the custom element isn't present immediately, e.g it's created dynamically at some point in the future, we can provide an async function that explicitly resolves your component:

define('hero-banner', () => Promise.resolve(HeroBanner));

This allows us to reduce the overall code in our bundle, and load the required component on demand when needed.

You can either resolve the component from your async function, as seen above, or preactement will try to infer the export key based on the provided tag name. For example:

import { define } from 'preactement';

/*[...]*/

define('hero-banner', () => import('./heroBanner'));

As the heroBanner.ts file is exporting the component as a key, e.g export { HeroBanner };, and this matches the tag name in kebab-case, e.g hero-banner, the component will be correctly rendered.

On the server (SSR)

You can also use define() to generate a custom element container if you're rendering your page in Node. When wrapping your component, e.g:

define('hero-banner', () => HeroBanner);

A functional component is returned that you can include elsewhere in your app. For example:

import { define } from 'preactement';

/*[...]*/

const Component = define('hero-banner', () => HeroBanner);

/*[...]*/

function HomePage() {
  return (
    <main>
      <Component />
    </main>
  );
}

Properties

If you're not running preactement on the server, you have several ways of defining props for your component.

1. Nested block of JSON:

<hero-banner>
  <script type="application/json">
    { "titleText": "Hero Banner Title" }
  </script>
</hero-banner>

2. A props attribute (this must be an encoded JSON string)

<hero-banner props="{'titleText': 'Hero Banner Title'}"></hero-banner>

3. Custom attributes

<hero-banner title-text="Hero Banner Title"></hero-banner>

You'll need to define your custom attributes up front when using define(), e.g:

define('hero-banner', () => HeroBanner, { attributes: ['title-text'] });

These will then be merged into your components props in camelCase, so title-text will become titleText.

HTML

You can also provide nested HTML to your components children property. For example:

<hero-banner>
  <h2>Banner Title</h2>
</hero-banner>

This will correctly convert the <h2> into virtual DOM nodes for use in your component, e.g:

/*[...]*/

function HeroBanner({ children }) {
  return <section>{children}</section>;
}

Important

Any HTML provided to the custom element must be valid; As we're using the DOM's native parser which is quite lax, any html passed that is not properly sanitised or structured might result in unusual bugs. For example:

This will result in a Preact error:

<p Hello

This will result in an H1 tag:

<h1>Hello
<h1>Hello</h3>

Slots

preactement now supports the use of <* slot="{key}" /> elements, to assign string values or full blocks of HTML to your component props. This is useful if your server defines layout rules that are outside of the scope of your component. For example, given the custom element below:

<login-form>
  <h2>Please Login</h2>
  <div slot="successMessage">
    <p>You have successfully logged in, congrats!</p>
    <a href="/account">Continue</a>
  </div>
</login-form>

All elements that have a slot attribute will be segmented into your components props, using the provided slot="{value}" as the key, e.g:

function LoginForm({ successMessage }) {
  const [isLoggedIn, setLoggedIn] = useState(false);

  return (
    <Fragment>
      {isLoggedIn && successMessage}
      <form onSubmit={() => setLoggedIn(true)}>{/*[...]*/}</form>
    </Fragment>
  );
}

It's important to note that slot keys will be normalised into camelCase, for example: slot="my-slot" will be accessed via mySlot in your component's props. It's recommended to use camelCase for slot keys, but this isn't always possible. preactement will do it's best to handle all common casing conventions, e.g kebab-case, snake_case and PascalCase. Slot values can be either primitive strings, or full HTML structures, as seen in the example above.

Options

define has a third argument, "options". For example:

define('hero-banner', () => HeroBanner, {
  /*[options]*/
});

attributes

If you require custom attributes to be passed down to your component, you'll need to specify them in this array. For example:

define('hero-banner', () => HeroBanner, { attributes: ['banner-title'] });

And the custom element will look like the following:

<hero-banner banner-title="Welcome"></hero-banner>

formatProps

This allows you to provide a function to process or format your props prior to the component being rendered. One use case is changing property casings. If the data provided by your server uses Pascal, but your components make use of the standard camelCase, this function will allow you to consolidate them.

wrapComponent

If you need to wrap your component prior to render with a higher order function, you can provide it here. For example, if you asynchronously resolve your component, but also make use of Redux, you'll need to provide a wrapComponent function to apply the Provider HOC etc. It can also be useful for themeing, or other use cases.

Useful things

By default, all components will be provided with a parent prop. This is a reference to the root element that the component has been rendered within. This can be useful when working with Web Components, or you wish to apply changes to the custom element. This will only be defined when run on the client.

ES5 Support

To support ES5 or older browsers, like IE11, you'll need to either transpile preactement, or import the ES5 version via preactement/es5, while also installing the official Web Component Custom Element polyfill. Once installed, you'll need to import the following at the very top of your entry files:

import '@webcomponents/custom-elements';
import '@webcomponents/custom-elements/src/native-shim';

Acknowledgement

This function takes heavy inspiration from the excellent preact-custom-element. That library served as a starting point for this package, and all of the Preact guys deserve a massive dose of gratitude. I had slightly different needs, so decided to build this as part solution, part learning excersize.

Package Sidebar

Install

npm i preactement

Weekly Downloads

1,314

Version

1.8.5

License

MIT

Unpacked Size

26.2 kB

Total Files

8

Last publish

Collaborators

  • jahilldev