PWA で作った台本ビューア を Svelte で作り直すことにしたので、やったことをメモしていきます。
- 前回 : <第5回> データを読込む
 - 次回 : <第7回> 目次によるジャンプ
 - シリーズ記事一覧
 
<第6回> 更新の動作の改善
今回の概要
- アプリを更新できない時があったので、その問題について考察します。
 - ブラウザで実行している時は更新のプロンプトを出さないようにします。
 - 更新の動作について注意書きで補足します。
 
今回使うソースコードは GitHub のこちらのコミット にあります。
アプリを更新できない問題
<第3回>「更新プロンプトを出す」 で Service Worker の更新について実装しましたが、その後、更新プロンプトの「更新する」ボタンを押しても何も起こらないことがありました。
結論からいうと、この問題は vite-plugin-pwa をアップデートすると解決しました。

※↑この「更新する」ボタンを押しても何も起こらないことがあった。
原因は、ブラウザでもこの PWA のページを開いて更新していたからだと分かりました。
ブラウザとインストールした PWA の両方で同じ更新をしようとした時に、この問題が起こっていたのです。
あまり起きない状況かとは思いますが、ボタンが利かなくなるのは気持ち悪いです。
考察
おそらくですが、ブラウザと PWA の両方で更新を検知してプロンプトを出して、一方が更新を完了してしまうと、もう一方が「更新は完了しているので何もしない」という状態になっていたのだと思います。
この時とにかくリロードするという手もあります。
updateServiceWorker() は Promise を返すので、以下のように書き換えればリロードするようになります。
// 変更前
updateServiceWorker(true)
// 変更後
updateServiceWorker(true).then(() => { location.reload() })
しかし引数 (reloadPage) に true を指定しているのに自前でリロードをするのもおかしな話です。
そう思って vite-plugin-pwa の GitHub を見てみると、アップデートのロジックについての修正がありました。
vite-plugin-pwa をアップデート
これを入れてみたところ先ほどの問題は解消され、自前でリロードする必要はなくなりました。
{
  (中略)
  "devDependencies": {
    (中略)
    "vite-plugin-pwa": "^0.14.1",  <- このように更新して yarn install
    (中略)
  }
}
具体的には、ブラウザと PWA の両方で更新プロンプトを開いて一方で「更新する」を実行すると、もう一方でもリロードが起きるようになりました。
別のアプリによって更新された場合でも、更新が完了したことを検知できるようになったのだと思います。
ブラウザ実行の扱い
そもそもブラウザで実行している時に更新プロンプトを出すべきか、というのも気になります。
普通に Web アプリを見ていて「更新しますか?」と言われても、ユーザは何のことかと思うかも知れません。
なので、ブラウザで実行している時は更新のプロンプトを出さないことにしました。
プロンプトを出さなくともサーバ上の Service Worker の更新は監視されますし、新しくインストールされた Service Worker はブラウザを再起動すれば有効化されます。
Service Worker の自動更新について
ブラウザがサーバ上の Service Worker の更新を確認する時、最後の更新から24時間は更新を検知しません (または24時間以内のキャッシュを参照します)。自動更新の動作を確認するにはサーバ上の Service Woreker を更新して24時間待つ必要があります。
ブラウザ実行であることを判定する
「ブラウザで実行している時は更新のプロンプトを出さない」ということを実現するために、(インストールした PWA ではなく) ブラウザで実行していることを判定する必要があります。
その方法を PWA Night の Slack で質問したところ、以下のページを教えていただけました。
これをそのままコピーして、lib フォルダの中に env.ts を作って isPwa を定義しました。
// PWA として実行しているかの判定
export const isPwa = window.matchMedia('(display-mode: standalone)').matches
ブラウザ実行時はプロンプトを出さない
あとは、これを使って isPwa が false ならプロンプトを出さないようにするだけです。
App.svelte のコード全文は こちら にあります。
<script lang="ts">
  import { isPwa, isAndroid } from './lib/env'
  /* 中略 */
</script>
.....
{#if isPwa && reloadIsOpen}
  <ReloadPrompt
    on:close="{() => { reloadIsOpen = false }}"
  />
{/if}
上記の方法では、ブラウザで最初に実行した時に Service Worker が見つからないという問題があることが分かりました。ReloadPrompt コンポーネント自体をマウントしないのではなく、その中のプロンプトの表示制御でブラウザ実行かどうかを条件とすべきでした。
その改善をしたコードは こちら にあります。
注意書きでより親切にする
以上で更新の動作が改善できたと思いますが、さらに注意書きを入れることで、よりユーザに親切になると思いました。
注意書きは About コンポーネント (バージョン情報を表示するパネル) に以下の2つを表示することにしました。
- 更新が途中で止まった場合について
「更新が途中で止まった場合は、一度アプリを終了して起動しなおしてください。」 - ブラウザで実行している場合の更新について
「新しいバージョンは自動的に配信されます。常に最新のバージョンを使うには、このページを PWA としてインストールするか、キャッシュをクリアして再読込みしてください。」 
About.svelte のコード全文は こちら にあります。
<div class="panel" class:gone>
  .....
  <div class="container">
    <p>バージョン<br>{APP_VERSION}</p>
    {#if $appUpdateFunc}
      <button on:click|once="{() => $appUpdateFunc(true)}">
        今すぐ台本ビューアを更新する
      </button>
      <div style="text-align: left;">
        <p>
          更新が途中で止まった場合は、一度アプリを終了して起動しなおしてください。
        </p>
      </div>
    {/if}
    {#if !isPwa}
      <div style="text-align: left;">
        <p>
          新しいバージョンは自動的に配信されます。<br />
          常に最新のバージョンを使うには、このページを PWA としてインストールするか、キャッシュをクリアして再読込みしてください。
        </p>
      </div>
    {/if}
    <p>{COPY_RIGHT}</p>
  </div>
</div>
更新が途中で止まった場合については、更新ボタンが表示されている時だけ表示すれば良いので、更新ボタンと同じ if ブロックの中に入れました。

ブラウザで実行している場合の更新については、ブラウザで実行している時だけ表示されるように、新しく if ブロックを作ってその中に入れました。
