A look into the future
Angular Elements
data:image/s3,"s3://crabby-images/df021/df0214f90053ca18b8523c9e6137054286a4bb8d" alt=""
Andrei Antal
ngBucharest
ng-europe
2 february 2018
data:image/s3,"s3://crabby-images/5983c/5983cedc33a4dd08e6b12536081129a9ef774c36" alt=""
Bonjour!
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
Bucharest, Romania
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/bab76/bab763ae894627a9a31ded019b353ab99d8cba1b" alt=""
WARNING!
data:image/s3,"s3://crabby-images/3cf6c/3cf6c6549103a75eb7bc59739d1a05ebed78bf53" alt=""
DON'T TRY THIS AT HOME
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/644c1/644c16410f026abaf15461da0cf974a7e48efe83" alt=""
data:image/s3,"s3://crabby-images/6905b/6905be8585dafd739e3ce294b36f5c66a062e490" alt=""
Web Components Standard
HTML Templates
Shadow DOM
HTML Imports
Custom Elements
data:image/s3,"s3://crabby-images/cd1cc/cd1cc680bf02db7f977a0da9591d5f31bc4fb25b" alt=""
<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/1a40d/1a40d4cf5d1aea6aa953299e6d6914d7abdd8a49" alt=""
data:image/s3,"s3://crabby-images/83acc/83acc816073c01c1d3800349c4117b6fd1f64b61" alt=""
Web Components work with Angular out of the box
<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=""
Angular Components can be tough to use outside Angular
data:image/s3,"s3://crabby-images/5983c/5983cedc33a4dd08e6b12536081129a9ef774c36" alt=""
Angular Elements
- Angular Components packaged
as Web Components -
data:image/s3,"s3://crabby-images/1f001/1f001d220725dd2ab07cdf6c924b89109b8c1842" 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
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';
@NgModule({
imports: [BrowserModule],
declarations: [MyNgComponent],
entryComponents: [MyNgComponent]
})
export class CustomElementsModule {
ngDoBootstrap() {}
}
Register the custom elements
import { platformBrowserDynamic } from '@angular/platform...';
import { registerAsCustomElements } from '@angular/elements';
import { CustomElementsModule } from './app';
import { MyNgComponent } from './ng-component';
registerAsCustomElements([MyNgComponent], () => {
return platformBrowserDynamic()
.bootstrapModule(CustomElementsModule);
})
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=""
DEMO TIME!
fingers crossed!
data:image/s3,"s3://crabby-images/8266c/8266c674dd1a17f9182b89d73e7ce74d030e76c7" alt=""
The magical, reusable web component compiler
import { Component, Prop } from '@stencil/core';
@Component({
tag: 'my-first-component',
styleUrl: 'my-first-component.scss'
})
export class MyComponent {
@Prop() name: string;
@Event() someEvent: EventEmitter;
someEventHandler(data) {
this.someEvent.emit(data);
}
render() {
return (
<p>
My name is {this.name}
</p>
);
}
}
data:image/s3,"s3://crabby-images/87b8e/87b8e7176182dbf6742a1d40df36380fe251353d" alt=""
SkateJS
Effortless custom elements for modern view libraries.
import 'skatejs-web-components';
import { Component, h, prop } from 'skatejs';
import styles from './styles';
class SKTags extends Component {
static props = {
delimiter: prop.string({attribute: true, default: ' '})
}
renderCallback() {
return <div>
<style>{styles}</style>
<div class="wrapper">
<span class="tags"></span>
<input type="text" autofocus="true" class="input"/>
</div>
</div>
}
}
customElements.define('sk-tags', SKTags);
data:image/s3,"s3://crabby-images/2510b/2510b06601a7d3ebc171f523f7ca9ead8b43dd90" alt=""
data:image/s3,"s3://crabby-images/fa6d2/fa6d269060a538035be473cb085017c87751abd2" alt=""
What's next?
What's next?
data:image/s3,"s3://crabby-images/69035/69035407539ce1b90ea97d7278e0c4de5af290fb" alt=""
What's next?
data:image/s3,"s3://crabby-images/b91d2/b91d2ce5b2cb13f550ea18055342b7bf27cb67b8" alt=""
Angular dynamic pages talk - AngularConnect 2017
Ward Bell and Jesus Rodriguez
Angular docs - angular.io
data:image/s3,"s3://crabby-images/7e13e/7e13e5bde96364d99c7a78352da0afd9333558c0" alt=""
data:image/s3,"s3://crabby-images/8c710/8c7100777a628bc464bcf7d22f8f52aae3d81a3f" alt=""
data:image/s3,"s3://crabby-images/ea40c/ea40c459b422eaadb45adbaa3f0aa72a54380bef" alt=""
The future?
WARNING!
data:image/s3,"s3://crabby-images/3cf6c/3cf6c6549103a75eb7bc59739d1a05ebed78bf53" alt=""
DON'T TRY THIS AT HOME
Merci Beaucoup!
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
data:image/s3,"s3://crabby-images/c44db/c44dbf958424a20d789bc71cfeacebd94d1e9a65" alt=""
data:image/s3,"s3://crabby-images/df021/df0214f90053ca18b8523c9e6137054286a4bb8d" alt=""
#community4thewin
A look into the future: Angular Elements
By Andrei Antal
A look into the future: Angular Elements
ng-europe 2018 Lightning Talk - A look into the future: Angular Elements by Andrei Antal
- 3,285