Defining Elements
Basic elements
To define a custom element, begin by creating a class that extends FASTElement and decorate it with the @customElement decorator, providing the element name.
Example: A Basic FASTElement Definition
import { FASTElement, customElement } from '@microsoft/fast-element';
@customElement('name-tag')
export class NameTag extends FASTElement {
}
With this in place, you can now use your name-tag element anywhere in HTML with the following markup:
Example: Using a Web Component
<name-tag></name-tag>
Web Component names must contain a - in order to prevent future conflicts with built-in elements and to namespace components from different libraries. For more information on the basics of Web Components see this set of articles.
HTML has a few special tags known as "self-closing tags". Common examples include <input> and <img>. However, most HTML elements and all web components must have an explicit closing tag.
We've got a basic Web Component in place, but it doesn't do much. So, let's add an attribute and make it render something.
Example: Adding Attributes to a FASTElement
import { FASTElement, customElement, attr } from '@microsoft/fast-element';
@customElement('name-tag')
export class NameTag extends FASTElement {
@attr greeting: string = 'Hello';
// optional method
greetingChanged() {
...
}
}
To add attributes to your HTML element, create properties decorated by the @attr decorator. All attributes defined this way will be automatically registered with the platform so that they can be updated through the browser's native setAttribute API as well as the property. You can optionally add a method with the naming convention propertyNameChanged to your class (e.g. greeting and greetingChanged()), and this method will be called whenever your property changes, whether it changes through the property or the attribute API.
All properties decorated with @attr are also observable. See observables and state for information about how observables enable efficient rendering.
By default, anything extending from FASTElement will automatically have a ShadowRoot attached in order to enable encapsulated rendering.
To see it in action, you can use the same HTML as above, or change the default greeting with the following:
Example: Using a Web Component with Attributes
<name-tag greeting="Hola"></name-tag>
Customizing attributes
By default, any attribute created with @attr will perform no explicit type coercion other than when it reflects its value to the HTML DOM via the setAttribute API. However, you can convert DOM attribute string values to and from arbitrary types as well as control the mode that is used to reflect property values to the DOM. There are three modes available through the mode property of the attribute configuration:
reflect- The default mode that is used if none is specified. This reflects property changes to the DOM. If aconverteris supplied, it will invoke the converter before calling thesetAttributeDOM API.boolean- This mode causes your attribute to function using the HTML boolean attribute behavior. When your attribute is present in the DOM or equal to its own name, the value will be true. When the attribute is absent from the DOM, the value of the property will be false. Setting the property will also update the DOM by adding/removing the attribute.fromView- This mode skips reflecting the value of the property back to the HTML attribute, but does receive updates when changed throughsetAttribute.
In addition to setting the mode, you can also supply a custom ValueConverter by setting the converter property of the attribute configuration. The converter must implement the following interface:
interface ValueConverter {
toView(value: any): string;
fromView(value: string): any;
}
Here's how it works:
- When the DOM attribute value changes, the converter's
fromViewmethod will be called, allowing custom code to coerce the value to the proper type expected by the property. - When the property value changes, the converter's
fromViewmethod will also be called, ensuring that the type is correct. After this, themodewill be determined. If the mode is set toreflectthen the converter'stoViewmethod will be called to allow the type to be formatted before writing to the attribute usingsetAttribute.
When the mode is set to boolean, a built-in booleanConverter is automatically used to ensure type correctness so that the manual configuration of the converter is not needed in this common scenario.
Example: An Attribute in Reflect Mode with No Special Conversion
import { FASTElement, customElement, attr } from '@microsoft/fast-element';
@customElement('name-tag')
export class NameTag extends FASTElement {
@attr greeting: string = 'Hello';
}
Example: An Attribute in Boolean Mode with Boolean Conversion
import { FASTElement, customElement, attr } from '@microsoft/fast-element';
@customElement('my-checkbox')
export class MyCheckbox extends FASTElement {
@attr({ mode: 'boolean' }) disabled: boolean = false;
}
Example: An Attribute in Reflect Mode with Custom Conversion
import { FASTElement, customElement, attr, ValueConverter } from '@microsoft/fast-element';
const numberConverter: ValueConverter = {
toView(value: any): string {
// convert numbers to strings
},
fromView(value: string): any {
// convert strings to numbers
}
};
@customElement('my-counter')
export class MyCounter extends FASTElement {
@attr({ converter: numberConverter }) count: number = 0;
}
The element lifecycle
All Web Components support a series of lifecycle events that you can tap into to execute custom code at specific points in time. FASTElement implements several of these callbacks automatically in order to enable features of its templating engine (described in declaring templates). However, you can override them to provide your own code. Here's an example of how you would execute custom code when your element is inserted into the DOM.
Example: Tapping into the Custom Element Lifecycle
import { FASTElement, customElement, attr } from '@microsoft/fast-element';
@customElement('name-tag')
export class NameTag extends FASTElement {
@attr greeting: string = 'Hello';
greetingChanged() {
this.shadowRoot!.innerHTML = this.greeting;
}
connectedCallback() {
super.connectedCallback();
console.log('name-tag is now connected to the DOM');
}
}
The full list of available lifecycle callbacks is:
| Callback | Description |
|---|---|
| constructor | Runs when the element is created or upgraded. FASTElement will attach the shadow DOM at this time. |
| connectedCallback | Runs when the element is inserted into the DOM. On first connect, FASTElement hydrates the HTML template, connects template bindings, and adds the styles. |
| disconnectedCallback | Runs when the element is removed from the DOM. FASTElement will remove template bindings and clean up resources at this time. |
| attributeChangedCallback(attrName, oldVal, newVal) | Runs any time one of the element's custom attributes changes. FASTElement uses this to sync the attribute with its property. When the property updates, a render update is also queued, if there was a template dependency. |
| adoptedCallback | Runs if the element was moved from its current document into a new document via a call to the adoptNode(...) API. |
Working without decorators
The examples above and those throughout our documentation leverage TypeScript, and in particular, the decorators feature of the language. Decorators are an upcoming feature planned for a future version of JavaScript, but their design is not yet finished. While the syntax for decorator usage is not likely to change in the final version of the feature, some of our community members may feel uncomfortable using this feature at this stage. Additionally, since decorators are transpiled into code that uses helper functions (both in TypeScript and Babel) the compiled output will be larger than the equivalent non-decorator code.
While there are size implications of using decorators prior to full language support, they do present the most declarative and readable form of the API, and we recommend their use for the average project. To strike a balance between declarative readability and size, we recommend that TypeScript be used in combination with the "importHelpers": true compiler option. When this option is set, instead of generating helper functions for decorators into every file, TypeScript will import a set of shared helpers published in the tslib package.
For those that require the smallest possible builds, FAST Elements can be completely defined in Vanilla JS, without using decorators, by leveraging a static definition field on your class. The definition field only needs to present the same configuration as the @customElement decorator. Here's an example that shows the use of the definition field along with a manual call to define the element:
import { FASTElement, html, css } from '@microsoft/fast-element';
const template = html`...`;
const styles = css`...`;
const converter = { ... };
export class MyElement extends FASTElement {
static definition = {
name: 'my-element',
template,
styles,
attributes: [
'value', // same attr/prop
{ attribute: 'some-attr', property: 'someAttr' }, // different attr/prop
{ property: 'count', converter } // derive attr; add converter
]
};
value = '';
someAttr = '';
count = 0;
}
FASTElement.define(MyElement);
The definition can also be separated from the class and passed into the define call directly if desired. Here's what that would look like: FASTElement.define(MyElement, myDefinition);