Web components as a compile target
Angular Elements
and friends
Andrei Antal
ngBucharest
data:image/s3,"s3://crabby-images/f607c/f607cc25c23b17d6e4b778b26c4889ca605235c1" alt=""
data:image/s3,"s3://crabby-images/afa2f/afa2f594eef25f4a6e1237015b49858b96102e93" alt=""
Netcentric Meetup, Bucharest, 20 Feb 2019
slides: bit.ly/2BF1cSX
Hello World!
data:image/s3,"s3://crabby-images/4f076/4f07615da2b7e71cb8ebdb98d8f75a034a41cf02" alt=""
Andrei Antal
@andrei_antal
- frontend engineer, since i can remember
- web & JS technologies enthusiast
- perpetual learner
Frontend Developer, ThisDot Labs
data:image/s3,"s3://crabby-images/c44db/c44dbf958424a20d789bc71cfeacebd94d1e9a65" alt=""
organizer for ngBucharest
data:image/s3,"s3://crabby-images/31eb0/31eb09e6432514a147770b53355949a49fef5625" alt=""
data:image/s3,"s3://crabby-images/c1db9/c1db96562cc8156c3ad5ea2d9ed271e64b42b337" alt=""
@ngBucharest
groups/angularjs.bucharest
data:image/s3,"s3://crabby-images/1b4c1/1b4c181f0dde0bbb877b4a16b8bdfd59108b6410" alt=""
Angular Labs
data:image/s3,"s3://crabby-images/57e32/57e3289922e8fca1303a7cf1a3ce049132aa2f38" alt=""
@AngularMIX, October 2017
data:image/s3,"s3://crabby-images/5d9bb/5d9bbb0f32a1129498e7a0b8d83e317b90cf7b05" alt=""
@AngularConnect, November 2017
There's no excuse for not using Angular on your next project...
...but it might get challenging to use if your app is not a SPA
Rob Wormald, Angular Team (paraphrase)
Angular Components can be tough to use outside Angular
Rob Wormald, Angular Team
Why components outside of Angular?
data:image/s3,"s3://crabby-images/04457/0445755ce1fa316c77dedcfd2516e0a08c5bd782" alt=""
Web Components Standard
HTML Templates
Shadow DOM
HTML Imports
Custom Elements
data:image/s3,"s3://crabby-images/cd1cc/cd1cc680bf02db7f977a0da9591d5f31bc4fb25b" alt=""
data:image/s3,"s3://crabby-images/afa2f/afa2f594eef25f4a6e1237015b49858b96102e93" alt=""
Web Components Standard
-> HTML Templates
Shadow DOM
HTML Imports
Custom Elements
<template>
<h1> Hello <h1>
<div>
...content
</div>
</template>
Web Components Standard
HTML Templates
-> Shadow DOM
HTML Imports
Custom Elements
<my-app>
<#shadow-root>
<div class="main">
<h1 class="title">
Hello
</h1>
...
<div>
</shadow-root>
</my-app>
Web Components Standard
HTML Templates
Shadow DOM
-> HTML Imports
Custom Elements
<html>
<head>
<link
rel="import"
href="file.html">
</link>
</head>
</html>
Dropped in favor of ES Module spec
Web Components Standard
HTML Templates
Shadow DOM
HTML Imports
-> Custom Elements
<my-app>
<custom-header />
<main-content />
</my-app>
The Web Componet promise
<my-datepicker></my-datepicker>
Web Components and the DOM
class MyDatepicker extends HTMLElement {
static observedAttributes = ['my-date'];
attributeChangedCallback(oldvalue, newvalue, key) {
// update the DOM
}
}
Attributes
<my-datepicker date="7/12/2017"> </my-datepicker>
const myPicker = document.querySelector('my-dateicker');
myPicker.setAttribute('my-date', new Date().toString());
Web Components and the DOM
class MyElement extends HTMLElement {
set someValue(value) { ... }
get someValue() { return ... }
}
Properties
const myPicker = document.querySelector('my-dateicker');
elem.somevalue = "...";
Web Components and the DOM
class MyDatepicker extends HTMLElement {
emitDateChange() {
let datechangeEv =
new CustomEvent('date-change', {dateDetails});
this.dispatchEvent(datechangeEv);
}
}
Events
const myPicker = document.querySelector('my-dateicker');
elem.addEventListener('date-change',() => { /* change */});
Web Components and the DOM
class MyDatepicker extends HTMLElement {
connectedCallback() { ... }
disconnectedCallback() { ... }
attributeChangedCallback(
attributeName,
oldValue,
newValue,
namespace ) { ... }
adoptedCallback(oldDocument, newDocument)
}
Lifecycle hooks
<body>
....
<my-datepicker date="02/02/2018"></my-datepicker>
...
</body>
Exposes:
- attributes/properties
- bindable events
Encapsulates
- template (structure)
- styles
- logic
const myPicker = document.querySelector('my-dateicker');
elem.addEventListener('date-change', ev => { /* change */});
data:image/s3,"s3://crabby-images/1a9a6/1a9a6f9d459ff0f19639def43f9fc93ec7a62059" alt=""
data:image/s3,"s3://crabby-images/83acc/83acc816073c01c1d3800349c4117b6fd1f64b61" alt=""
Web Components work with Angular out of the box
data:image/s3,"s3://crabby-images/afa2f/afa2f594eef25f4a6e1237015b49858b96102e93" alt=""
<angular-app>
...
<my-datepicker
[attr.locale]="someLocale"
[date]="someDate"
(dateChange)="changeDate()"
>
</my-datepicker>
...
</angular-app>
Why not web components everywhere?
data:image/s3,"s3://crabby-images/9a282/9a2824fabe7c7d94dcb190ee6640271b9a21b46c" alt=""
Why not Polymer?
data:image/s3,"s3://crabby-images/7861c/7861c7ade689be6c4979de82a4fa1e41b0b5d43b" alt=""
<iron-ajax
auto
url="https://www.googleapis.com/youtube/v3/search"
params='{"part":"snippet", "q":"polymer", "key": "YOUTUBE_API_KEY", "type": "video"}'
handle-as="json"
on-response="handleResponse"
debounce-duration="300"></iron-ajax>
data:image/s3,"s3://crabby-images/2cb14/2cb145557829c02f270bdf9437cd484da7e605b4" alt=""
data:image/s3,"s3://crabby-images/4047b/4047be668e876bdffcd48a2402145db06398e416" alt=""
Angular ecosystem
Angular Components can be tough to use outside Angular
Angular Elements
- Angular Components packaged
as Web Components -
data:image/s3,"s3://crabby-images/1f001/1f001d220725dd2ab07cdf6c924b89109b8c1842" alt=""
data:image/s3,"s3://crabby-images/f607c/f607cc25c23b17d6e4b778b26c4889ca605235c1" alt=""
How does it work?
-
Hosting Angular Components inside Custom Elements (NgElement) - "Angular on the inside, standards on the outside"
-
Bridge between Angular Components and DOM
-
@Inputs - properties
-
@HostBinding - attributes
-
@Outputs - events
-
Lifecycle hooks
-
-
Generate a bundle.js file than you need to include in your app
-
Use it in any application - Angular, static HTML, React, Vue, etc.
-
Self-bootstrapping - drop the element on the page and it works
@Component()
NgElement
@HostBinding
@Input()
@Output()
Lifecycle Hooks
Attributes
Properties
Events
Reactions
"compile"
register as Custom Element
How does it work?
Bridging Angular and DOM APIs
How does it work?
data:image/s3,"s3://crabby-images/35bd4/35bd4c37ba206d885eeb7992ad41b2401c3e594b" alt=""
How does it work?
data:image/s3,"s3://crabby-images/1b701/1b7011204ff8b274abc8d37d0bf3ff456d2778fe" alt=""
Creating an Angular Elements project
ng new custom-element
ng add @angular/elements --project=custom-element
1. Create a new project with the Angular CLI
2. Add the Angular Elements Schematics
ng generate component NgComponent
3. Generate a simple component
npm install @webcomponents/webcomponentsjs
4. (Optional) Install web components polyfill
Write your average Angular Component
@Component({
selector: 'ng-component',
template: `
<h1> Hello World, my name is {{myName}} ! </h1>
<button (click)="onClick()">Hi !</button>
`,
styles: [...],
encapsulation: ViewEncapsulation.Native,
})
export class NgComponent implements OnInit {
@Input() myName;
@Output() sayHi = new EventEmitter<string>();
ngOnInit() {
this.myName = 'Andrei'
}
onClick() {
this.sayHi.emit(`Yo, wassup ${this.myName}?.`);
}
}
Create the Components Module
import { MyNgComponent } from './ng-component';
import { createCustomElement } from '@angular/elements';
@NgModule({
imports: [BrowserModule],
declarations: [MyNgComponent],
entryComponents: [MyNgComponent]
})
export class CustomElementsModule {
constructor(private injector: Injector) {
const el = createCustomElement(MyRatingComponent,
{injector : this.injector});
customElements.define('my-rating', el);
}
ngDoBootstrap() {
}
}
Register the custom elements
import { platformBrowserDynamic } from '@angular/platform...';
import { enableProdMode } from '@angular/core';
import { CustomElementsModule } from './app';
if (environment.production) {
enableProdMode();
}
platformBrowserDynamic()
.bootstrapModule(CustomElementModule)
Consume the custom element
<head>
...
<script src="my-ngComponent.bundle.js">
</script>
...
</head>
<body>
...
<ng-component></ng-component>
...
</body>
<head>
...
<script src="mini-angular.js"></script>
<script src="my-ngComponent.js"></script>
...
</head>
<body>
...
<my-ngComponent></my-ngComponent>
...
</body>
Looks familiar?
<head>
...
<script src="jquery.min.js"></script>
<script src="jquery.my-datepicker.js">
</script>
...
</head>
<body>
<div class="datepicker"></div>
<script>
$('.datepicker').myDatepicker({...})
</script>
</body>
data:image/s3,"s3://crabby-images/16290/16290a1dae536f9bc74c604eab8c3f25b9f9ef2b" alt=""
Using ngx-build-plus
data:image/s3,"s3://crabby-images/06350/063504d19440eca92589bf6b07d21d1c148d9ab9" alt=""
ng add ngx-build-plus
1. Add the ngx-build-plus library
"build:elements": "ng build --prod
--output-hashing none
--single-bundle true
--keep-polyfills"
2. Add a npm script to build the elements
3. Use generated files
data:image/s3,"s3://crabby-images/c5dc0/c5dc0cea51c7cd7af68b12759336b8af158cc255" alt=""
Using ngx-build-plus
DEMO TIME!
fingers crossed!
data:image/s3,"s3://crabby-images/8266c/8266c674dd1a17f9182b89d73e7ce74d030e76c7" alt=""
The magical, reusable web component compiler
Ionic Framework
data:image/s3,"s3://crabby-images/c0429/c04295c890e0ec53d2792485a11ef5de64e4c931" alt=""
Ionic Framework
The good
- Open source, cross-platform UI framework
- Based 100% on web technologies (HTML5, CSS, JS)
- Used to develop Native mobile apps, alongside Cordova
The not so good
- Only uses Angular
- Big bundles - problematic for websites or PWA
STENCIL
What is it?
- Stencil is a compiler that generates Web Components (Custom elements)
- Some features:
- Written in TypeScript
- Uses Virtual DOM
- Async rendering (inspired by React Fiber)
- Reactive data-binding
- JSX for templating
STENCIL
A simple component
import { Component, Prop } from '@stencil/core';
@Component({
tag: 'my-component',
styleUrl: 'my-first-component.scss'
})
export class MyComponent {
@Prop() name: string;
render() {
return (
<p>
Hello {this.name}
</p>
);
}
}
<my-component name="World"></my-component>
STENCIL
Using JSX
render() {
return (
<div>Hello {this.name}</div>
)
}
render() {
return (
<div>
{this.name
? <p>Hello {this.name}</p>
: <p>Hello World</p>
}
</div>
);
}
JSX - Conditional rendering
STENCIL
JSX - Slots
// child component
render(){
return [
<slot name="item-start" />,
<h1>Here is my main content</h1>,
<slot name="item-end" />
]
}
// parent component
render(){
return(
<my-component>
<p slot="item-start">I'll be placed before the h1</p>
<p slot="item-end">I'll be placed after the h1</p>
</my-component>
)
}
STENCIL
JSX - Events
export class MyComponent {
handleClick(event: UIEvent) {
alert('Received the button click!');
}
render() {
return (
<button onClick={this.handleClick(event).bind(this)}>
Click Me!
</button>
);
}
}
STENCIL
Methods
import { Method } from '@stencil/core';
...
export class TodoList {
@Method()
showPrompt() {
// show a prompt
}
}
const todoListElement = document.querySelector('todo-list');
todoListElement.showPrompt();
STENCIL
State
import { State } from '@stencil/core';
...
export class TodoList {
@State() completedTodos: Todo[];
completeTodo(todo: Todo) {
this.completedTodos = [...this.completedTodos, todo];
}
render() {
// render the todo list
}
}
STENCIL
Events
import { Event, EventEmitter } from '@stencil/core';
...
export class TodoList {
@Event() completed: EventEmitter;
completedHandler(todo: Todo) {
this.todoCompleted.emit(todo);
}
}
const todoListElement = document.querySelector('todo-list');
todoListElement.addEventListener('completed', (ev) => {/*handle*/})
STENCIL
Styling
@Component({
tag: 'shadow-component',
styleUrl: 'shadow-component.css',
shadow: true
})
export class ShadowComponent {
}
this.el.shadowRoot.querySelector()
// global styles
:root {
--app-primary-color: #488aff;
}
// shadow-component.css
h1 {
color: var(--app-primary-color)
}
data:image/s3,"s3://crabby-images/87b8e/87b8e7176182dbf6742a1d40df36380fe251353d" alt=""
SkateJS
Effortless custom elements for modern view libraries.
import { props, withComponent } from 'skatejs';
import withReact from '@skatejs/renderer-react';
import React from 'react';
class WithReact extends withComponent(withReact()) {
static get props() {
return {
name: props.string
};
}
render({ name }) {
return <span>Hello, {name}!</span>;
}
}
customElements.define('with-react', WithReact);
SkateJS React renderer
import { props, withComponent } from 'skatejs';
import withLitHtml from '@skatejs/renderer-lit-html';
import { html } from 'lit-html';
class WithLitHtml extends withComponent(withLitHtml()) {
static get props() {
return {
name: props.string
};
}
render({ name }) {
return html`Hello, ${name}!`;
}
}
customElements.define('with-lit-html', WithLitHtml);
SkateJS lit-html renderer
data:image/s3,"s3://crabby-images/2510b/2510b06601a7d3ebc171f523f7ca9ead8b43dd90" alt=""
data:image/s3,"s3://crabby-images/832eb/832ebdece136039f13c4e5e1396d0f8c42c9c792" alt=""
credits @julienrenaux
credits @julienrenaux
data:image/s3,"s3://crabby-images/057b6/057b66a3aeb4eff19b95ec4105bbc826c63b1996" alt=""
data:image/s3,"s3://crabby-images/82ffb/82ffb1edf2daef454ab2420ef34713b6e09e0424" alt=""
data:image/s3,"s3://crabby-images/fa6d2/fa6d269060a538035be473cb085017c87751abd2" alt=""
What's next?
data:image/s3,"s3://crabby-images/377ec/377eccd75c7e9ab29fbd6d1c33c16381b04970a0" alt=""
What's next?
data:image/s3,"s3://crabby-images/8c710/8c7100777a628bc464bcf7d22f8f52aae3d81a3f" alt=""
data:image/s3,"s3://crabby-images/ea40c/ea40c459b422eaadb45adbaa3f0aa72a54380bef" alt=""
The future?
Microfrontends
data:image/s3,"s3://crabby-images/79ab7/79ab7e6ffdc64e83e654ed286dd2934fa8b34db2" alt=""
Thanks!
data:image/s3,"s3://crabby-images/31eb0/31eb09e6432514a147770b53355949a49fef5625" alt=""
@andrei_antal
Reach me at:
data:image/s3,"s3://crabby-images/7aedd/7aeddfaa1f4d104499af43426b4b9a27d24c3f90" alt=""
antal.a.andrei@gmail.com
#community4thewin
Web Components as a compile target
By Andrei Antal
Web Components as a compile target
20 Feb 2019 @Netcentric meetup, Bucharest
- 1,104