vue PWA serviceWorker 有新内容时,如何自动刷新内容

vue PWA serviceWorker 有新内容时,如何自动刷新内容

一、问题描述

vue 自带的 pwa 插件可以很方便管理 serviceWorker 的使用,但会有一个问题。

ServiceWorker 的运行机制是这样的:

  1. 后台检测到新版本
  2. 新版 ServiceWorker 下载并安装
  3. 安装完成之后等待生效,只有当用户关系所有相关窗口并重新打开,新的 ServiceWorker 才会接管

而我需要在检测到新版本的内容之后自动刷新页面载入新内容。

就需要在检测到新版本安装成功之后,刷新所有实例窗口。这个操作包含两个操作:

  1. ServiceWorker 方面在安装成功后,向外 postMessage 信息,告诉其它窗口已经准备好更新了,skipWaiting
  2. 其它窗口在打开的时候就监听 postMessage 信息,当监听到指定的信息的时候,刷新自己。

具体 ServiceWorker 的响应机制可以看这篇文章,很的很详细

Handling Service Worker updates -- how to keep the app updated and stay sane

二、探索解决办法

使用 register-service-worker ,项目中 registerServiceWorker.js 的内容是这样的

js 复制代码
/* eslint-disable no-console */

import { register } from 'register-service-worker'
  register(`${process.env.BASE_URL}service-worker.js`, {
    ready () {
      console.log(
        'App is being served from cache by a service worker.\n' +
        'For more details, visit https://goo.gl/AFskqB'
      )
    },
    registered () {
      console.log('Service worker has been registered.')
    },
    cached () {
      console.log('Content has been cached for offline use.')
    },
    updatefound () {
      console.log('New content is downloading.')
    },
    updated (registration) {
      console.log('New content is available; please refresh.')
    },
    offline () {
      console.log('No internet connection found. App is running in offline mode.')
    },
    error(error) {
      console.error('Error during service worker registration:', error)
    }
  })

需要做的就是在 updated 方法中发出一个内容为 SKIP_WAITING 的事件。

我第一次发的时候只发了一个字符串,结果不行。

后认真查看了最终生成的 dist/service-worker.js 的代码,并在其中加入了代码查看它获取到的 data 是什么内容


js 复制代码
define(["./workbox-d6430d1c"], (function (e) {
    "use strict";
    e.setCacheNameDetails({prefix: "iphone"}), self.addEventListener("message", (e => {
        e.data && "SKIP_WAITING" === e.data.type && self.skipWaiting()
    })), e.precacheAndRoute([{

结果出来是这个结果,所以就知道 postMessage 的内容格式肯定是错了。改成下面这样就好了

js 复制代码
 updated (registration) {
   if (registration.waiting) {
     registration.waiting.postMessage({
       type: 'SKIP_WAITING'
     })
   }
   console.log('新的 ServiceWorker.js 已经下载并安装')
 },
相关推荐
安卓开发者11 分钟前
鸿蒙NEXT Web组件与JavaScript交互:打通原生与前端的桥梁
前端·javascript·harmonyos
fdc201733 分钟前
Avalonia 基础导航实现:从页面切换到响应式交互全指南
开发语言·javascript·ecmascript
小菜花291 小时前
利用H5实现svg图片中各部分监听事件
前端·javascript·svg·object标签
前端小巷子2 小时前
JS 打造「放大镜 + 缩略图」一体组件
前端·javascript·面试
知识分享小能手2 小时前
React学习教程,从入门到精通,React AJAX 语法知识点与案例详解(18)
前端·javascript·vue.js·学习·react.js·ajax·vue3
朗迹 - 张伟3 小时前
Gin-Vue-Admin学习笔记
vue.js·学习·gin
古夕3 小时前
前端文件下载的三种方式:a标签、Blob、ArrayBuffer
前端·javascript·vue.js
李李记3 小时前
Node.js 打包踩坑?NCC+PKG 从单文件到多平台可执行文件,解决 axios 缺失等 80% 问题
javascript
武昌库里写JAVA3 小时前
Java设计模式中的几种常用设计模式
vue.js·spring boot·sql·layui·课程设计