Text
data:image/s3,"s3://crabby-images/168a7/168a7fd05dac7e70d4976f72a636a64016d418a3" alt=""
Emmanuel DEMEY
@EmmanuelDemey
GDE Web Technologies
data:image/s3,"s3://crabby-images/e1d85/e1d854305cd1ffbd824b3ad7cf44c4a23f4e5ad1" alt=""
PWAs are the missing technology link to bring the best features of the web to solve the issues of native apps and closed marketplaces.
Chris Heilmann
Alibaba
(e-commerce/China)
- 76% de conversions en plus entre navigateurs;
- 14% en plus d'utilisateurs actifs/mois sous iOS; 30% en plus sous Android;
- 4X plus de taux d'itéraction à partir de "Add to Homescreen".
Source : https://developers.google.com/web/showcase/2016/pdfs/alibaba.pdf
data:image/s3,"s3://crabby-images/3e9aa/3e9aa7774226c54e0472b1854b2ac795b85351c2" alt=""
FlipKart
(e-commerce/India)
- Temps passé sur Flipkart lite vs. expériences mobile précedentes: 3.5 minutes vs 70 seconds.
- 3x plus de temps passé sur le site;
- 40% en plus de taux d'engagement
- 70% en plus de taux de conversion entre ceux arrivant via "Add to Homescreen";
- 3x mois d'usage de données.
https://developers.google.com/web/showcase/2016/flipkart
data:image/s3,"s3://crabby-images/d61a4/d61a48722248ce5b1548c0fe12da84357513b4ea" alt=""
data:image/s3,"s3://crabby-images/08ad0/08ad0c39006d33f950559e84b2ef06b60f94051e" alt=""
data:image/s3,"s3://crabby-images/393ef/393ef18a2e59b8b50281ad465da032e9c2678fca" alt=""
data:image/s3,"s3://crabby-images/765c9/765c9013038b1d76f8a60b6d780007a9a271cd82" alt=""
data:image/s3,"s3://crabby-images/d37fa/d37fa7a1281ab181cc10b1603cab7cd76becd5d0" alt=""
Caracteristiques
Connectivity-independant
Linkable
Installable
Re-engageable
Discoverable
Fresh
App-Like
Les prérequis
data:image/s3,"s3://crabby-images/56de0/56de03abca592aec47c23804b1608a5ed2b826aa" alt=""
Les prérequis
data:image/s3,"s3://crabby-images/5052d/5052dcd3d91f4a3d3de740e3e4499814c423498c" alt=""
data:image/s3,"s3://crabby-images/8201f/8201fbcfbb8af0b43346a68f644d9c838b35c245" alt=""
Vous avez dit Progressive ?
if("geolocation" in navigator) {
navigator.geolocation.getCurrentPosition(pos => {
});
} else {
}
- Feature Detection
- Balises ignorées
<meta property="og:title" content="titre"/>
<meta property="og:type" content="website"/>
Par où commencer?
data:image/s3,"s3://crabby-images/49d7d/49d7d8665816de0e5222455f5a8bd4e01e004d4d" alt=""
Les fonctionnalités présentées font parties juste d'un toolkit.
Audit
data:image/s3,"s3://crabby-images/a8d06/a8d0667991ecffc2b137f06354e40a3b7eea33b1" alt=""
LightHouse
- Outil permettant d'auditer une application web
- Intégré à Chrome Devtools (onglet Audits) ou via un module NPM
- Retourne un score (sur 100) indiquant le taux de conformité avec les PWA
- Règles sur différents domaines : Performance, Sécurité, A11Y, RWD, ...
npm i -g lighthouse
lighthouse https://google.fr
LightHouse
LightHouse
data:image/s3,"s3://crabby-images/f63ac/f63ac5a2abfb2e655aca78921a7ae8ddfcfeb45b" alt=""
Le Web App Manifest
Le Manifest c'est...
-
Méta-données associées à une web app (format json)
-
Permet "d'installer" une web app
- Il "fait autorité"
Lien vers le Manifest
<!doctype html>
<html lang="fr">
<head>
<meta http-equiv="Content-Type"
content="text/html; charset=utf-8" />
<meta content="width=device-width, initial-scale=1.0"
name="viewport" />
<title>Ma première PWA</title>
<link rel="manifest" href="manifest.json">
</head>
<body>
</body>
</html>
Conditions d'installation
- name meta-data
- Une icône de taille adequate
- Connexion sécurisée
- Doit "fonctionner" sans connexion
- Visites récurrentes
- Service Worker enregisté
data:image/s3,"s3://crabby-images/dd992/dd99226f21a039aa08801e5a2da643e86633562b" alt=""
Style
data:image/s3,"s3://crabby-images/81bd4/81bd444ebd25b875ad7e1e926376ad0d66d7b6e9" alt=""
Display
Manifest - Validation
data:image/s3,"s3://crabby-images/40f19/40f19510e05e38e318d90e89971ae2b34618a14b" alt=""
https://manifest-validator.appspot.com
Support
data:image/s3,"s3://crabby-images/156a4/156a49eab3c536cebf201df43d31a25db7ef5810" alt=""
Install banner
data:image/s3,"s3://crabby-images/88ae0/88ae0a33b10d9998708181984b6235e15b4dbe44" alt=""
window.addEventListener(
'beforeinstallprompt', (e) => {
if (isLoggedIn()) {
e.preventDefault();
}
});
Manifest - Event
Service Worker
data:image/s3,"s3://crabby-images/4eff9/4eff9b53b2b4a5d5a8865974903510d6f6d312e4" alt=""
data:image/s3,"s3://crabby-images/37714/3771444eea5094f11e1c31f94f4406d05f4aef8e" alt=""
data:image/s3,"s3://crabby-images/4d6f1/4d6f1ab11a1426a792e368780facfe8f17a6420b" alt=""
Service Worker
Service Worker
- Proxy entre votre application et votre serveur
- Nécessite HTTPs
- Pas d'accès au DOM
- Fonctionnalité de Cache, Push, Geofencing, Background Sync
- Evenement d'un SW : install, activate, fetch et sync
if ('serviceWorker' in navigator) {
navigator.serviceWorker.register('./sw.js')
.then(req => {
console.log('Success');
}).catch(error => {
console.log('Error');
});
};
Service Worker
self.addEventListener('fetch', event => {
event.respondWith(
fetch(event.request)
);
});
Service Worker
self.addEventListener('fetch', event => {
event.respondWith(
fetch(event.request)
.then(response => {
if(response.status === 404) {
...
}
return response;
})
);
});
interface Request {
method
url
headers
context
referrer
referrerPolicy
mode
credentials
redirect
integrity
cache
bodyUsed
}
self.addEventListener('fetch', event => {
/*
let options = {
status: 200,
statustext: 'OK',
headers: { ... }
};
*/
event.respondWith(
new Response('Hello World', /*opts*/)
);
});
Support
data:image/s3,"s3://crabby-images/d3f64/d3f6409be3d69247ff8a65a7170d41dd173f3ec0" alt=""
App Shell
- Rendre le layout du site accessible offline
- Afficher la structure du site le plus rapidement possible
- Utilisation de l'API Cache
- Precache des ressources statiques durant l'événement install
- Réutilisation du Cache dans l'événement fetch
App Shell
App Shell - Cache
CacheStorage.open(cacheName)
Cache.match(request, options)
Cache.matchAll(request, options)
Cache.add(request)
Cache.addAll(requests)
Cache.put(request, response)
Cache.delete(request, options)
App Shell - install
const cacheName = 'codelab';
const filesToCache = [
'/',
'/script.js',
'/css/style.css'
];
self.addEventListener('install', e => {
e.waitUntil(
caches.open(cacheName).then(cache => {
return cache.addAll(filesToCache);
})
);
});
App Shell - fetch
self.addEventListener('fetch', e => {
e.respondWith(
caches.match(e.request)
.then(response => {
return response || fetch(e.request);
})
);
});
App Shell - activate
self.addEventListener('activate', e => {
e.waitUntil(
caches.keys().then(keyList => {
return Promise.all(keyList.map(key => {
if (key !== cacheName) {
return caches.delete(key);
}
}));
})
);
});
Prévenir vos utilisateurs
data:image/s3,"s3://crabby-images/6a4a9/6a4a937ebcc55048668fbb7454d5ff56fb85cea8" alt=""
data:image/s3,"s3://crabby-images/9bb98/9bb985982e639c0960664f035c05adf7f2810c14" alt=""
online / offline
if(!navigator.onLine) {
...
}
window.addEventListener("offline", () => {
...
}, false);
window.addEventListener("online", () => {
...
}, false);
IndexedDB
- API permettant de stocker des données côté client (clé => valeur)
- La donnée stockée peut être un objet complexe
- Base de données non relationnelle
- API asynchrone, utilisant des callbacks :(
- Syntaxe assez verbeuse
- Vendor Prefixes
- Cursor
- Indexes
- Transaction
- Utilisation de librairies externes pour simplifier l'utilisation
IndexedDB
const indexedDB = window.indexedDB || window.mozIndexedDB || window.webkitIndexedDB;
// Open (or create) the database
const open = indexedDB.open("Codelab", 1);
// Create the schema
open.onupgradeneeded = function() {
let db = open.result;
let store = db.createObjectStore("Conference", {keyPath: "id"});
let index = store.createIndex("SpeakerIndex", ["speaker"]);
};
open.onsuccess = function() {
// Start a new transaction
let db = open.result;
let tx = db.transaction("Conference", "readwrite");
let store = tx.objectStore("Conference");
let index = store.index("SpeakerIndex");
// Add some data
store.put({id: 12345, speaker: "Manu", "title": "PWA");
store.put({id: 67890, speaker: "Gauthier", "title": "VueJS");
// Query the data
let getPWA = store.get(12345);
let getVueJS = index.get(["Gauthier"]);
getPWA.onsuccess = function() {
console.log(getPWA.result);
};
getVueJS.onsuccess = function() {
console.log(getBob.result);
};
// Close the db when the transaction is done
tx.oncomplete = function() {
db.close();
};
}
- Librairie plus simple à utiliser
- Fallback vers WebSQL et localStorage
- Support des callbacks et des promises
- API similaire à localStorage
LocalForage
localforage.setItem('somekey', {}).then(value => {
console.log(value);
})
localforage.getItem('somekey').then(value => {
console.log(value);
});
localforage.removeItem('somekey').then(() => {
console.log('Key is cleared!');
});
IndexedDB
data:image/s3,"s3://crabby-images/1ca29/1ca2937b06d77ef71912d359c363a10e0488f8d3" alt=""
data:image/s3,"s3://crabby-images/cde71/cde71c45a4fc853aa2966bbac271e618cac5b319" alt=""
data:image/s3,"s3://crabby-images/bc24a/bc24a2075421d4c514c134f2ab7f033c2a9c7a67" alt=""
Support
data:image/s3,"s3://crabby-images/e92d9/e92d96baf53bc58589d447577255955248ace1e9" alt=""
Background Sync
data:image/s3,"s3://crabby-images/efe85/efe853ebee02fb59100207444b8a5634fbee4aad" alt=""
- Ajout d'un événement Sync aux Service Workers
- Assure l'envoie d'une requête lorsque l'utilisateur est offline
- Traitement exécuté tant que la Promise est rejetée
Background Sync
Background Sync
self.addEventListener('sync', event => {
if (event.tag == 'SYNC_SETTINGS') {
event.waitUntil(updateSettings());
}
});
navigator.serviceWorker.getRegistration()
.then(registration => {
registration.sync.register(
'SYNC_SETTINGS').then(() => {
console.log('Sync registered');
});
})
HTTP2
HTTP1
- 15 ans d'existence
- Applications de plus en plus lourdes
- Navigateur limitant le nombre de requêtes simultanées
- Utilisation de mécanismes permettant de réduire le temps de chargement
- Minification
- Concaténation
- Domain Sharding
HTTP2
- Compatible avec HTTP1
- Fonctionnement avec TLS
- Diminue le nombre d'allers retours
- Compression des Headers
- Limite le nombre de connexions
- Système de multiplexage
- Server Push
HTTP2
data:image/s3,"s3://crabby-images/aebff/aebff38f6f0b7c69cc9d8c31a08012852e80596a" alt=""
HTTP2 avec Express
const fs = require('fs')
const javascript = fs.readFileSync('./app/assets/script.js');
app.get('/', (req, res) => {
res.push('/script.js', {
response: {'content-type': 'application/javascript'}})
.end(javascript)
res.sendFile(__dirname + '/app/index.html');
})
const options = {
key: fs.readFileSync('./server.key'),
cert: fs.readFileSync('./server.crt')
};
require('spdy').createServer(options, app).listen(3002);
Pour aller plus loin...
- Trusted Web Activities
- Notifications
- Payment Request API
- One-Tap Sign-In / Sign-Up
- Web Share API
data:image/s3,"s3://crabby-images/12833/12833a659aee64f931a76ec73f6cd935f090b3c0" alt=""
Emmanuel DEMEY
@EmmanuelDemey
data:image/s3,"s3://crabby-images/502d5/502d52f7df2d521d300b118ff5464ea3816cc36b" alt=""
Copy of Codelab PWA
By Emmanuel Demey
Copy of Codelab PWA
- 2,121