web Components
data:image/s3,"s3://crabby-images/6905b/6905be8585dafd739e3ce294b36f5c66a062e490" alt=""
- frontend bucharest meetup -
Because why the f*#k not?
(we've used worst things before)
Andrei Antal
KALON
Contents
- web components - the what and the how
- Polymer 1.0 (now at 1.2) - sweet sugar candy components
- conclusion and discussions
WEB COMPONENTS
- Set of standards that allow us to bundle markup, styles and scripts into custom HTML elements
- Slowly gaining support on browsers - but for now we need a polyfill ( http://webcomponents.org/ - 117KB minified)
- Google as the main driving force
- We've been using them but now ca can also create our own
WEB COMPONENTS
<input type="text" value="Please insert a value...">
data:image/s3,"s3://crabby-images/9e3a5/9e3a56f9fd3cbe5aca6928c8f23dc8c1714bdb1b" alt=""
data:image/s3,"s3://crabby-images/aad7c/aad7cb909afd3999dd9006d47b6170be490b92b3" alt=""
<input type="range">
WEB COMPONENTS
data:image/s3,"s3://crabby-images/3bd16/3bd16bdbc54ed31c0663bc4627afb10e3007a890" alt=""
WEB COMPONENTS
data:image/s3,"s3://crabby-images/ed55a/ed55a903c6ff2b27b932f9e1d2fade8f19b0106e" alt=""
WEB COMPONENTS
The sin...
data:image/s3,"s3://crabby-images/611d0/611d0b16d03f7b2eba74f42acf6bf0fd11c3c922" alt=""
WEB COMPONENTS
data:image/s3,"s3://crabby-images/46024/46024ac5dd2b5bfcedf521c5d97c121a06966435" alt=""
Scumbag browser vendors
WEB COMPONENTS
data:image/s3,"s3://crabby-images/b5294/b5294823fc91041008bc572bfd963e050f76e6ef" alt=""
Enabling the DOM they don't want you to see
WEB COMPONENTS
Core elements of web components:
- HTML Templates
- Custom elements
- Shadow DOM
- HTML Imports
WEB COMPONENTS - HTML TEMPLATES
Hiding HTML elements
<div style="display:none">
<p>My image here: </p>
<img src="image.jpeg">
</div>
<script type="text/template" id="template">
<div>
<h1>Web Components</h1>
<img src="image.jpeg">
</div>
</script>
Template scripts
WEB COMPONENTS - HTML TEMPLATES
Defining HTML templates
<template id="template">
<style>
...
</style>
<div>
<h1>Web Components</h1>
<img src="image.jpeg">
</div>
</template>
WEB COMPONENTS - HTML TEMPLATES
Initialising the template
<div id="host"></div>
<script>
var template = document.querySelector('#template');
var clone = document.importNode(template.content, true);
var host = document.querySelector('#host');
host.appendChild(clone);
</script>
<div id="host"></div>
<script>
var template = document.querySelector('#template');
var clone = document.importNode(template.content, true);
var host = document.querySelector('#host');
host.appendChild(clone);
</script>
<div id="host"></div>
<script>
var template = document.querySelector('#template');
var clone = document.importNode(template.content, true);
var host = document.querySelector('#host');
host.appendChild(clone);
</script>
<div id="host"></div>
<script>
var template = document.querySelector('#template');
var clone = document.importNode(template.content, true);
var host = document.querySelector('#host');
host.appendChild(clone);
</script>
WEB COMPONENTS - HTML TEMPLATES
if(!window.HTMLTemplateElement) {
....
}
data:image/s3,"s3://crabby-images/fc313/fc31338b3dcf4a263ac40928c3f5a9c83e4fb666" alt=""
Feature detection
WEB COMPONENTS - CUSTOM ELEMENTS
Build and use custom tags
data:image/s3,"s3://crabby-images/94b87/94b871e920c001626cedd09b5cdfda775dc33df4" alt=""
data:image/s3,"s3://crabby-images/ab6ba/ab6ba02b21923b56dc1aa3296692fd482ceaff33" alt=""
WEB COMPONENTS - CUSTOM ELEMENTS
Building a custom element
var XComponent = document.registerElement('x-component');
<x-component></x-component>
var XComponent = document.registerElement('x-component');
var dom = new XComponent();
document.body.appendChild(dom);
document.registerElement('x-component');
var dom = document.createElement('x-component');
document.body.appendChild(dom);
WEB COMPONENTS - CUSTOM ELEMENTS
Extending a custom element
var proto = Object.create(HTMLElement.prototype);
proto.name = 'Custom Element';
proto.alert = function() {
alert('This is ' + this.name);
};
document.registerElement(
'x-component', {
prototype: proto
});
data:image/s3,"s3://crabby-images/39166/39166ed7e0f01833eff3d0765e7f49f694ea818b" alt=""
var proto = Object.create(HTMLElement.prototype);
proto.name = 'Custom Element';
proto.alert = function() {
alert('This is ' + this.name);
};
document.registerElement(
'x-component', {
prototype: proto
});
var proto = Object.create(HTMLElement.prototype);
proto.name = 'Custom Element';
proto.alert = function() {
alert('This is ' + this.name);
};
document.registerElement(
'x-component', {
prototype: proto
});
WEB COMPONENTS - CUSTOM ELEMENTS
Extending an existing element
var XComponent = document.registerElement('x-component', {
extends: 'input',
prototype: Object.create(HTMLInputElement.prototype)
});
<input is="x-component">
data:image/s3,"s3://crabby-images/525c2/525c2f12359619ad3431dc78a2b8fd93a0020430" alt=""
data:image/s3,"s3://crabby-images/b173e/b173e9294157225b579ccf8bf35c200ccdb376c9" alt=""
Custom element example: github
WEB COMPONENTS - CUSTOM ELEMENTS
Lifecycle hooks
Callback name | Called when |
createdCallback | an instance of the element is created |
attachedCallback | an instance was inserted into the document |
detachedCallback | an instance was removed from the document |
attributeChangedCallback(attrName, oldVal, newVal) | an attribute was added, removed, or updated |
WEB COMPONENTS - CUSTOM ELEMENTS
Feature detection
if(!document.registerElement) {
....
}
data:image/s3,"s3://crabby-images/56786/56786478374890a5e2e2720396f9891cbb8088f8" alt=""
WEB COMPONENTS - SHADOW DOM
data:image/s3,"s3://crabby-images/cd1cc/cd1cc680bf02db7f977a0da9591d5f31bc4fb25b" alt=""
WEB COMPONENTS - SHADOW DOM
data:image/s3,"s3://crabby-images/9ddef/9ddef6b3989f83920d7644b51158d042b7b16d99" alt=""
WEB COMPONENTS - SHADOW DOM
<div id="host"></div>
<script>
var host = document.querySelector('#host');
var root = host.createShadowRoot();
var div = document.createElement('div');
div.textContent = 'This is Shadow DOM';
root.appendChild(div);
</script>
Building the Shadow DOM
data:image/s3,"s3://crabby-images/1e4c8/1e4c88fc4fd31fda0f46e69b8f84b65f39065efb" alt=""
<div id="host"></div>
<script>
var host = document.querySelector('#host');
var root = host.createShadowRoot();
var div = document.createElement('div');
div.textContent = 'This is Shadow DOM';
root.appendChild(div);
</script>
<div id="host"></div>
<script>
var host = document.querySelector('#host');
var root = host.createShadowRoot();
var div = document.createElement('div');
div.textContent = 'This is Shadow DOM';
root.appendChild(div);
</script>
WEB COMPONENTS - SHADOW DOM
Structure of shadow DOM
document
element (shadow host)
shadow root
contents
WEB COMPONENTS - SHADOW DOM
<div id="host">Light DOM</div>
<script>
var container = document.querySelector('#host');
var root1 = container.createShadowRoot();
var root2 = container.createShadowRoot();
root1.innerHTML = '<div>Root 1 FTW</div>';
root2.innerHTML = '<div>Root 2 FTW</div>';
</script>
Multiple shadow roots
data:image/s3,"s3://crabby-images/d7c8f/d7c8fd9730f7876e3cc26b3d0c76f6056efeff4c" alt=""
WEB COMPONENTS - SHADOW DOM
<h1>Outer html</h1>
<div id="host">this content will be replaced</div>
<template id="hostTemplate">
<style>
h1 {
border: 1px solid black;
font-size:16px;
color:blue;
}
:host {
color:red;
}
</style>
<div class="outer">
<h1> Host HTML </h1>
styled from the shadow DOM
</div>
</template>
Using templates
WEB COMPONENTS - SHADOW DOM
<script>
var shadow = document.querySelector("#host").createShadowRoot();
var template = document.querySelector("#hostTemplate");
var clone = document.importNode(template.content, true);
shadow.appendChild(clone);
</script>
Using templates
data:image/s3,"s3://crabby-images/6c455/6c4557dc68d4c0766b46e4ea535fdf4326c78d2d" alt=""
<script>
var shadow = document.querySelector("#host").createShadowRoot();
var template = document.querySelector("#hostTemplate");
var clone = document.importNode(template.content, true);
shadow.appendChild(clone);
</script>
<script>
var shadow = document.querySelector("#host").createShadowRoot();
var template = document.querySelector("#hostTemplate");
var clone = document.importNode(template.content, true);
shadow.appendChild(clone);
</script>
<h1>Outer html</h1>
<div id="host">this content will be replaced</div>
WEB COMPONENTS - SHADOW DOM
Using templates
data:image/s3,"s3://crabby-images/bc03a/bc03a7d1f4a288d373dae8348a6f59667f08e3e0" alt=""
WEB COMPONENTS - SHADOW DOM
<div id="host">
<h1>This is Shadow DOM</h1>
</div>
Adding host element content
data:image/s3,"s3://crabby-images/0eb93/0eb93111dd43846c038dfdc8c0afa676a4bc4cf3" alt=""
<template id="hostTemplate">
<style>
...
</style>
<div id="container">
<span> Something something dark side </span>
<content select="h1"></content> // Insert h1 here
</div>
</template>
WEB COMPONENTS - SHADOW DOM
<div id="host">
<h2>Andrei</h2>
<h2>Antal</h2>
<div id="title">Frontend developer</div>
<h4 class="f">wannabe</h4>
</div>
<template id="hostTemplate">
<header>
<content select="h2"></content>
</header>
<section>
<content select="#title"></content>
</section>
<footer>
<content select=".f"></content>
</footer>
</template>
Multiple insertion points
WEB COMPONENTS - SHADOW DOM
data:image/s3,"s3://crabby-images/f97f9/f97f95db21f280b47ed8edb133e9501cf2f197df" alt=""
WEB COMPONENTS - SHADOW DOM
if(!document.body.createShadowRoot) {
....
}
Feature detection
data:image/s3,"s3://crabby-images/87d9a/87d9a1d09b75c6f21c39ad6efa3fbad0f04b6aae" alt=""
WEB COMPONENTS - HTML IMPORTS
<!-- warning.html -->
<div class="warning">
<style scoped>
h3 {
color: red;
}
</style>
<h3>Warning!</h3>
<p>This page is under construction</p>
</div>
<div class="outdated">
<h3>Heads up!</h3>
<p>This content may be out of date</p>
</div>
Importing HTML documents into other HTML documents
WEB COMPONENTS - HTML IMPORTS
<head>
<link rel="import" href="warnings.html">
</head>
Importing HTML documents into other HTML documents
// Grab DOM from warning.html's document.
var el = content.querySelector('.warning');
<body>
...
<script>
var link = document.querySelector('link[rel="import"]');
var content = link.import;
document.body.appendChild(el.cloneNode(true));
</script>
</body>
WEB COMPONENTS - HTML IMPORTS
<head>
<link rel="import" href="bootstrap.html">
</head>
A practical example - including Bootstrap
<link rel="stylesheet" href="bootstrap.css">
<link rel="stylesheet" href="fonts.css">
<script src="jquery.js"></script>
<script src="bootstrap.js"></script>
<script src="bootstrap-tooltip.js"></script>
<script src="bootstrap-dropdown.js"></script>
...
<!-- scaffolding markup -->
<template>
...
</template>
WEB COMPONENTS - HTML IMPORTS
<!-- import.html -->
<template>
<h1>Hello World!</h1>
<!-- Img is not requested until the <template> goes live. -->
<img src="world.png">
<script>alert("Executed when the template is activated.");</script>
</template>
<head>
<link rel="import" href="import.html">
</head>
<body>
<div id="container"></div>
<script>
var link = document.querySelector('link[rel="import"]');
// Clone the <template> in the import.
var template = link.import.querySelector('template');
var clone = document.importNode(template.content, true);
document.querySelector("#container").appendChild(clone);
</script>
</body>
Including templates
<!-- import.html -->
<template>
<h1>Hello World!</h1>
<!-- Img is not requested until the <template> goes live. -->
<img src="world.png">
<script>alert("Executed when the template is activated.");</script>
</template>
<head>
<link rel="import" href="import.html">
</head>
<body>
<div id="container"></div>
<script>
var link = document.querySelector('link[rel="import"]');
// Clone the <template> in the import.
var template = link.import.querySelector('template');
var clone = document.importNode(template.content, true);
document.querySelector("#container").appendChild(clone);
</script>
</body>
<!-- import.html -->
<template>
<h1>Hello World!</h1>
<!-- Img is not requested until the <template> goes live. -->
<img src="world.png">
<script>alert("Executed when the template is activated.");</script>
</template>
<head>
<link rel="import" href="import.html">
</head>
<body>
<div id="container"></div>
<script>
var link = document.querySelector('link[rel="import"]');
// Clone the <template> in the import.
var template = link.import.querySelector('template');
var clone = document.importNode(template.content, true);
document.querySelector("#container").appendChild(clone);
</script>
</body>
WEB COMPONENTS - HTML IMPORTS
if(!"import" in document.createElement("link")) {
....
}
Feature detection
data:image/s3,"s3://crabby-images/ba7ad/ba7ad1066e246cfe59ec85afe2704d02943260f3" alt=""
WEB COMPONENTS
Browser compatibility
POLYMER
- WEB COMPONENTS MADE SWEET -
- build on top of web components - by GOOGLE
- offers syntactic sugar for fast creation of reusable components
- adds a few tricks to components (eg. data binding)
- EXTENSIVE POLYMER ELEMENT CATALOG
data:image/s3,"s3://crabby-images/db6bd/db6bdfc9b18e6ebbe477dc585d2c8d29cd11f0d1" alt=""
POLYMER - element starter
<link rel="import" href="../bower_components/polymer/polymer.html">
.....
<link rel="import" href="comp/hello-folks.html">
</head>
<body>
<hello-folks></hello-folks>
....
<script>
Polymer({
is: "hello-folks",
})
</script>
<dom-module id="hello-folks">
<style>
h1 { color: red; }
h3 { color: blue; }
</style>
<template>
<h1>Hello</h1>
<h3>folks</h3>
</template>
</dom-module>
POLYMER - CUSTOM Properties
<script>
Polymer({
is: "hello-folks",
properties : {
aSimpleProp : String,
aComputedProp : {
type: String,
computed: "comp(aSimpleProp)"
}
},
comp(prop){
return "computed: " + prop
}
});
</script>
.....
<hello-folks a-simple-prop="a prop"></hello-folks>
....
<dom-module id="hello-folks">
<style>
h1 { color: red; }
h3 { color: blue; }
</style>
<template>
<h1>Hello</h1>
<h3>folks <span>{{aComputedProp}}</span></h3>
</template>
</dom-module>
POLYMER - DATA BINDING
<dom-module id="hello-folks">
<style>
h1 { color: red; }
h3 { color: blue; }
</style>
<template>
Name: <input value={{nameValue::input}}>
<h1>Hello</h1>
<h3> <span>[[nameValue]]</span></h3>
<img src$="https://www.example.com/profiles/{{userId}}.jpg">
</template>
</dom-module>
- [[ property]] - one way data binding
- {{ property }} - automatic (one/two way) data binding
POLYMER - DATA BINDING
<!-- Property binding -->
<my-element selected="{{value}}"></my-element>
<!-- results in <my-element>.selected = this.value; -->
<!-- Attribute binding -->
<my-element selected$="{{value}}"></my-element>
<!-- results in <my-element>.setAttribute('selected', this.value); -->
<!-- class -->
<div class$="{{foo}}"></div>
<!-- style -->
<div style$="{{background}}"></div>
<!-- href -->
<a href$="{{url}}">
<!-- label for -->
<label for$="{{bar}}"></label>
<!-- dataset -->
<div data-bar$="{{baz}}"></div>
Binding properties and attributes
Native elements binding
POLYMER - Iterating over data
<script>
Polymer({
is: "hello-folks",
ready: function(){
this.things = [
{name: "a thing"},
{name: "some other thing"},
{name: "the last thing"}]
}
});
</script>
.......
<dom-module id="things-list">
<template>
<h1>A list of things</h1>
<ul>
<template is="dom-repeat"
items={{things}}
as="thing"
index-as="thing_no">
<li>thing <i>[[thing_no+1]]</i> - <b>[[thing.name]]</b></li>
</template>
</ul>
</template>
........
.......
<dom-module id="things-list">
<template>
<h1>A list of things</h1>
<ul>
<template is="dom-repeat"
items={{things}}
as="thing"
index-as="thing_no">
<li>thing <i>[[thing_no+1]]</i> - <b>[[thing.name]]</b></li>
</template>
</ul>
</template>
........
.......
<dom-module id="things-list">
<template>
<h1>A list of things</h1>
<ul>
<template is="dom-repeat"
items={{things}}
as="thing"
index-as="thing_no">
<li>thing <i>[[thing_no+1]]</i> - <b>[[thing.name]]</b></li>
</template>
</ul>
</template>
........
.......
<dom-module id="things-list">
<template>
<h1>A list of things</h1>
<ul>
<template is="dom-repeat"
items={{things}}
as="thing"
index-as="thing_no">
<li>thing <i>[[thing_no+1]]</i> - <b>[[thing.name]]</b></li>
</template>
</ul>
</template>
........
.......
<dom-module id="things-list">
<template>
<h1>A list of things</h1>
<ul>
<template is="dom-repeat"
items={{things}}
as="thing"
index-as="thing_no">
<li>thing <i>[[thing_no+1]]</i> - <b>[[thing.name]]</b></li>
</template>
</ul>
</template>
........
.......
<dom-module id="things-list">
<template>
<h1>A list of things</h1>
<ul>
<template is="dom-repeat"
items={{things}}
as="thing"
index-as="thing_no">
<li>thing <i>[[thing_no+1]]</i> - <b>[[thing.name]]</b></li>
</template>
</ul>
</template>
........
POLYMER - ELEMENT COMPOSITION
<link rel="import" href="bower_components/polymer/polymer.html">
<link rel="import" href="components/entry-detail.html">
<dom-module id="entry-list">
<style>
</style>
<template>
<h1>ELEMENT LIST</h1>
<template is="dom-repeat" items="{{entries}}">
<h3> Element #<span>[[index]]</span></h3>
<entry-detail id="{{item.id}}">
</template>
</template>
</dom-module>
<script>
Polymer({
is: "hello-folks",
ready: function(){
this.entries = [...]
}
})
</script>
POLYMER - Routing
other options
CONCLUSION
- HTML sucks...
- CSS sucks...
data:image/s3,"s3://crabby-images/349f0/349f0f9b4f4cbfb9da2c8da4df3b40a073e09823" alt=""
CONCLUSION
data:image/s3,"s3://crabby-images/11f2d/11f2d7ff10e5769aef9c0d6de6eb1c6ce98a0c5e" alt=""
CONCLUSION
data:image/s3,"s3://crabby-images/70bf4/70bf424acf941d5235a7df75ef3b5adafadcf1aa" alt=""
THANKS
data:image/s3,"s3://crabby-images/a14f1/a14f1753d803546e4a7d6e647fece38429cf786d" alt=""
data:image/s3,"s3://crabby-images/56bd1/56bd1edba47db327f1378be41d49fc64703afcb8" alt=""
Reach me @
fb/antal.a.andrei
@andrei_antal
antal.andrei@icloud.com
Frontend Meetup - Web components
By Andrei Antal
Frontend Meetup - Web components
- 1,493