Component registration
In order for Boyo to know which components are available, how to hook up their functionality and how DOM elements should identify as that component, components need to be registered.
Registering a regular component
Registering a component requires you to provide a unique name and a component definition.
Then, simply use an application instance .component
method:
import { createApp } from '@boyojs/boyo-core';
const app = createApp();
app
.component('example1', () => {
//
})
.component('example2', () => {
//
})
.component('my-minimal-component', () => {
console.log('I am a component, whoot whoot!');
});
When registering multiple components, Boyo provides a convenient .components
method allowing you to register all your components in one go istead of having multiple calls to the .component
method.
The .components
method accepts an object where the keys will serve as the component name and the values as the component definition:
import { createApp } from '@boyojs/boyo-core';
const app = createApp();
app.components({
example1() {
//
},
example2() {
//
},
'my-minimal-component'() {
console.log('I am a component, whoot whoot!');
},
});
Custom elements
Boyo allows for the easy creation of custom elements with the same API as the one used to create "regular" components. In fact, a component can be used as both a regular component and a custom element at the same time!
Registering a custom element has the same exact requirements as registering a regular component but allows for customization of how the custom element is defined when given an object as the component's definition.
Registering a custom element
To register a custom element simply use an application instance .customElement
method:
import { createApp } from '@boyojs/boyo-core';
const app = createApp();
app
.customElement('example1', () => {
//
})
.customElement('example2', () => {
//
})
.customElement('my-minimal-component', () => {
console.log('I am a component, whoot whoot!');
});
Similarly, when registering multiple custom elements, you may use the convenient .customElements
method allowing you to register all your custom elements in one go.
The .customElements
method accepts an object where the keys will serve as the custom element's node name and the values as the custom element's component definition:
import { createApp } from '@boyojs/boyo-core';
const app = createApp();
app.customElements({
example1() {
//
},
example2() {
//
},
'my-minimal-component'() {
console.log('I am a component, whoot whoot!');
},
});
Custom element registration options
The .customElement
method allows for the customization of the custom element's definition by passing in an object instead of a function as second argument.
The options allows you to specify which HTML element the custom element should extend, whether or not to attach a shadow DOM and more.
Similarly the values of the object passed to the .customElements
method can either be the registration options or a function.
Example:
import { createApp } from '@boyojs/boyo-core';
const app = createApp();
app
.customElement('example1', {
//
})
.customElement('example2', {
//
});
app.customElements({
example3: {
//
},
example4: {
//
},
});
Here's a list of all the options' properties:
-
component (required): The component definition. The same definition that would be given to
.component
. -
extends: String specifying the name of a built-in element to extend. Used to create an extended built-in custom element. See
CustomElementRegistry.define()
-
class: The built-in HTML element constructor the custom element class should extend. Used when creating an extended built-in custom element. The default is:
HTMLElement
. -
attachShadow: Either a boolean or an object customizing the shadow root to be attached to the custom element. If an object is given, it should be the exact same one that would be given to
Element.attachShadow()
. Iffalse
is given, no shadow root will be attached. Iftrue
is given, a shadow root will be attached with it's mode set to open:{ mode: 'open' }
. When a shadow root can be attached, the default option is:{ mode: 'open' }
.
Example:
// Will create an autonomous custom element
app.customElement('example1', ({ el }) => {
console.log(el);
});
// Will create an autonomous custom element with a closed shadow DOM
app.customElement('example2', {
attachShadow: { mode: 'closed' },
class: HTMLElement, // Redundant here as `HTMLElement` will be the default and required for autonomous custom element
component({ el }) {
console.log(el);
},
});
// Will create an extended built-in custom element with no shadow DOM
app.customElement('example3', {
attachShadow: false,
class: HTMLDivElement,
extends: 'div', // Signals the creation of an extended built-in custom element
component({ el }) {
console.log(el);
},
});
Here's a simplified version of the registration options' types:
// Built-in type
interface ElementDefinitionOptions {
extends?: string;
}
// Built-in type
interface ShadowRootInit {
delegatesFocus?: boolean;
mode: "closed" | "open";
slotAssignment?: "manual" | "named";
}
type AppCustomElementRegistryOptions = ElementDefinitionOptions & {
component: ComponentCallback;
class?: typeof HTMLElement;
attachShadow?: boolean | ShadowRootInit;
};