This library is designed to analyze components no matter where they are found. This means it needs to find and analyze components in both javascript, typescript and in typescript definition files (when consumed from a library).
As you can see we loose a lot of information when the element is defined in a typescript definition file.
export class MyElement extends HTMLElement {
myProp = "foo";
static get observedAttributes() {
return ["attr1", "attr2"];
}
set value (val) {
}
}
customElements.define("my-element", MyElement);
- Tag name:
my-element
- Properties:
value
,myProp (String)
- Attributes:
attr1
,attr2
export class MyElement extends HTMLElement {
myProp: string;
static readonly observedAttributes: string[];
value: string;
}
declare global {
interface HTMLElementTagNameMap {
"my-element": MyElement;
}
}
- Tag name:
my-element
- Properties:
value (String)
,myProp (String)
- Attributes: No attributes because
observedAttributes
is just a function that returnsstring[]
@customElement("my-element")
export class MyElement extends LitElement {
myProp = "myProp";
@property({type: String}) prop4 = "hello";
@property({type: Boolean, attribute: "prop-5"}) prop5;
static get properties() {
return {
prop1: { type: String },
prop2: { type: Number, attribute: "prop-two" },
prop3: { type: Boolean, attribute: false }
};
}
}
- Tag name:
my-element
- Properties:
myProp (String)
,prop1 (String)
,prop2 (Number)
,prop3 (Boolean)
,prop4 (String)
,prop5 (Boolean)
- Attributes:
prop1 (String)
,prop-two (Number)
,prop4 (String)
,prop-5 (Boolean)
,
export class MyElement extends LitElement {
myProp: string;
prop4: string;
prop5: string;
static get properties(): Object;
}
class IconToggle extends PolymerElement {
static get properties () {
return {
toggleIcon: {
type: String
}
};
}
}
import { Component, Prop, EventEmitter } from '@stencil/core';
@Component({
tag: 'my-embedded-component'
})
export class MyEmbeddedComponent {
@Prop() color: string = 'blue';
@Event() change: EventEmitter;
@Event({eventName: 'todoCompleted'}) todoCompleted: EventEmitter;
render() {
return (
<div>My favorite color is {this.color}</div>
);
}
}
- Tag name:
my-embedded-component
- Properties:
color (String)
- Attributes:
color (String)
- Events:
change
,todoCompleted
When building Stencil components the compiler spits out alot of typescript definition files that describe both attributes and properties. These are parsed by this library. StencilIntrinsicElements
is of interest because it describes the attributes of the element.
import './stencil.core';
export namespace Components {
interface ProgressRing {
'decimalSize': number;
}
interface ProgressRingAttributes extends StencilHTMLAttributes {
'decimalSize'?: number;
}
}
declare global {
interface StencilElementInterfaces {
'ProgressRing': Components.ProgressRing;
}
interface StencilIntrinsicElements {
'progress-ring': Components.ProgressRingAttributes;
}
interface HTMLProgressRingElement extends Components.ProgressRing, HTMLStencilElement {}
var HTMLProgressRingElement: {
prototype: HTMLProgressRingElement;
new (): HTMLProgressRingElement;
};
interface HTMLElementTagNameMap {
'progress-ring': HTMLProgressRingElement
}
interface ElementTagNameMap {
'progress-ring': HTMLProgressRingElement;
}
}
- Tag name:
progress-ring
- Properties:
decimalSize (Number)
- Attributes:
decimalSize (Number)
In addition, the entire content of HTMLElement is copied into the interfaces, so WCA makes sure to filter that out.
An LWC component class is automatically identified from a JavaScript/Typescript file if:
- It is the default export of the file
- The JS file name equals the parent folder name (ex:
/../mycomp/mycomp.{js|ts}
) - There is an HTML template with the same name in the same folder (ex:
/../mycomp/mycomp.html
)
The component is named after the last 2 parts of its parent directory. For example, if the file path is
/../myns/mycomp/mycomp.js
, then the component tag is myns-mycomp
. The name is also transformed from camel case
to dash case: /../myns/mycomp/myComp.js
leads to myns-my-comp
.
For components that are not following the previous naming scheme, it uses 2 other methods:
-
Checks if the class extends
LightningElement
.
This only works with a direct inheritance as it doesn't check the whole hierarchy. -
Looks for a
@lwcelement
JSDoc tag.
The JSDoc tag can also override the default tag name.
/**
* @lwcelement my-element
*/
export default class MyElement extends BaseComponent {
export default class MyElement extends LightningElement {
prop1 = "myProp";
@api prop2 = "hello";
/**
* @type number
*/
@api prop3;
}
All the member definitions are set as protected unless the are decorated with @api
. In this case,
they become public and the properties also becomes attributes. It will automatically guess the type
of the property, or JSDoc tag can be used as well.
- Tag name:
my-element
- Properties:
prop1 (String)
,prop2 (String)
,prop3 (Number)
- Attributes:
prop2 (String)
,prop3 (Number)
export default class MyElement extends LightningElement {
prop1: string;
@api prop2: string;
/**
* @type number
*/
@api prop3;
}