如何在页面加载之前读取Client.postMessage?

时间:2018-03-01 20:17:54

标签: javascript service-worker

我有一个服务工作程序在缓存资源发生更改时在获取期间发出Client.postMessage。我使用它来通知用户他们可能想要刷新。

我的问题是,当活动页面资源发生更改且服务工作人员发出该消息时,该页面尚未加载,因此没有javascript可以接收该消息。

有没有更好的方法来处理这样的情况,而不是使用waitUntil在发出消息之前暂停几秒钟?

1 个答案:

答案 0 :(得分:4)

另一种选择是从服务工作者写入IndexedDB,然后在第一次加载页面之前读取它,然后再建立message监听器。

为简单起见,使用ibd-keyval库,这可能如下所示:

// In your service worker:
importScripts('https://unpkg.com/idb-keyval@2.3.0/idb-keyval.js');

async function notifyOfUpdates(urls) {
  const clients = await self.clients.matchAll();
  for (const client of clients) {
    client.postMessage({
      // Structure your message however you'd like:
      type: 'update',
      urls,
    });
  }

  // Read whatever's currently saved in IDB...
  const updatedURLsInIDB = await idb.get('updated-urls') || [];
  // ...append to the end of the list...
  updatedURLsInIDB.push(...urls);
  // ...and write the updated list to IDB.
  await idb.set('updated-urls', updatedURLsInIDB);
}


// In your web page:
<script src="https://unpkg.com/idb-keyval@2.3.0/idb-keyval.js"></script>
<script>
  async listenForUrlUpdates() {
    const updatedURLsInIDB = await idb.get('updated-urls');
    // Do something with updatedURLsInIDB...

    // Clear out the list now that we've read it:
    await idb.delete('updated-urls');

    // Listen for ongoing updates:
    navigator.serviceWorker.addEventListener('message', event => {
      if (event.data.type === 'update') {
        const updatedUrls = event.data.urls;
        // Do something with updatedUrls
      }
    });
  }
</script>