Web Components
can do that ?!
data:image/s3,"s3://crabby-images/470b2/470b2b7d995dc4a2744935468166aec564cb7a01" alt=""
data:image/s3,"s3://crabby-images/e4db5/e4db5fe931c759be6ba6a2ffde3d2457e2067dff" alt=""
Why
Web Components ?
a very big company
🖥 Many applications
🛠 Many technologies
👥 Many teams
Need consistency
Need interoperability
Need stability
data:image/s3,"s3://crabby-images/dcea6/dcea6dfe3bad44218e7cfbcc189cb3e2ea725f5b" alt=""
In big companies
At an UI level
A Design system
👇🏼
In big companies
At a technical level
Web components
👇🏼
Framework
Web components
orchestrate
plays
Agenda
2.
Use cases
3.
What is next ?
1.
The 4 trends for Web Components
Vincent Ogloblinsky
Frontend software architect
data:image/s3,"s3://crabby-images/0b986/0b986c8a3daeb5d2de589cae33fc9f1dac1f5f3f" alt=""
data:image/s3,"s3://crabby-images/640dd/640dd4f0b12a05af3a8225a54df31c8e67063df2" alt=""
The
4 trends
for
Web Components
Librairies
Compilers
Backporters
Framework
Librairies
2 approaches :
class based
functional based
Librairies - Class based
data:image/s3,"s3://crabby-images/fe563/fe56304d6a73300156fd81030e0a173202612699" alt=""
import {
LitElement,
html,
property,
customElement
} from 'lit-element';
@customElement('simple-greeting')
export class SimpleGreeting extends LitElement {
@property() name = 'World';
render() {
return html`<p>Hello, ${this.name}!</p>`;
}
}
<simple-greeting name="Everyone"></simple-greeting>
Librairies - Class based
data:image/s3,"s3://crabby-images/32d3b/32d3b538a814108f986114b404a0f38b4f8e27fd" alt=""
import {Slim} from 'slim-js'
import {
tag,
template,
attribute
} from 'slim-js/Decorators'
@tag('simple-greeting')
@template('<p>Hello, {{name}}!</p>')
class MyTag extends Slim {
@attribute
name
}
<simple-greeting name="Everyone"></simple-greeting>
Librairies - Functional based
data:image/s3,"s3://crabby-images/c6fee/c6feeda0e689899acb1618279877b566d55ad6fa" alt=""
import
Element, { html }
from '@skatejs/element-lit-html';
export default class extends Element {
static get props() {
return {
name: String
};
}
render() {
return html`
<p>Hello, ${this.name}!</p>
`;
}
}
<simple-greeting name="Everyone"></simple-greeting>
Librairies - Functional based
data:image/s3,"s3://crabby-images/78432/78432abe08d1db5be0a82669f23062e88a226457" alt=""
import { html, define } from 'hybrids';
export const HelloWorld = {
name: '',
render: ({ name }) => html`
<p>Hello, ${name}</p>
`,
};
define('simple-greeting', HelloWorld);
<simple-greeting name="Everyone"></simple-greeting>
Framework
data:image/s3,"s3://crabby-images/2829b/2829bef6e3ca223f7d722154fec1deae9da3a048" alt=""
import { define, WeElement, html, render } from 'omi'
define('simple-greeting', class extends WeElement {
render(props) {
return html`<p>{props.text}</p>`
}
});
<simple-greeting name="Everyone"></simple-greeting>
Compilers
data:image/s3,"s3://crabby-images/44083/44083c0aa4299f4689427525d0bf699d1b1079d3" alt=""
<!-- simple-greeting.riot -->
<p>{ props.message }</p>
import { register, mount, component } from 'riot';
import simpleGreeting from './simple-greeting.riot';
// register the my-component as global component
register('simple-greeting', simpleGreeting);
// find all the DOM nodes called `<simple-greeting>` and
// mount them with the component previously registered
mount('simple-greeting');
<simple-greeting name="Everyone"></simple-greeting>
Compilers
data:image/s3,"s3://crabby-images/d027a/d027add5be8ffbd99029ddc07fe09aa608e66e7e" alt=""
<p>{name}</p>
<script>
export default {
tag: 'simple-greeting',
props: ['name']
}
</script>
/**
* -> Compile it with Svelte CLI
*/
<simple-greeting name="Everyone"></simple-greeting>
Compilers
data:image/s3,"s3://crabby-images/13ab0/13ab0e201332b2f5b97e7635477394d26f08978b" alt=""
import { h, Component, Prop } from '@stencil/core';
@Component({
tag: 'simple-greeting'
})
export class SimpleGreeting {
@Prop() name: string;
render() {
return <p>{this.name}</p>;
}
}
<simple-greeting name="Everyone"></simple-greeting>
Backporters
data:image/s3,"s3://crabby-images/13446/1344678d27aa8f03a9d52cc3a2ba3738546b4fe0" alt=""
import { Component, Input } from '@angular/core';
@Component({
selector: 'simple-greeting',
template: `
<p>{{ name }}</p>
`
})
export class SimpleGreeting {
@Input() name: string;
}
<simple-greeting name="Everyone"></simple-greeting>
import { ɵrenderComponent } from '@angular/core';
import { SimpleGreeting } from './simple-greeting.component';
export class SimpleGreetingElement extends HTMLElement {
private comp: SimpleGreeting;
constructor() {
super();
this.comp = ɵrenderComponent(SimpleGreeting, { host: this });
}
}
Backporters
data:image/s3,"s3://crabby-images/b38d5/b38d5e46a18198022f850cabd83fd23ca4cbb1f0" alt=""
<template>
<p>{name}</p>
</template>
<script>
export default {
name: 'SimpleGreeting',
props: ['name']
}
</script>
/**
* -> Compile it with Vue CLI
*/
<simple-greeting name="Everyone"></simple-greeting>
Backporters
data:image/s3,"s3://crabby-images/9622b/9622b1926f94397ff0644d1da51f9cf324651cef" alt=""
import { WidgetBase } from '@dojo/framework/widget-core/WidgetBase';
import customElement from '@dojo/framework/widget-core/decorators/customElement';
import { v } from '@dojo/framework/widget-core/d';
import { ThemedMixin } from '@dojo/framework/widget-core/mixins/Themed';
@customElement({
tag: 'simple-greeting',
events: [],
attributes: ['name'],
properties: []
})
export default class SimpleGreeting extends ThemedMixin(WidgetBase) {
protected render() {
const { name } = this.properties;
return v('s', {}, [name]);
}
}
/**
* -> Compile it with Dojo CLI
*/
<simple-greeting name="Everyone"></simple-greeting>
Backporters
data:image/s3,"s3://crabby-images/29a85/29a850d831a90aee3021dc58bf046699ca35e3e4" alt=""
data:image/s3,"s3://crabby-images/c4202/c42024cbf57c4c7355589657993649a5b79ca52f" alt=""
Benchmark
data:image/s3,"s3://crabby-images/98f8b/98f8b869973d77bf026caf247f7ee31ed0df7081" alt=""
Make a choice
Warning !
Take care of Developer eXperience (DX)
versus
User eXperience (UX)
Use-cases
data:image/s3,"s3://crabby-images/28f00/28f0054f20b2a9da2c1ea59e1293d575610431cd" alt=""
1 - Design System
Centralized library of components that can be shared across teams and projects to simplify design and development while ensuring consistent brand experiences, at scale.
Design System
data:image/s3,"s3://crabby-images/fc281/fc2812b949fec1eb51aa16af89b62a2faa002faa" alt=""
Design System - Advantages
📉 Design debt
📉🤜🏻🤛🏻 Collaboration and communication problems
📉 Repetitive work
Consistency within a product family
Design System - Examples
data:image/s3,"s3://crabby-images/4e787/4e787b70a86981d0062b6444a4c79c9e1985805a" alt=""
data:image/s3,"s3://crabby-images/3ea4b/3ea4b15208029c7804e98abd79399e74d93e399d" alt=""
data:image/s3,"s3://crabby-images/a263e/a263e5d79e8b4c1735b50c559e79c2463f1ced28" alt=""
data:image/s3,"s3://crabby-images/4cc17/4cc17b9b14134c820488ae4132f998ae3791116c" alt=""
data:image/s3,"s3://crabby-images/097c6/097c66c46ed4ffe0d188a5efab80513b78a48899" alt=""
Design System - How to ?
Balance consistency with flexibility
Component describes visual language & functional behavior
Journey which designers and developers take together
Design System - How to ?
Code with Web Components
data:image/s3,"s3://crabby-images/e27f8/e27f80f9326514db571e79dbee4e99f77bb09a56" alt=""
data:image/s3,"s3://crabby-images/77627/77627910888aedfe429110b5b249baf43ce140a6" alt=""
data:image/s3,"s3://crabby-images/3c62f/3c62fe04201b500abc77143dfecd17af8fb4848f" alt=""
data:image/s3,"s3://crabby-images/43c44/43c440d2ff3e6c3aff14f53867fdac88a6ba8037" alt=""
...
Design System - How to ?
Tooling with open-wc
data:image/s3,"s3://crabby-images/e5462/e54628cd3ee21fd8b46f9c508fcac4949338e160" alt=""
Design System - How to ?
Documentation with Storybook
data:image/s3,"s3://crabby-images/b23ea/b23ea2469437bc3cd2db839b469a43ed5483248b" alt=""
Design System - How to ?
Share them with
Bit or npm
data:image/s3,"s3://crabby-images/df073/df073ea81f5c3ab52991dfcc58ba2fd756d815d9" alt=""
2 - Standalone widget
3 - Framework migration
Most of the time
⛪️ Catholic wedding with framework
🏗 Architecture is based on a framework
Framework migration
data:image/s3,"s3://crabby-images/bc696/bc696c8615b6df17b1d4600869ce9f816439f051" alt=""
data:image/s3,"s3://crabby-images/68fa3/68fa39e389d4ec2a1ac2911bf361e762a5451146" alt=""
data:image/s3,"s3://crabby-images/56538/56538a2331846564581db55b93e7133bd9bf2730" alt=""
data:image/s3,"s3://crabby-images/7101d/7101dd90c08d931e9735af5453134cd50e811b34" alt=""
data:image/s3,"s3://crabby-images/3f1cc/3f1ccbc32a1a773d7a2d789c9f6c1e12ee98da98" alt=""
?
?
?
?
Framework migration
2 strategies :
💣 Big bang
🏗 Incremental
Framework migration
data:image/s3,"s3://crabby-images/bc696/bc696c8615b6df17b1d4600869ce9f816439f051" alt=""
data:image/s3,"s3://crabby-images/68fa3/68fa39e389d4ec2a1ac2911bf361e762a5451146" alt=""
data:image/s3,"s3://crabby-images/56538/56538a2331846564581db55b93e7133bd9bf2730" alt=""
data:image/s3,"s3://crabby-images/7101d/7101dd90c08d931e9735af5453134cd50e811b34" alt=""
data:image/s3,"s3://crabby-images/3f1cc/3f1ccbc32a1a773d7a2d789c9f6c1e12ee98da98" alt=""
WC
WC
WC
WC
Framework migration
Web component integration
Migrate existing component to WC
Check your framework's WC support
Framework migration
Check framework WC support
data:image/s3,"s3://crabby-images/6db6c/6db6c2bcab59a199beb9a9c91beb68ed89406a52" alt=""
Framework migration
Feedbacks from companies
Warning on internal UI libraries and compatibility
Transition time between old & new screen
Stop coupling a lot
It just "works"
4 - Micro frontends
The idea behind Micro Frontends is to think about a website or web app as a composition of features which are owned by independent teams. Each team has a distinct area of business or mission it cares about and specialises in. A team is cross functional and develops its features end-to-end, from database to user interface. (micro-fontend.org)
Micro frontends
data:image/s3,"s3://crabby-images/82e0d/82e0dbc7a2feb026156f47f5a264ecd320211414" alt=""
Micro frontends
Strategies
Fragment concept (Zalando) - Project Mosaic
Single-spa - Framework which connects frameworks with iframes
Web Components
Micro frontends
Starting questions
How big is your application in terms of team and code ?
Can you divide your app into small pieces by domain accessory ?
How easy can you release small features for your application ?
What's next ?
Move from static to dynamic template, also during update
<template id="person">
<section>
<h1>{{name}}</h1>
Email: <a href="mailto:{{email}}">{{email}}</a>
</section>
</template>
let template = document.querySelector('#person');
let instance = template.createInstance({
name: "Ryosuke Niwa",
email: "rniwa@webkit.org"
});
CSS Shadow parts
Style elements inside Shadow DOM
<x-foo>
#shadow-root
<div part="some-box"><span>...</span></div>
<input part="some-input">
<div>...</div> /_ not styleable _/
</x-foo>
x-foo::part(some-box) { ... }
data:image/s3,"s3://crabby-images/8a739/8a73962f222615aafe9e300cb6cc5f0aee4625e8" alt=""
Chrome 74
Create stylesheet objects from script & remove the need for declarative style elements
const myElementSheet = new CSSStyleSheet();
class MyElement extends HTMLElement {
constructor() {
super();
const shadowRoot = this.attachShadow({ mode: "open" });
shadowRoot.adoptedStyleSheets = [myElementSheet];
}
connectedCallback() {
if (myElementSheet.cssRules.length == 0) {
myElementSheet.replaceSync(styleText);
}
}
}
ECMAScript 6 native module for HTML & CSS
<script type="module">
import {content} from "import.html"
document.body.appendChild(content);
</script>
<div id="blogPost">
<p>Content...</p>
</div>
<script type="module">
let blogPost = import.meta.document.querySelector("#blogPost");
export {blogPost}
</script>
ECMAScript 6 native module for HTML & CSS
import {content} from "import.html"
import styles from './styles.css';
class MyElement extends HTMLElement {
constructor() {
this.attachShadow({mode: open});
// push() doesn't actually exist yet
this.shadowRoot.moreStyleSheets.push(styles);
this.shadowRoot.appendChild(content);
}
}
data:image/s3,"s3://crabby-images/8a739/8a73962f222615aafe9e300cb6cc5f0aee4625e8" alt=""
Chrome 73
Namespacing for Custom Element definition
// Create a new registry that inherits from the global registry
const myRegistry = new CustomElementRegistry(window.customElements);
// Use the local registry when creating the ShadowRoot
element.attachShadow({mode: 'open', customElements: myRegistry});
// Define a trivial subclass of XFoo so that we can register it ourselves
class MyFoo extends XFoo {}
// Register it as `my-foo` locally.
myRegistry.define('my-foo', MyFoo);
// the definiton of my-foo will be resolved from the custom registry
const myFoo = this.shadowRoot.createElement('my-foo');
element.shadowRoot.appendChild(myFoo);
Conclusion
Analyze your application(s)
Start small, mix 1 by 1
Choose the correct scope of your WC(s)
Take care of performance issues
Automate everything
Resources
9 Web Components UI Libraries You Should Know in 2019
https://blog.bitsrc.io/9-web-component-ui-libraries-you-should-know-in-2019-9d4476c3f103
Stencil for Production design systems
Constructable Stylesheets - Chrome 73
https://developers.google.com/web/updates/2019/02/constructable-stylesheets
Resources
Design Systems Should be JavaScript Framework Agnostic
https://hackernoon.com/design-systems-should-be-javascript-framework-agnostic-2a0c47129ec8
UI in Microservices World — Micro Frontends pattern and Web Components
Keep It Simple Stupid
#UseThePlatform
Thanks for listening !
Ask me anything
Slides : bit.ly/wc-can-do-that
Web Components can do that ?
By Vincent Ogloblinsky
Web Components can do that ?
- 4,095