Progressive Web App, 简称 PWA,是提升 Web App 的体验的一种新方法,能给用户原生应用的体验。 create-react-app在2.0版本后,自动加上了对server worker的配置,本项目是基于create-react-app2.0版本的增强,添加了推送功能,修改了缓存策略以及增加了web app的兼容和配置。
- 2018-11-02 增加http缓存的demo
- 2018-11-14 增加基于PWA做的网页崩溃demo(注:需要手动在控制台写一个死循环来触发网页崩溃,看是否监控到)
npm i
npm run pwa
npm run start //启开发环境
npm run build //打生产包
npm run server //启PWA的服务端
npm run rm //删除pushSubscription文件(此文件夹模拟装订阅对象的数据库)
npm run proxy //在客户端无操作的时候模拟一次服务端推送
npm run mutiProxy //在客户端无操作的时候模拟多次服务端推送
npm run pwa //打包,删除pushSubscription文件,启PWA的服务端
- 要求 HTTPS 的环境,本地调试支持host为localhost 或者 127.0.0.1
- Service Worker 的缓存机制是依赖 Cache API 实现的
- 依赖 HTML5 fetch API,PWA可以拦截请求且发送请求
- 依赖 Promise 实现,链式调用
生命周期分为这么几个状态 安装中, 安装后, 激活中, 激活后, 废弃,通过onstatechange来监听状态进行dom操作及其他。
在sw注册时添加监听事件,此时上下文不在window上,不能进行dom操作
- 注册
if ('serviceWorker' in navigator) {
navigator.serviceWorker.register('/service-worker.js').then(function (reg) {
reg.onupdatefound = function () {
var installingWorker = reg.installing
installingWorker.onstatechange = function () {
switch (installingWorker.state) {
//监听当前状态
case 'installed':; break
case 'activating':; break
case 'activated':; break
}
}
}
})
.catch(function (e) {
console.error('Error during service worker registration:', e)
})
- 监听
self.addEventListener('install', function () {
self.skipWaiting();
});
self.addEventListener('activate', function (e) {
});
self.addEventListener('fetch', function (e) {
});
self.addEventListener('push', function (e) {
});
- 缓存
// 监听 service worker 的 install 事件,初始缓存
this.addEventListener('install', function (event) {
// 如果监听到了 service worker 已经安装成功的话,就会调用 event.waitUntil 回调函数
event.waitUntil(
// 安装成功后操作 CacheStorage 缓存,使用之前需要先通过 caches.open() 打开对应缓存空间。
caches.open('my-test-cache-v1').then(function (cache) {
// 通过 cache 缓存对象的 addAll 方法添加 precache 缓存
return cache.addAll([
'/',
'/index.html',
'/main.css',
'/main.js',
'/image.jpg'
]);
})
);
});
// 监听 service worker 的 fetch 事件,拦截fetch请求,缓存资源
this.addEventListener('fetch', function (event) {
event.respondWith(
caches.match(event.request).then(function (response) {
// 来来来,代理可以搞一些代理的事情
// 如果 Service Worker 有自己的返回,就直接返回,减少一次 http 请求
if (response) {
return response;
}
// 如果 service worker 没有返回,那就得直接请求真实远程服务
var request = event.request.clone(); // 把原始请求拷过来
return fetch(request).then(function (httpRes) {
// http请求的返回已被抓到,可以处置了。
// 请求失败了,直接返回失败的结果就好了。。
if (!httpRes || httpRes.status !== 200) {
return httpRes;
}
// 请求成功的话,将请求缓存起来。
var responseClone = httpRes.clone();
caches.open('my-test-cache-v1').then(function (cache) {
cache.put(event.request, responseClone);
});
return httpRes;
});
})
);
});
// 自动更新缓存
// 监听 service worker 的 install 事件,安装阶段跳过等待,直接进入 active
self.addEventListener('install', function (event) {
event.waitUntil(self.skipWaiting());
});
// 监听 service worker 的 activate 事件,更新客户端,清理旧版本
self.addEventListener('activate', function (event) {
event.waitUntil(
Promise.all([
// 更新客户端
self.clients.claim(),
// 清理旧版本
caches.keys().then(function (cacheList) {
return Promise.all(
cacheList.map(function (cacheName) {
if (cacheName !== 'my-test-cache-v1') {
return caches.delete(cacheName);
}
})
);
})
])
);
});
也可以直接用谷歌提供的workbox的库里做缓存策略,可以通过谷歌提供的webpack插件workbox-webpack-plugin来自动注入
/* globals workbox */
workbox.core.setCacheNameDetails({
prefix: 'pwa',
suffix: 'v1',
precache: 'install-time',
runtime: 'run-time',
googleAnalytics: 'ga'
})
workbox.skipWaiting()
workbox.clientsClaim()
// 设置缓存都数组,默认为[]
self.__precacheManifest = [].concat(self.__precacheManifest || [])
workbox.precaching.suppressWarnings()
// 设置需要缓存的url,默认为__precacheManifest文件里的数组
workbox.precaching.precacheAndRoute(self.__precacheManifest || [])
-
推送
- 服务端基于web-push生成公私钥
// server/index.js var webpush = require('web-push'); // 1.生成公私钥 var vapidKeys = webpush.generateVAPIDKeys(); // 2.设置公私钥 webpush.setVapidDetails( 'mailto:sender@example.com', vapidKeys.publicKey, vapidKeys.privateKey );
- 客户端请求对应的公钥,注册订阅推送,并将生成的订阅对象传回服务端存入数据库
// src/sw-register.js //订阅pwa推送 function subscribe (serviceWorkerReg, publicKey) { // 询问用户是否要订阅消息 serviceWorkerReg.pushManager.subscribe({ // 2. 订阅 userVisibleOnly: true, applicationServerKey: urlBase64ToUint8Array(publicKey) }) .then(function (subscription) { fetch(`/save?body=${JSON.stringify(subscription)}`) .then((res) => { console.log(res) }) }) .catch(function () { if (Notification.permission === 'denied') { // 用户拒绝了订阅请求 } }) }
- 当有推送请求的时候,取出数据库的订阅对象,发送推送
//向订阅的浏览器发送消息 app.get('/push',function(req,res,next){ // pushSubscription 从数据库取出 console.log('[准备推送]',pushSubscription) if(pushSubscription){ webpush.sendNotification(JSON.parse(pushSubscription),'推送demo',{} ) .then(data=>{ console.log('[ 推送成功 ]',JSON.stringify(data)) }).catch(function (err) { console.log('[推送失败]',err) if (err.statusCode === 410 || err.statusCode === 404) { // 从数据库中删除推送订阅对象 } }); } });
- sw对推送的监听
//监听sw的push事件,发起推送 self.addEventListener('push', function (e) { var data = e.data if (e.data) { try { data = data.json() } catch (error) { data = data.text() } console.log('push的数据为:', data) //浏览器推送api self.registration.showNotification('来自PWA的推送',{}) })
- 对推送ui的设置 通过对showNotification函数第一、二个参数进行设置,我们可以自定义推送的内容及ui
//第一参数 : 标题 title
//第二参数 : options
{
// 视觉相关
"body": "<String>",//内容
"icon": "<URL String>",//小图标
"image": "<URL String>",//预览图
"badge": "<URL String>",//手机上通知缩略信息小图标
"vibrate": "<Array of Integers>",//震动
"sound": "<URL String>",//声音
"dir": "<String of 'auto' | 'ltr' | 'rtl'>",//文字方向
// 行为相关
"tag": "<String>",//标签,同一地址的推送是否合并规则
"data": "<Anything>",
"requireInteraction": "<boolean>",//一直显示推送,不会自动消失
"renotify": "<Boolean>",//重新通知,配合tag使用
"silent": "<Boolean>",//推送的时候无震动和声音
// 视觉行为均会影响
"actions": "<Array of Objects>",//自定义按钮
// 定时发送时间戳
"timestamp": "<Long>"
}
-
自定义按钮 我们可以通过自定义按钮和监听sw的推送点击事件来自定义功能,如:跳转其他页面,唤起页面
-
1、自定义按钮对文案,id,icon
//service-worker.js
self.registration.showNotification('来自PWA的推送',{
...
//推送的按钮
actions: [
{
action: 'go-baidu',
title: '去百度',
icon: '/favicon.ico'
},
{
action: 'go-github',
title: '去github',
icon: '/favicon.ico'
}
]
})
- 2、sw监听推送点击事件
//service-worker.js
self.addEventListener('notificationclick', event => {
switch (event.action) {
case 'go-baidu':
console.log('点击了去百度按钮');
break;
case 'go-github':
console.log('点击了去github按钮');
break;
default:
console.log(`Unknown action clicked: '${event.action}'`);
break;
}
event.notification.close();
event.waitUntil(
// 获取所有clients
self.clients.matchAll().then(function (clients) {
if (!clients || clients.length === 0) {
// 当不存在client时,打开该网站
self.clients.openWindow && self.clients.openWindow(' localhost:3004');
return;
}
// 切换到该站点的tab
console.log('[clients]',clients)
clients[0].focus && clients[0].focus();
clients.forEach(function (client) {
// 使用postMessage进行通信
client.postMessage(event.action);
});
})
)
});
- 3、在网页的上下文中监听sw的message事件 必须是在window下监听,否则无法操作dom
//监听serviceWorker通过postmessage传过来的信息
navigator.serviceWorker.addEventListener('message', function (e) {
var action = e.data;
console.log(`receive post-message from sw, action is '${e.data}'`);
switch (action) {
case 'go-baidu':
location.href = 'https://www.baidu.com';
break;
case 'go-github':
location.href = 'https://github.com/asyalas/create-react-app-pwa';
break;
default:
break;
}
});