PWA的學(xué)習(xí)之路

PWA介紹

PWA是什么

MDN解釋

PWAProgressive Web Apps,漸進(jìn)式Web應(yīng)用)運(yùn)用現(xiàn)代的Web API以及傳統(tǒng)的漸進(jìn)式增強(qiáng)策略來創(chuàng)建跨平臺(tái) Web應(yīng)用程序。這些應(yīng)用無處不在、功能豐富,使其具有與原生應(yīng)用相同的用戶體驗(yàn)優(yōu)勢(shì)。

  • 可以提升WebApp體驗(yàn)的一種新方法,能給用戶原生應(yīng)用的用戶體驗(yàn)
  • 創(chuàng)建跨平臺(tái)Web桌面應(yīng)用程序

總結(jié):PWA能做到如此,不是指某一項(xiàng)特定的技術(shù),而是在原有的WebApp基礎(chǔ)之上通過添加Manifest配置文件和Service Worker以及搭配緩存使用,可以讓原有的應(yīng)用具備安裝離線的功能

PWA優(yōu)勢(shì)

  • 漸進(jìn)式
    • 在原有的基礎(chǔ)上添加PWA功能,不影響原有應(yīng)用
  • 可安裝/原生體驗(yàn)
    • 可以像原生App應(yīng)用一樣添加在桌面,點(diǎn)擊主屏幕圖標(biāo)可以實(shí)現(xiàn)啟動(dòng)動(dòng)畫以及隱藏地址欄
  • 離線功能
    • Service Worker和緩存 api實(shí)現(xiàn)離線緩存功能。可以做到離線訪問使用一些離線功能
  • 推送
    • 通過推送消息點(diǎn)擊,快速打開應(yīng)用,用戶促活

這里展示一下手機(jī)端Demo運(yùn)行效果

動(dòng)態(tài)圖.gif

PWA相關(guān)技術(shù)點(diǎn)

源碼請(qǐng)點(diǎn)擊這里下載

一個(gè)標(biāo)準(zhǔn)的PWA應(yīng)用應(yīng)該具備以下3個(gè)特征

  • https訪問或者localhost
  • Service Worker
  • Manifest

Manifest

manifest.json可以使應(yīng)用具備添加桌面圖標(biāo)的能力。配置文件內(nèi)容:應(yīng)用名稱,圖標(biāo),啟動(dòng)屏,主題色,應(yīng)用樣式等等

相比傳統(tǒng)的WebApp入口:網(wǎng)址,收藏夾,書簽;PWA應(yīng)用入口更直接,方便快捷

  • 添加桌面圖標(biāo)
  • 添加啟動(dòng)動(dòng)畫,避免白屏
  • 隱藏瀏覽器的默認(rèn)相關(guān)UI,看上去更像一個(gè)原生應(yīng)用

index.html引入manifest.json文件

<link rel="manifest“  href="/manifest.json" />

manifest.json常用的配置信息:

{
    "name": "pwa-basic",
    "short_name": "PWA基礎(chǔ)",
    "start_url": "/index.html",
    "icons":[
        {
            "src": "/images/icon128.png",
            "sizes": "128x128",
            "type":"image/png"
        },
        {
            "src": "/images/icon144.png",
            "sizes": "144x144",
            "type":"image/png"
        }
    ],
    "description": "這個(gè)是一個(gè)PWA基礎(chǔ)的Demo",
    "background_color": "skyblue",
    "theme_color": "black",
    "display": "standalone"
}
  • name 應(yīng)用名稱,添加時(shí)顯示的名稱,啟動(dòng)屏下方顯示的名稱
  • short_name 當(dāng)應(yīng)用名稱過長(zhǎng),桌面上顯示的名稱
  • start_url 點(diǎn)擊桌面圖標(biāo)加載的入口地址
  • icons 桌面的圖標(biāo),啟動(dòng)屏中間顯示圖標(biāo)
  • description
  • background_color 啟動(dòng)屏的背景顏色
  • theme_color 主題顏色
  • display
    • fullscreen全屏顯示, 所有可用的顯示區(qū)域都被使用, 并且不顯示狀態(tài)欄
    • standalone 狀態(tài)欄除外,無地址欄
    • minimal-ui 有狀態(tài)欄,有瀏覽器地址欄

更多配置項(xiàng)

Cache Api

MDN-Cache文檔

Service Worker

什么是Service Worker

MDN解釋

Service workers 本質(zhì)上充當(dāng) Web應(yīng)用程序、瀏覽器與網(wǎng)絡(luò)(可用時(shí))之間的代理服務(wù)器。這個(gè) API旨在創(chuàng)建有效的離線體驗(yàn),它會(huì)攔截作用域下邊的網(wǎng)絡(luò)請(qǐng)求并根據(jù)網(wǎng)絡(luò)是否可用來采取適當(dāng)?shù)膭?dòng)作、更新來自服務(wù)器的的資源

Service Worker的功能(除了具備Web Worker功能以外)

  • 安裝成功,運(yùn)行于瀏覽器后臺(tái)。可以攔截作用域范圍下所有的頁面http請(qǐng)求
  • 利用緩存API可以讓離線內(nèi)容可控(由開發(fā)者控制)
  • 必須使用https。除了使用本地開發(fā)環(huán)境調(diào)試時(shí)域名可以使用localhost

Service Worker兼容性 94%的瀏覽器都支持

image.png

Service Worker 的使用

以下簡(jiǎn)稱SW

SW獨(dú)立于 Web 頁面的生命周期。分為5步:注冊(cè)、安裝成功(安裝失敗)、激活、運(yùn)行、銷毀。如下圖:

引用網(wǎng)絡(luò)圖片

SW注冊(cè)

if (window.navigator.serviceWorker) {
  const serviceWorker = window.navigator.serviceWorker;
  serviceWorker.register('./service-worker.js').then(registion => {
    console.log('注冊(cè)成功', registion);
  }).catch(error => {
    console.log("注冊(cè)失敗")
  })
}

SW常見事件監(jiān)聽如下

// service worker 安裝
self.addEventListener('install', (event) => {
    console.log('install 成功', event);
    // 這里一般靜態(tài)資源的預(yù)緩存
});

// service worker 激活
self.addEventListener('activate', (event) => {
    console.log('activate 成功', event);
    // 這里建議刪除舊的緩存,所有客戶端獲取控制權(quán)
})

self.addEventListener('fetch', (event) => {
    const req = event.request;
    // 攔截請(qǐng)求,并利用fetch api 和 cache api把結(jié)果給到event.respondWith() 即可實(shí)現(xiàn)離線緩存
});

1、SW第一次加載

用戶首次訪問PWA的頁面時(shí),SW會(huì)立刻被下載并注冊(cè)。注冊(cè)結(jié)束之后,會(huì)進(jìn)行安裝SW,執(zhí)行install事件(會(huì)在這里做預(yù)緩存,也就是靜態(tài)文件的緩存),第二次打開就可以使用緩存的靜態(tài)資源了(這里也要看使用的緩存策略,下邊會(huì)講)

這里的靜態(tài)資源(HTML, JS, CSSImages)可以理解為web應(yīng)用的外殼(應(yīng)用外殼緩存),也可以理解成首頁。因?yàn)轭A(yù)緩存一個(gè)網(wǎng)站離線工作需要的所有資源顯然是不現(xiàn)實(shí)的。

const staticFiles = [
    '/',
    '/index.html',
    '/css/index.css',
    '/js/index.js'
];
caches.open("cache_name_v1").then(cache => {
    return cache.addAll(staticFiles);
}).then(self.skipWaiting)

然后會(huì)立即激活SW,也就會(huì)觸發(fā)監(jiān)聽激活的事件,該事件會(huì)做移除舊緩存和獲取控制權(quán)的操作

caches.keys().then(keys => {
    keys.forEach(key => {
        if ("當(dāng)前緩存key值" !== key) {
            caches.delete(key);
        }
    });
})

激活之后就是SW正常運(yùn)行階段了。在運(yùn)行階段,監(jiān)聽了fetch事件,所有的請(qǐng)求都會(huì)走到這個(gè)事件。主要做兩件事

  • 使用cache.put(request,response)緩存一些運(yùn)行時(shí)資源,比如api,其它文件資源
  • 使用用event.respondWith返回?cái)r截的請(qǐng)求響應(yīng)結(jié)果,那么這個(gè)如何返回這個(gè)響應(yīng)結(jié)果就是我們開發(fā)可控的

響應(yīng)結(jié)果從原子上分為2中方式來獲取

  • 網(wǎng)絡(luò)請(qǐng)求fetch(request)
const req = event.request;
event.respondWith(
    fetch(req).then((response) => {
        return response;
    })
);
  • 直接取緩存
const req = event.request;
caches.open("緩存key").then(cache => {
    return cache.match(req);
})

以上2種其實(shí)是常用緩存策略中的2個(gè)而已

  • NetworkOnly
  • CacheOnly
    這兩種基本是不可取的,太過單一。還有一下幾種
  • CacheFirst 首先去緩存,取緩存失敗,再去請(qǐng)求
  • NetworkFirst 首先請(qǐng)求緩存,請(qǐng)求失敗,再取緩存
  • StaleWhileRevalidate 從緩存中讀取資源的同時(shí)發(fā)送網(wǎng)絡(luò)請(qǐng)求更新本地緩存

為防止緩引起問題,一般使用NetworkFirst。優(yōu)先取最新得網(wǎng)路資源,失敗再去取緩存。這樣既可以做到離線緩存,也可以資源/數(shù)據(jù)得一致

2、SW更新
當(dāng)沒有更新的時(shí)候再次注冊(cè)SW,不會(huì)觸發(fā)installactivate事件。

如果SW有更新,和上述第一次加載路程基本相似。有2個(gè)注意事項(xiàng)

  • install事件需要處理skipWaiting,可以讓更新的sw跳過等待,直接激活。防止有頁面正在使用老的SW,導(dǎo)致新的SW一直等待激活。
  • activate事件需要獲取控制權(quán)clients.claim()

3、SW銷毀

sw一旦安裝激活成功,會(huì)一直存在,需要手動(dòng)卸載

if ('serviceWorker' in navigator) {
  navigator.serviceWorker.ready.then(registration => {
    registration.unregister();
  })
}

PWA實(shí)操

注冊(cè)SW文件

可以使用register-service-worker插件注冊(cè)sw.js

點(diǎn)擊這個(gè)查看具體使用

基本使用搞懂了,可以在項(xiàng)目中實(shí)踐一下。

對(duì)于大部分項(xiàng)目的現(xiàn)狀,使用webpack/vite構(gòu)建工具編譯項(xiàng)目,每次發(fā)版編譯的靜態(tài)資源文件名存在變化,直接寫原生的SW,不可能在SW文件中每次都要手動(dòng)去修改需要緩存的靜態(tài)資源。而且編寫也比較繁瑣和復(fù)雜。所以這里需要借助輪子

PWA相關(guān)工具的使用

sw-precache 默認(rèn)集成了 sw-toolbox,但是兩個(gè)工具官方已經(jīng)不更新了。google官方建議使用workboxworkbox是現(xiàn)在PWA最優(yōu)的絕決方案

workbox基本使用

workbox的緩存分為兩種,一種的precache,一種的runtimecache

workbox常用的包

workbox-build  自動(dòng)構(gòu)建生成sw.js
workbox-core  核心包,基本的類
workbox-expiration  配置資源過期
workbox-precaching  處理預(yù)緩存
workbox-cacheable-response 處理緩存資源相應(yīng)
workbox-strategies  緩存策略對(duì)象

precache對(duì)應(yīng)的是在installing階段進(jìn)行讀取緩存的操作。它讓開發(fā)人員可以確定緩存文件的時(shí)間和長(zhǎng)度,以及在不進(jìn)入網(wǎng)絡(luò)的情況下將其提供給瀏覽器,這意味著它可以用于創(chuàng)建Web離線工作的應(yīng)用。
precache api具體使用

precache([
    '/',
    '/index.html',
    '/css/index.css',
    '/js/index.js',
])

或者

precacheAndRoute(self.__WB_MANIFEST);

第二種方式需要依賴 workbox-build插件InjectManifest模式下,根據(jù)配置動(dòng)態(tài)替換__WB_MANIFEST占位符的形式

runtimecache api具體使用如下:(運(yùn)行時(shí)的資源,比如api,其它資源等等)

registerRoute(
    ({ request }) => request.url === 'http://localhost:3030/api/food/getFoodList',
    new NetworkFirst({
        // cacheName: 'api',
        plugins: [
            new CacheableResponsePlugin({
                statuses: [200, 0],
            }),
            new ExpirationPlugin({
                maxEntries: 50,
                maxAgeSeconds: 60 * 60 * 24 * 30, // 30 Days
            }),
        ],
    }),
);

work-build的使用

2種模式

  • injectManifest
    需要按照workbox規(guī)范手動(dòng)維護(hù)sw.js文件,動(dòng)態(tài)東城預(yù)緩存的資源列表。根據(jù)sw.js文件中__WB_MANIFEST占位符進(jìn)行替換。這里的動(dòng)態(tài)緩存需要自己維護(hù)開發(fā),可拓展性及強(qiáng)
const { injectManifest } = require('workbox-build')
injectManifest({
    swSrc: path.resolve(__dirname, './sw.js'),  // 必傳
    swDest: path.resolve(__dirname, './service-worker.js'),
    globDirectory: path.resolve(__dirname, '../'),
    globPatterns: ["**\/*.{js,css,html,ttf}"]
});
  • generateSW
    不需要提供sw.js文件,只需傳入配置,就可自動(dòng)生成sw.js文件
const { generateSW } = require('workbox-build')
generateSW({
    cacheId: 'generatesw-pwa', // 設(shè)置前綴
    cleanupOutdatedCaches: true, // 刪除舊的預(yù)緩存
    skipWaiting: true,
    clientsClaim: true,
    globDirectory: path.resolve(__dirname, '../'),
    globPatterns: ["**\/*.{js,css,html,ttf}"],
    sourcemap: false,
    mode: 'development', // 生成sw文件的模式
    swDest: path.resolve(__dirname , 'service-worker.js'), // 默認(rèn) service-worker.js
    runtimeCaching: [
        {
            urlPattern: /^http:\/\/juheimg.oss-cn-hangzhou.aliyuncs.com/,
            handler: 'NetworkFirst', // 網(wǎng)絡(luò)優(yōu)先
            options: {
                cacheName: 'images-cache',
                expiration: {
                    maxEntries: 20, // 針對(duì)改類型的緩存的最大數(shù)量,超過替換
                },
            },
        }
    ]
});
workbox-webpack-plugin的使用

這是分為兩種模式,和上述work-build差不多,需要增加webpack插件配置,需要在html-webpack-plugin之后,否則生成的html文件不在預(yù)緩存列表

const { GenerateSW , InjectManifest } = require('workbox-webpack-plugin')
new GenerateSW(config) 或者 new InjectManifest(config)

webpack項(xiàng)目中使用

vue項(xiàng)目

vue新創(chuàng)建的項(xiàng)目可以自己選擇支持pwa如下圖

企業(yè)微信20220506-221636.png

如果不是,則添加@vue/cli插件即可。執(zhí)行vue add pwa即可添加支持pwa

具體配置請(qǐng)看配置文檔

react項(xiàng)目

react項(xiàng)目新建
使用cra腳手架工具指定pwa模板創(chuàng)建,如下

create-react-app my-app --template cra-template-pwa

老項(xiàng)目直接根據(jù)項(xiàng)目情況添加webpack配置以及manifest.json,注冊(cè)sw.js即可

vite項(xiàng)目中使用

安裝

npm install -D vite-plugin-pwa

vite.config.js配置

import { VitePWA } from 'vite-plugin-pwa'

export default {
  plugins: [
    VitePWA({
            mode: 'development',
            workbox: {
                // 省略
            }
        })
  ]
}

更多vite-plugin-pwa插件使用詳情

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,401評(píng)論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,011評(píng)論 3 413
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,263評(píng)論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,543評(píng)論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,323評(píng)論 6 404
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 54,874評(píng)論 1 321
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 42,968評(píng)論 3 439
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,095評(píng)論 0 286
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,605評(píng)論 1 331
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,551評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,720評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,242評(píng)論 5 355
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 43,961評(píng)論 3 345
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,358評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,612評(píng)論 1 280
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,330評(píng)論 3 390
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,690評(píng)論 2 370

推薦閱讀更多精彩內(nèi)容