背景
平台各个子应用静态资源加载速度慢,用户体验有待提升
主要目的:
- 解决资源加载慢问题,提升体验
- 支持应用自启动(原生
app
体验)
项目使用到的技术栈为:React
umi3
qiankun
PWA 介绍
简介
PWA 它不是特指某一项技术,而是应用多项技术来改善用户体验的 Web App
,其核心技术包括 Web App Manifest
,Service Worker
等,用户体验才是 PWA
的核心。
特点:
- 可靠 - 即使在网络不稳定甚至断网的环境下,也能瞬间加载并展现
- 用户体验 - 快速响应,具有平滑的过渡动画及用户操作的反馈
- 用户黏性 - 和
Native App
一样,可以被添加到桌面,具有沉浸式的用户体验
核心技术
Web App Manifest
主要为项目配置manifest.json
,提供浏览器安装PWA
所需的信息,例如应用程序名称和图标等。Web app manifests
允许开发者配置隐藏浏览器多余的 UI(地址栏,导航栏等),让PWA
具有和Native App
一样的沉浸式体验。
Service Worker
- 使用到的时候浏览器会自动唤醒,不用的时候自动休眠
- 可拦截并代理请求和处理返回,可以操作本地缓存,如 CacheStorage,IndexedDB 等
- 离线内容开发者可控
- 能接受服务器推送的离线消息
- 必须在 HTTPS 环境下才能工作
实现过程
注册 Service Worker
- 根目录新建
service-worker.js
文件,用于编写Service Worker
具体逻辑。
主要添加安装、激活、缓存捕获后的处理逻辑(会在后续缓存策略章节详细描述)
- 注册 service worker 服务
在 index.html 文件中注入以下代码
1<script>
2 if ('serviceWorker' in navigator) {
3 window.addEventListener('load', () => {
4 navigator.serviceWorker
5 .register('/service-worker.js')
6 .then(registration => {
7 console.log('SW registered: ', registration);
8 })
9 .catch(registrationError => {
10 console.log('SW registration failed: ', registrationError);
11 });
12 });
13 }
14</script>
我们项目是基于umi开发的,在umi3中在document.ejs
添加script
标签,在umi4中则需要在umirc.ts
中的headScript
添加内容。
安装到主屏幕
- 根目录(或是public目录)下新建
manifest.json
文件,添加需要的配置,具体可参考文档。 Web App Manifest,下面提供一份实例:
1{
2 "short_name": "灵思 Aicity",
3 "name": "灵思 Aicity",
4 "icons": [
5 {
6 "src": "https://xxx/144x144.png",
7 "sizes": "144x144",
8 "type": "image/png"
9 },
10 {
11 "src": "https://xxx/192x192.png",
12 "sizes": "192x192",
13 "type": "image/png",
14 "purpose": "maskable"
15 }
16 ],
17 "start_url": ".",
18 "display": "minimal-ui",
19 "background_color": "#fff",
20 "theme_color": "#fff"
21}
- index.html中进行manifest引入
1<!DOCTYPE html>
2<html lang="en">
3 <head>
4 <meta charset="UTF-8" />
5 <meta http-equiv="X-UA-Compatible" content="IE=edge" />
6 <meta
7 name="viewport"
8 id="scale-view"
9 content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=0"
10 />
11 <link rel="manifest" href="/manifest.json" />
12 <link rel="icon" href="/favicon.ico" type="image/x-icon" />
13 </head>
14</html>
到这一步启动项目之后就会在地址栏多出一个按钮
点击按钮就可以进行web app
安装
安装完文件夹(或桌面)就会多出一个应用信息。
到这web应用的可安装操作就完成了,如果没有出现下载按钮,可打开控制台查看 Application-Manifest相关报错提示!!!
缓存策略
其核心是使用了 service worker
相关技术,详细可参考文档 Service Worker API。
为了减少一些处理细节上的时间,采用了google的插件workbox进行快速集成,省去了很多准备工作。
- 如果你只需要实现一些基本的缓存,不做预加载这些其他操作,workbox GenerateSW这个就足以满足你的需求,以下是一个建议配置:
1import { defineConfig } from 'umi';
2import { GenerateSW } from 'workbox-webpack-plugin';
3
4export default defineConfig({
5
6
7 chainWebpack(memo) {
8 memo.plugin('workbox').use(GenerateSW, {
9
10 cacheId: 'webpack-pwa',
11
12 skipWaiting: true,
13 clientsClaim: true,
14 swDest: 'service-wroker.js',
15 globPatterns: ['**/*.{html,js,css,png.jpg}'],
16 globIgnores: ['service-wroker.js'],
17 runtimeCaching: [
18
19 {
20
21 urlPattern: /.*\.js/,
22
23 handler: 'NetworkFirst'
24 }
25 ]
26 })
27 }
28})
- 遇到复杂点的场景,还是需要一定自由度去编写策略,我们可以选择workbox InjectManifest进行处理,详细实现如下:
umirc.ts
1import { defineConfig } from 'umi';
2import { InjectManifest } from 'workbox-webpack-plugin';
3
4export default defineConfig({
5
6
7 chainWebpack(memo) {
8
9 memo.plugin('workbox').use(InjectManifest, [
10 {
11
12 swSrc: './service-worker.js',
13
14 swDest: 'service-worker.js',
15
16 maximumFileSizeToCacheInBytes: 10 * 1024 * 1024
17 },
18 ]);
19 }
20})
service-worker.js
registerRoute
和GenerateSW
中的runtimeCaching
配置是一样的,主要区分下集中缓存策略:
CacheFirst
会在有缓存的时候返回缓存,没有缓存才会去请求并且把请求结果缓存CacheOnly
只返回缓存,不请求NetworkFirst
请求将会发出,成功的话就返回结果添加到缓存中,如果失败则返回立即缓存NetworkOnly
只请求,不读写缓存StaleWhileRevalidate
类似于 CacheFirst,区别在于在返回 Cache 缓存结果的同时会在后台发起网络请求拿到请求结果并更新 Cache 缓存,如果本来就没有 Cache 缓存的话,直接就发起网络请求并返回结果
1import { registerRoute } from 'workbox-routing';
2import {
3 StaleWhileRevalidate,
4 NetworkFirst,
5 CacheFirst,
6} from 'workbox-strategies';
7
8
9self.__WB_DISABLE_DEV_LOGS = true;
10
11
12registerRoute(
13 /.*(gif|jpg|jpeg|png|svg|otf|woff|woff2|ttf|mp4|pbf).*/,
14 new CacheFirst({
15 cacheName: 'cache-static',
16 expiration: {
17 maxEntries: 1000,
18 maxAgeSeconds: 30 * 24 * 60 * 60,
19 },
20 }),
21);
22
23
24registerRoute(
25 /.*\.js.*/,
26 new StaleWhileRevalidate({
27 cacheName: 'cache-js',
28 expiration: {
29 maxEntries: 100,
30 maxAgeSeconds: 30 * 24 * 60 * 60,
31 },
32 }),
33);
34
35
36registerRoute(
37 /.*\.css.*/,
38 new StaleWhileRevalidate({
39 cacheName: 'cache-style',
40 expiration: {
41 maxEntries: 100,
42 maxAgeSeconds: 30 * 24 * 60 * 60,
43 },
44 }),
45);
46
47
48registerRoute(
49 /^.*(-api\.).*$/,
50 new NetworkFirst({
51 cacheName: 'cache-api',
52 cacheableResponse: {
53 statuses: [200],
54 },
55 }),
56);
57
58
59registerRoute(
60 ({ url }) => {
61
62 const reg = /.*(\.flv|\.m3u8|\.ts|\.tsx).*/;
63 return !reg.test(url.href);
64 },
65 new NetworkFirst({
66 cacheName: 'cache-others',
67 cacheableResponse: {
68 statuses: [200],
69 },
70 }),
71);
72
到这再请求页面的时候Application => Storage =>Cache storage
里就能看到被缓存下来的文件了。
预加载实现
在上述基础上,我们对service-worker.js
进行一些改造,让他可以进行预缓存。其实在配置InjectManifest
的时候也提到了,workbox
会帮我们生成一份构建产物的映射(manifest
)。借助这个我们可以实现预缓存。具体操作如下:
1import { precacheAndRoute } from 'workbox-precaching';
2
3
4
5
6const routes = ['/xxx', '/aaa'];
7
8const precacheList = self.__WB_MANIFEST || [];
9precacheAndRoute([...precacheList, ...routes]);
10
11
12
13
14const precacheApps = ['portrait', 'map', 'graph'];
15precacheApps.forEach((app) => {
16 fetch(`/${app}/asset-manifest.json`)
17 .then((response) => response.json())
18 .then((data) => {
19 const files = Object.values(data).filter(
20 (file) => !file.includes('/index.html'),
21 );
22 precacheAndRoute(files);
23 })
24 .catch((e) =>
25 console.error(
26 `Error: can not fetch ${app} app asset-manifest.json`,
27 ),
28 );
29});
Features
多 Tab 探索
这是目前比较常见的一个问题,在pwa应用内使用window.open()
打开新tab会跳转到浏览器中,这样体验感是很不好的;对此我们想到了两种解决方案:
- 自己实现一套 Tab UI
大概长这样…
主要就是通过拦截window.open()
,维护一份tabs的相关数据,使用iframe
渲染每一个子页面
- 优点:功能齐全,UI展示效果好
- 缺点:性能差(主要原因你想想,开20个自定义tab,实际使用的是一个浏览器Tab性能,这不得炸了…);刷新、全屏、多个pwa应用等场景数据维护成本高;
- 使用chrome实验性api(目前楼主的方案)
在大概chrome@89
版本左右开始,chrome 提出了 display: tabbed
模式,使用该模式需要手动开启:
这样在安装的时候会提示“在新标签页中打开”
大概长这样
- 优点:原生交互体验好
- 缺点:实验性api,用户使用成本大(目前我们平台还是小范围的推多tab形式,主要是内部人员使用,用于演示这些操作)
唤起方式
主要是针对pwa应用之间的相互唤起、开机启动这些
持续探索中,有好方案欢迎留言…
POST 缓存方案
由于cache storage
不支持缓存 POST
请求,所以在首页展示有POST
相关接口进行数据拉取时,离线效果不理想。这个问题更多的是探讨是否有必要这样做,理论上遵守restful api
规范的话就不会存在这个问题(具体的原因就不进行深究了…)。
抛开业务,我们只探讨技术方案;对此找了下相关文档,还真有实现方案,大致分以下几部:
- Service Worker 拦截一个 POST 请求,并根据request中的查询字符串组成一个 MD5 加密的缓存key。
- Service Worker 使用缓存key将新的 JSON 响应存储在 IndexedDB 中。
- 如果POST请求失败,Service Worker 使用缓存key检查 IndexedDB。如果密钥存在,则返回缓存的 JSON。
具体的可以参考原文Service worker无法缓存POST请求?来我教你
结语
这是一次很好的学习过程,上述所提到的插件是Webpack
的,Vite
也有相关插件vite-plugin-pwa
,配置大同小异。有什么问题欢迎评论区交流👏👏👏