<?xml version="1.0" encoding="UTF-8"?><rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	xmlns:series="http://organizeseries.com/"
	>

<channel>
	<title>Service Worker &#8211; HTML5Experts.jp</title>
	<atom:link href="/tag/service-worker/feed/" rel="self" type="application/rss+xml" />
	<link>https://html5experts.jp</link>
	<description>日本に、もっとエキスパートを。</description>
	<lastBuildDate>Sat, 07 Jul 2018 03:14:05 +0000</lastBuildDate>
	<language>ja</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
	<generator>https://wordpress.org/?v=4.7.19</generator>
	<item>
		<title>進化するWeb ～Progressive Web Appsの実装と応用～(de:code2018より)</title>
		<link>/osamum_ms/25709/</link>
		<comments>/osamum_ms/25709/#respond</comments>
		<pubDate>Thu, 05 Jul 2018 02:00:10 +0000</pubDate>
		<dc:creator><![CDATA[物江 修]]></dc:creator>
				<category><![CDATA[最新動向]]></category>
		<category><![CDATA[Microsoft]]></category>
		<category><![CDATA[PWA]]></category>
		<category><![CDATA[Progressive Web Apps]]></category>
		<category><![CDATA[Service Worker]]></category>
		<category><![CDATA[de:code]]></category>

		<guid isPermaLink="false">/?p=25709</guid>
		<description><![CDATA[2018年5月に開催された日本マイクロソフト主催のイベントde:code 2018で「進化するWeb ～Progressive Web Appsの実装と応用～」というセッションを担当しました。 イベントに参加できなかった...]]></description>
				<content:encoded><![CDATA[<p>2018年5月に開催された日本マイクロソフト主催のイベント<a href="https://www.microsoft.com/ja-jp/events/decode/2018/" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">de:code 2018</a>で「進化するWeb ～Progressive Web Appsの実装と応用～」というセッションを担当しました。</p>

<p>イベントに参加できなかった方に向けてセッションの内容を記事にまとめましたので、ぜひご覧ください。</p>

<div style="margin-bottom:5px"> <strong> <a href="https://html5experts.jp//www.slideshare.net/osamum/web-progressive-web-apps" title="進化する Web ～ Progressive Web Apps の実装と応用 ～" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">進化する Web ～ Progressive Web Apps の実装と応用 ～</a> </strong> from <strong><a href="https://www.slideshare.net/osamum" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">Osamu Monoe</a></strong> </div>

<p>この記事では、昨今話題に上ることが多いPWAことProgressive Webアプリケーションについて、実際の作り方を解説しながら、それがいったいどういったものであるかを詳(つまび)らかに紹介することを目的としています。</p>

<p>Webでは、「ネイティブアプリと同じことができる」「ネイティブアプリを置き換える」など、期待に胸を膨らませずにいられない浪漫に満ちた噂がありますが、それが本当かどうか記事をご覧いただくとご理解いただけると思います。</p>

<p>Progressive Web Appsの作り方を紹介する前に、簡単に概要を紹介しましょう。</p>

<h2>Progressive Web Appsとは?</h2>

<p>Progressive Web Apps(以下PWAと記述) は、一言でいうならば「ネイティブアプリのようなUXを提供するWebアプリの概念」といったところでしょう。2015年の11月に開催された<a title="Chrome Dev Summit 2015" href="https://codelabs.developers.google.com/chrome-dev-summit" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">Chrome Dev Summit 2015</a>のキーノートで発表されて話題となりました。</p>

<p>もともとは、その年の10月にAlex Russell氏(Google)が<a title="Progressive Web Apps: Escaping Tabs Without Losing Our Soul" href="https://infrequently.org/2015/06/progressive-apps-escaping-tabs-without-losing-our-soul/" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">ブログ記事</a>に投稿した、クライアントの性能に合わせて段階的に進歩するWebアプリケーションのコンセプトでした。</p>

<p>PWAというと、&#8221;ネイティブアプリのような体験&#8221;といった特徴ばかりが注目されますが、大きな特徴がもう一つあります。それは、PWAという名前にも含まれているProgressiveというところです。</p>

<p>これはHTML5が普及しはじめた頃にあった、「HTML5が解釈できるモダンブラウザーにはリッチな体験を、そうでないブラウザーには従来とおなじ体験を」というデザインの考え方&nbsp;<a href="https://www.blwisdom.com/dictionary/item/6559-001474.html" title="プログレッシブ・エンハンスメント" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Progressive Enhancement</a>に由来します。</p>

<p>PWAはこの思想を踏襲し、性能の低いWebブラウザーを切り捨てることなく、クライアントの性能に合わせた機能提供を行います。</p>

<p>PWAは、性能の低いWebブラウザーからアクセスがあった場合は従来と同じWebページとして動作し、PWAが動作する機能を備えたWebブラウザーからアクセスがあった場合は、その機能を活かし、これまでのWebアプリにはなかったネイティブアプリのような体験を提供します。</p>

<p><img src="https://msdnshared.blob.core.windows.net/media/2018/06/pwa.png" alt="Web アプリケーションの Progressive Enhancement" width="1634" height="900" class="alignnone size-full wp-image-5485" /></p>

<h2>Progressive Web Appsを実現する機能</h2>

<p>それでは、&#8221;これまでのWebアプリにはなかったネイティブアプリのような体験&#8221;というのはどのようなものがあるでしょう? </p>

<p>代表的なものとしては以下の4つが挙げられます。</p>

<ol style="font-size:15px">
<li>オフラインサポート</li>
<li>プッシュ通知</li>
<li>バックグラウンド処理</li>
<li>デバイスへのインストール(デバイスのホーム画面から起動できる)</li>
</ol>

<p>これらはの機能はそれぞれ以下のような新しいAPIによって実現されます。</p>

<ol style="font-size:15px">
<li>オフラインサポート&nbsp;&#8211;&nbsp;<span style="color:blue;font-weight:bold">Cache API</span></li>
<li>プッシュ通知&nbsp;&#8211;&nbsp;<span style="color:blue;font-weight:bold">Push API</span></li>
<li>バックグラウンド処理&nbsp;&#8211;&nbsp;<span style="color:blue;font-weight:bold">Background Sync</span></li>
<li>デバイスへのインストール(OSのメニューから起動できる)&nbsp;&#8211;&nbsp;<span style="color:blue;font-weight:bold">Web App Manifest</span></li>
</ol>

<p>そして、これらの機能を提供しているのが1～3が<a href="https://developer.mozilla.org/ja/docs/Web/API/ServiceWorker_API" target="_blank" title="MDN - サービスワーカー API" data-wpel-link="external" rel="follow external noopener noreferrer">Service Worker</a>で、4が<a href="https://developer.mozilla.org/ja/docs/Web/Manifest" title="MDN - Web App Manifest" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">Web App Manifest</a>です。</p>

<p><img src="https://msdnshared.blob.core.windows.net/media/2018/06/sw_new2.png" alt="新しいAPIの提供の図" width="1083" height="456" class="aligncenter size-full wp-image-5505" /></p>

<p>PWAの定義に明確なものはありませんが、上記の機能の中からオフラインサポートと、インストール（アイコンをデバイスのホーム画面に追加）機能をサポートしていれば、PWAを名乗ってもそう否定されることはないでしょう。</p>

<p>これらの機能により、デバイスのホーム画面から、ナビゲーションバーなどのブラウザーUIがない状態でアプリケーションを起動し、オフラインの状態でも使用できる、という体験をユーザーに提供できます。これらの体験は、これまでのWebアプリケーションにはなかったネイティブアプリのならではのメリットと言えるでしょう。</p>

<p>それでは、逆に Web ならではのメリットとはなんでしょう?</p>

<h2>Webのメリット</h2>

<p>ネイティブアプリにはないWebならではのメリットを、Googleさんのイベントではよく以下の<a href="https://paul.kinlan.me/slice-the-web/" title="SLICE: The Web - Tales of a Developer Advocate by Paul Kinlan" target="_blank" style="font-weight:bold" data-wpel-link="external" rel="follow external noopener noreferrer">SLICE</a>という言葉で表しています。</p>

<ul style="font-size:15px">
<li><span style="font-weight:bold">S</span>ecure (安全)</li>
<li><span style="font-weight:bold">L</span>inkable (リンク可能)</li>
<li><span style="font-weight:bold">I</span>ndexable (インデックス可能)</li>
<li><span style="font-weight:bold">C</span>omposable (再構成可能)</li>
<li><span style="font-weight:bold">E</span>phemeral (一時的な利用)</li>
</ul>

<p>これら従来のWebとネイティブアプリの体験上のメリットを合わせたものが、PWAのメリットとなります。</p>

<h2>Progressive Web Appsが提供する価値</h2>

<p>PWAの提供する価値について、Googleさんのイベントではよく<a href="https://youtu.be/NITk4kXMQDw?t=5m48s" title="PWAs: building bridges to mobile, desktop, and native (Google I/O &#039;18)" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">FIRE</a>という言葉で紹介されていますが、 </p>

<ul style="font-size:15px">
<li><span style="font-weight:bold">F</span>ast: パフォーマンスの良い、軽快な動作</li>
<li><span style="font-weight:bold">I</span>ntegrated: OSと統合されたユーザー体験</li>
<li><span style="font-weight:bold">R</span>eliable: オフラインでも動作する利便性と信頼性</li>
<li><span style="font-weight:bold">E</span>ngaging: Webサイトの価値向上</li>
</ul>

<p>この記事ではもう少しかみ砕いて紹介しましょう。</p>

<p><img src="https://msdnshared.blob.core.windows.net/media/2018/06/PWA_Merit.png" alt="" width="1100" height="455" class="aligncenter size-full wp-image-5515" /></p>

<h3 style="font-weight:bold">発見性</h3>

<p>PWAであれば、アプリを公開するのにアプリストアにわざわざお金を払ってアプリを提出したり、審査を通すための不自由なルールに縛られる必要はありません。これまでのWebアプリと同じようにインターネットに公開しておけば、検索エンジンがクロールして見つけてくれます。</p>

<p>いままでのSEOのスキルがそのまま使える上、Webはアプリストアとは比べものにならない数のユーザーが、比べものにならないくらいの回数、常に検索を行っています。この圧倒的なオポチュニティ（機会）の違いは、自分自身が今日何回Webを検索したか、アプリストアを何回検索したか、比べてみればよくわかるでしょう。</p>

<h3 style="font-weight:bold">インストール可能</h3>

<p>PWAはさまざまなデバイスにインストール可能ですが、そのためにプラットフォームごとに違う開発言語で書き直したりパッケージンクしなおしたりする必要はありません。PWAをサポートしているWebブラウザーであれば、同じアプリケーションをその全部にインストールして使うことができます。</p>

<p>また、&#8221;インストールしなくても使える&#8221;点もメリットです。もともとがWebページなので、ちょっとだけ使ってみるということが可能です。</p>

<p>サービスを試用してみようと思ったときに、&#8221;アプリのインストールが障壁になって利用を中断してしまった&#8221;、という経験は誰にでもあることでしょう。また、スマートフォンでコンテンツを閲覧中に、突然、アプリのストア画面が表示され、アプリのインストールを促されて不快な思いをしたこともあるかと思いますが、そういったことを避けられます。</p>

<p>PWAであれば提供者もユーザーもアプリのインストールについてのネガティブな点を回避することができます。</p>

<h3 style="font-weight:bold">再エンゲージ可能</h3>

<p>従来のWebページであれば、ユーザーがコンテンツから離脱したあとは、再び訪問してくれることを祈るくらいしかできませんでしたが PWAはサーバー側から通知をPushすることができます。</p>

<p>これによりユーザーに即時的な価値を提供できます。例えば、ECショップであればタイムセールの開始であるとか、オークションサイトであればオークションの開始や、最高落札額の更新などです。</p>

<p>通知機能を適切に利用することで、ユーザー側の機会獲得を増やし、再エンゲージを促すことができます。</p>

<p>逆にどうでもいいことを通知しすぎるとユーザーの心象を害するので注意が必要です。特に、初めてページを訪問したユーザーに「プッシュ通知を許可しますか?」というメッセージを表示するのは避けたいところです。</p>

<h3 style="font-weight:bold">ネットワーク非依存</h3>

<p>PWAはオフラインでの使用が可能ですので、ネットワークの状態に左右されないように作ることが可能です。また、常に使用されるアセットをローカルにキャッシュすることで表示のスピードアップや、回線の使用料を減らすことにも貢献します。</p>

<h3 style="font-weight:bold">プログレッシブとレスポンシブ</h3>

<p>PWAはきちんとその思想を理解して開発すれば、低機能なブラウザーやPWAをサポートしないデバイスにもサービスを提供できます。また、さまざまな画面サイズに対応するためのレスポンシブな機能については、WebにはMedia Queries等、そのためのナレッジもリソースも豊富に存在するため、既存のスキルを活かして機能を実装できます。</p>

<h3 style="font-weight:bold">安全</h3>

<p>PWAはWebコンテンツと同じWebブラウザーの強力なサンドボックス内で動作するため、ネイティブアプリのようにユーザーの強い権限で動作しないため、誤動作や悪意のあるコードによってシステムに深刻なダメージを与えることはありません。またhttpsによる接続でのみ動作するのでサーバーとのやりとりを安全に行うことができます。</p>

<h3 style="font-weight:bold">リンク可能</h3>

<p>PWAはインターネット上でユニークな URL をもっており、ハイパーリンクのあるさまざまなところからサービスに接続することができます。PWAを使用するのにアプリ ストアは必要なく、面倒なインストールプロセスも必要ありません。</p>

<p></p>

<p>PWAのメリットは、Webとネイティブ アプリのメリットを合わせたということだけでなく、それらを状況に合わせて取捨選択できることです。</p>

<h2>Progressive Web Appsを実現するAPI</h2>

<p>ここからはPWAを実現するためのAPIと、それらをどのように使ってアプリケーションを構築していくかについて紹介します。</p>

<h3>Service Worker</h3>

<p>PWAを実現する上で中心となる機能を提供するのが、Service Workerです。</p>

<p>Service Workerとは何か？ 端的に言うと、「バックグラウンドで動作する<span style="font-weight:bold">&#8220;プログラミング可能な&#8221;ネットワークプロキシ</span>」です。</p>

<p>Service WorkerはWeb Workerの一つで、Webページのスクリプトとは独立して動作しており、DOMにアクセスしたりすることはできませんが、Webページのネットワークリクエストすべてをインターセプトできます。</p>

<p>つまり、Webページからのリクエストを横取りしてキャッシュしたものを返したり、改ざんしたりといったことができます。そのため localhost、127.0.0.1接続以外はhttpsの使用が必須となります。</p>

<h3>Service Workerが提供する機能</h3>

<p>Service Workerが提供する機能は主に以下の4つです。</p>

<ul style="font-size:15px">
<li>キャッシュ</li>
<li>リクエストのハンドリング</li>
<li>Push通知</li>
<li>バックグラウンド同期</li>
</ul>

<p>これらの機能を図で紹介します。</p>

<h3 style="font-weight:bold">キャッシュとリクエストのハンドリング</h3>

<p>Service WorkerはWebページからのリクエストの中に、自分のキャッシュリストに含まれるアセットを見つけると、レスポンスからこれを取得してキャッシュします。</p>

<p><img src="https://msdnshared.blob.core.windows.net/media/2018/06/SW_Chache01.png" alt="" width="1207" height="546" class="aligncenter size-full wp-image-5525" /></p>

<p>それ以降、Webページからリクエストされたアセットがキャッシュ内に存在する場合は、そのキャッシュされたアセットを返します。ネットワークから取得しないので、アセットの取得は高速に完了し、通信コストは発生しません。</p>

<p><img src="https://msdnshared.blob.core.windows.net/media/2018/06/SW_Chache02.png" alt="" width="1206" height="545" class="aligncenter size-full wp-image-5535" /></p>

<h3 style="font-weight:bold">Push通知</h3>

<p>関連付けられたWebページがアクティブでなくても、サーバーからのPushを受け取り、Notification API等を使用してユーザーに通知を行えます。</p>

<p><img src="https://msdnshared.blob.core.windows.net/media/2018/06/sw_push1.png" alt="" width="1206" height="544" class="aligncenter size-full wp-image-5555" /></p>

<h3 style="font-weight:bold">バックグラウンド同期</h3>

<p>オフライン中にユーザーが行った操作をキャッシュし、</p>

<p><img src="https://msdnshared.blob.core.windows.net/media/2018/06/SW_Offline.png" alt="" width="1207" height="546" class="aligncenter size-full wp-image-5565" /></p>

<p>オンライン時にその内容を同期することができます。</p>

<p><img src="https://msdnshared.blob.core.windows.net/media/2018/06/sw_backgroundsync.png" alt="" width="1205" height="544" class="aligncenter size-full wp-image-5575" /></p>

<p>このように Service Worker は<span style="font-weight:bold">プログラミング可能なネットワークプロキシ</span>として、これまでにないさまざまな機能を提供します。</p>

<p>これらの機能の2018年6月時点の<a href="https://caniuse.com/#search=service%20worker" target="_blank" title="Can I use... Support tables for HTML5, CSS3, etc" data-wpel-link="external" rel="follow external noopener noreferrer">サポート状況</a>は以下のとおりです。</p>

<p><img src="https://msdnshared.blob.core.windows.net/media/2018/06/SW_Browser_support.png" alt="" width="1218" height="496" class="aligncenter size-full wp-image-5585" /></p>

<p>ここからは、Service Workerの具体的使い方について解説していきます。</p>

<h2>Service Workerを使う準備 </h2>

<p>Service Workerを使った開発を行う際に用意しなければならないものがあります。</p>

<p>まず、Service Workerを利用するためのWebアプリケーション、もしくはWebコンテンツが必要です。これはけしてSPA(Single Page Application)のようなアプリケーション然としたものである必要はなく、既存の一般的な Web ページであってもかまいません。</p>

<p>このWebページ、もしくはWebアプリケーションは、htmlやcss、jsや画像に代表されるメディアといった複数のファイルで構成されていると思いますが、それとはべつにService Worker用のJavaScriptファイルをひとつ用意します。ここでは便宜上sw.jsと名付けます。</p>

<p><img src="https://msdnshared.blob.core.windows.net/media/2018/06/webAsser.png" alt="" width="800" class="aligncenter size-full wp-image-5595" /></p>

<p>このService Worker用のJavaScriptファイルはWebコンテンツ側のJavaScriptコードから登録され稼働を開始します。</p>

<p>Service Workerは、自身のファイルが配置された以下のディレクトリに対しスコープを持つので、コンテンツ/アプリケーション全体をキャッシュするなど管理下におきたい場合はWebサイトのルートに配置します。</p>

<h2>Service Workerの登録</h2>

<p>Service Workerの登録はWebコンテンツのJavaScript から行います。</p>

<p>登録に必要なコードは非常にシンプルで、最低限、以下のコードで問題を発生させることなく、Service Workerの登録が行えます。</p>

<div style="width:100%">
<div style="width:600px;background-color:gainsboro;margin:0 auto;padding:10px;font-size:15px">
<div style="text-align:center;width:140px;height:20px;background-color:black;float:right;color:white;margin-top:-10px;margin-right:-10px">Webコンテンツ側</div>
<span style="color:#32CD32">//Service Workerがサポートされているかチェック</span>
if&nbsp;(navigator.serviceWorker)&nbsp;{
&nbsp;&nbsp;&nbsp;<span style="color:#32CD32">//Service Worker を登録</span>
&nbsp;&nbsp;&nbsp;<span style="font-weight:bold;color:fuchsia">navigator.serviceWorker.</span><span style="font-weight:bold;color:blue">register</span><span style="font-weight:bold;color:fuchsia">(&#8216;</span><span style="font-weight:bold;color:blue">/sw.js</span><span style="font-weight:bold;color:fuchsia">&#8216;);</span>
<br />
}
</div>
</div>

<p>&nbsp;<br /></p>

<p>Webコンテンツ側のコードから、navigator.serviceWorker.registerメソッドにより、Service Workerの登録が行われると、Service Worker用のJavaScriptコード（ここではsw.js）ではinstallイベントが発生します。</p>

<div style="width:100%">
<div style="width:600px;background-color:gainsboro;margin:0 auto;padding:10px;font-size:15px">
<div style="text-align:center;width:140px;height:20px;background-color:black;float:right;color:white;margin-top:-10px;margin-right:-10px">sw.js側</div>
<span style="color:#32CD32">//キャッシュするアセットのリスト</span><br />
var <span style="font-weight:bold;color:green">urlsToCache</span> = [<br />
&nbsp;&nbsp;&nbsp;&#8216;/&#8217;,<br />
&nbsp;&nbsp;&nbsp;&#8216;/index.html&#8217;,<br />
&nbsp;&nbsp;&nbsp;&#8216;/css/index.css&#8217;,<br />
&nbsp;&nbsp;&nbsp;&#8216;/script/index.js&#8217;<br />
];<br />
<span style="color:#32CD32">//install イベントのハンドラ</span><br />
<span style="font-weight:bold;color:fuchsia">Self.addEventListener(&#8216;</span><span style="font-weight:bold;color:blue">install</span><span style="font-weight:bold;color:fuchsia">&#8216;, function(event) {</span><br />
&nbsp;&nbsp;&nbsp;event.waitUntil(<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<span style="color:#32CD32">//キャッシュを開く</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;caches.open(&#8216;キャッシュの名前&#8217;)<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;.then(function(cache) {<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<span style="color:#32CD32">//アセットのリストをキャッシュに登録</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return cache.addAll(<span style="font-weight:bold;color:green">urlsToCache</span>);<br />
&nbsp;&nbsp;&nbsp;}));<br />
});
</div>
</div>

<p><br /></p>

<p>install イベントハンドラ内では、キャッシュを開き</p>

<div style="width:100%">
<div style="width:600px;background-color:gainsboro;margin:0 auto;padding:10px;font-size:15px">
<span style="color:#32CD32">//キャッシュするアセットのリスト</span><br />
var urlsToCache = [<br />
&nbsp;&nbsp;&nbsp;&#8216;/&#8217;,<br />
&nbsp;&nbsp;&nbsp;&#8216;/index.html&#8217;,<br />
&nbsp;&nbsp;&nbsp;&#8216;/css/index.css&#8217;,<br />
&nbsp;&nbsp;&nbsp;&#8216;/script/index.js&#8217;<br />
];<br />
<span style="color:#32CD32">//install イベントのハンドラ</span><br />
Self.addEventListener(&#8216;install&#8217;, function(event) {<br />
&nbsp;&nbsp;&nbsp;event.waitUntil(<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<span style="color:#32CD32">//キャッシュを開く</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<span style="font-weight:bold;color:fuchsia">caches.open(</span><span style="font-weight:bold;color:blue">&#8216;キャッシュの名前&#8217;</span><span style="font-weight:bold;color:fuchsia">)</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;.then(function(cache) {<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<span style="color:#32CD32">//アセットのリストをキャッシュに登録</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return cache.addAll(urlsToCache);<br />
&nbsp;&nbsp;&nbsp;}));<br />
});
</div>
</div>

<p>&nbsp;<br /></p>

<p>キャッシュにアセットのリストを登録します。</p>

<div style="width:100%">
<div style="width:600px;background-color:gainsboro;margin:0 auto;padding:10px;font-size:15px">
<span style="color:#32CD32">//キャッシュするアセットのリスト</span><br />
var <span style="font-weight:bold;color:blue">urlsToCache</span> = [<br />
&nbsp;&nbsp;&nbsp;<span style="color:blue">&#8216;/&#8217;,<br />
&nbsp;&nbsp;&nbsp;&#8216;/index.html&#8217;,<br />
&nbsp;&nbsp;&nbsp;&#8216;/css/index.css&#8217;,<br />
&nbsp;&nbsp;&nbsp;&#8216;/script/index.js&#8217;</span><br />
];<br />
<span style="color:#32CD32">//install イベントのハンドラ</span><br />
Self.addEventListener(&#8216;install&#8217;, function(event) {<br />
&nbsp;&nbsp;&nbsp;event.waitUntil(<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<span style="color:#32CD32">//キャッシュを開く</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;caches.open(&#8216;キャッシュの名前&#8217;)<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;.then(function(<span style="font-weight:bold;color:fuchsia">cache</span>) {<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<span style="color:#32CD32">//アセットのリストをキャッシュに登録</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return <span style="font-weight:bold;color:fuchsia">cache.addAll</span>(<span style="font-weight:bold;color:blue">urlsToCache</span>);<br />
&nbsp;&nbsp;&nbsp;}));<br />
});
</div>
</div>

<p>&nbsp;<br /></p>

<p>Service Workerが登録され、installイベント内の処理が無事に完了すると、次にactivateイベントが発生します。（もしinstallイベント内の処理が失敗した際にはerrorイベントが発生します）</p>

<div style="width:100%">
<div style="width:600px;background-color:gainsboro;margin:0 auto;padding:10px;font-size:15px">
<span style="font-weight:bold;color:fuchsia">self.addEventListener(&#8216;</span><span style="font-weight:bold;color:blue">activate</span><span style="font-weight:bold;color:fuchsia">&#8216;,&nbsp;function(e)&nbsp;{</span><br />
&nbsp;&nbsp;e.waitUntil(<br />
&nbsp;&nbsp;&nbsp;&nbsp;caches.keys().then(function(keyList)&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;Promise.all(keyList.map(function(key)&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(key&nbsp;!==&nbsp;&#8216;キャッシュの名前&#8217;)&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;caches.delete(key);<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}));<br />
&nbsp;&nbsp;&nbsp;&nbsp;})<br />
&nbsp;&nbsp;);<br />
<span style="font-weight:bold;color:fuchsia">});</span>
</div>
</div>

<p>&nbsp;<br /></p>

<p>installイベント内で&#8221;しなければいけない処理&#8221;というものは特にないのですが、一般的に古いキャッシュを削除するのに使用されます。このサンプルコードではキャッシュからキーの一覧を取り出し、</p>

<div style="width:100%">
<div style="width:600px;background-color:gainsboro;margin:0 auto;padding:10px;font-size:15px">
self.addEventListener(&#8216;activate&#8217;,&nbsp;function(e)&nbsp;{<br />
&nbsp;&nbsp;e.waitUntil(<br />
&nbsp;&nbsp;&nbsp;&nbsp;<span style="font-weight:bold;color:fuchsia">caches.keys().then(function(</span><span style="font-weight:bold;color:blue">keyList</span><span style="font-weight:bold;color:fuchsia">)&nbsp;{</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;Promise.all(keyList.map(function(key)&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(key&nbsp;!==&nbsp;&#8216;キャッシュの名前&#8217;)&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;caches.delete(key);<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}));<br />
&nbsp;&nbsp;&nbsp;&nbsp;<span style="font-weight:bold;color:fuchsia">})<br />
&nbsp;&nbsp;);</span><br />
});
</div>
</div>

<p>&nbsp;<br /></p>

<p>キャッシュの名前（新しく指定された）と比較し、異なるものをすべて削除しています。</p>

<div style="width:100%">
<div style="width:600px;background-color:gainsboro;margin:0 auto;padding:10px;font-size:15px">
self.addEventListener(&#8216;activate&#8217;,&nbsp;function(e)&nbsp;{<br />
&nbsp;&nbsp;e.waitUntil(<br />
&nbsp;&nbsp;&nbsp;&nbsp;caches.keys().then(function(keyList)&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;<span style="font-weight:bold;color:fuchsia">Promise.all(keyList.map(function(</span><span style="font-weight:bold;color:blue">key</span>)&nbsp;<span style="font-weight:bold;color:fuchsia">{</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<span style="font-weight:bold;color:fuchsia">if&nbsp;(</span><span style="font-weight:bold;color:blue">key</span>&nbsp;<span style="font-weight:bold;color:fuchsia">!==&nbsp;&#8216;キャッシュの名前&#8217;)&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;caches.delete(</span><span style="font-weight:bold;color:blue">key</span><span style="font-weight:bold;color:fuchsia">)<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}));</span>;<br />
&nbsp;&nbsp;&nbsp;&nbsp;})<br />
&nbsp;&nbsp;);<br />
});
</div>
</div>

<p>&nbsp;<br /></p>

<p>上記サンプルコードの、&#8221;キーの一覧と新しく登録されたキャッシュの名前を比較して異なるものをすべて削除してしまう&#8221;、というやり方は少々乱暴かもしれませんので、実際に実装を行う際にはどのようにキャッシュの削除を行うかは十分に考慮してください。</p>

<h3 style="font-weight:bold">なぜ、activateイベントでキャッシュを削除するのか？</h3>

<p>Service Worker自体は、Service Worker用のJSファイルを更新することで再登録されますが、キャッシュが削除されるわけではありません。よって、いずれどこかのタイミングで用済みとなった古いキャッシュを削除する必要が出てきます。</p>

<p>例えば、キャッシュの対象となっているindex.htmlに更新があった場合は、更新前のindex.htmlを保持しているキャッシュを削除しないかぎりユーザーにはいつまでも更新前のindex.htmlが表示されることになります。</p>

<p>Service Worker用のJSファイルが更新されたあと、新しいService Workerはバックグラウンドでインストールされますが、まだ動作はしません。新しいService Workerに制御が移るタイミングは、古いService Workerが制御しているすべてのページが閉じた後からになります。そのためinstallイベントで古いキャッシュを削除してしまうと、まだ動作している古いService Workerはキャッシュを使用できなくなってしまいます。</p>

<p>一方、activateイベントではページの制御が新しいService Workerに移っているので、古いService Workerが使用していたキャッシュを削除することができます。</p>

<h3 style="font-weight:bold">何をキャッシュさせるか？</h3>

<p>Service Workerにキャッシュさせるべきは、App Shellです。App ShellはアプリケーションのUIが機能するために必要な最小限のリソースです。App Shellをローカルにキャッシュすることで、アプリケーションのUI表示と使用可能となるまでの時間を短縮することができ、オフラインでの使用も可能になります。また要件に応じて、キャッシュが有効となるアセット類を指定してもよいでしょう。</p>

<p>しかし、何でもキャッシュさせればよいというものでもありません。例えば、WebGLベースのゲームは、一般的に使用するアセット類のサイズが大きく、これを毎度起動する際にサーバーからダウンロードするのは時間もかかり、通信コストもかさみます。たしかに、これらをキャッシュさせておけば、通信コストを下げ、ゲーム開始までの時間も短縮できます。</p>

<p>しかし、キャッシュストレージの容量には制限があり、また、cacheオブジェクトの代わりに容量の大きいIndexedDBを使用したとしてもクライアントのストレージを占有することになります。</p>

<p>PCなどのストレージ容量の大きいデバイスではあまり問題にならないかもしれませんが、スマートフォンのようなモバイルデバイスではストレージ容量を圧迫することにつながります。よって、なにをキャッシュさせるかは十分に考慮する必要があります。</p>

<h2>リクエストのハンドリング</h2>

<p>Service Workerは有効化(activate)された後、アイドル状態となり、WebページからのリクエストやサーバーからのPushを待ちます。</p>

<p>アイドル状態のとき、制御下にあるWebページでリンクをクリックするなどしてネットワークリクエストが発生するとService Worker ではfetchイベントが発生します。</p>

<div style="width:100%">
<div style="width:600px;background-color:gainsboro;margin:0 auto;padding:10px;font-size:15px">
<span style="font-weight:bold;color:fuchsia">self.addEventListener(&#8216;</span><span style="font-weight:bold;color:blue">fetch</span><span style="font-weight:bold;color:fuchsia">&#8216;,&nbsp;function(e)&nbsp;{</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;e.respondWith(<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;caches.match(e.request)<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;.then(function(response)&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;response&nbsp;||&nbsp;fetch(e.request);<br />
&nbsp;&nbsp;&nbsp;&nbsp;}));<br />
<span style="font-weight:bold;color:fuchsia">})</span>
</div>
</div>

<p>&nbsp;<br /></p>

<p>fetchイベントハンドラの引数として渡されるオブジェクトに、発生したリクエストが含まれるので、同リクエストがキャッシュ内に存在するのか調べ、</p>

<div style="width:100%">
<div style="width:600px;background-color:gainsboro;margin:0 auto;padding:10px;font-size:15px">
self.addEventListener(&#8216;fetch&#8217;,&nbsp;function(<span style="font-weight:bold;color:blue">e</span>)&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;e.respondWith(<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<span style="font-weight:bold;color:fuchsia">caches.match(<span style="font-weight:bold;color:blue">e</span>.request)<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;.then(function(response)&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;response&nbsp;||&nbsp;fetch(e.request);<br />
&nbsp;&nbsp;&nbsp;&nbsp;}));<br />
})
</div>
</div>

<p>&nbsp;<br /></p>

<p>キャッシュ内にリクエストが存在すればそれを返し、存在しなければfetchメソッドを使用してネットワークにリクエストを投げます。</p>

<div style="width:100%">
<div style="width:600px;background-color:gainsboro;margin:0 auto;padding:10px;font-size:15px">
self.addEventListener(&#8216;fetch&#8217;,&nbsp;function(e)&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;e.respondWith(<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;caches.match(<span style="font-weight:bold;color:blue">e.request</span>)<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<span style="font-weight:bold;color:fuchsia">.then(function(</span><span style="font-weight:bold;color:green">response</span><span style="font-weight:bold;color:fuchsia">)&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return</span>&nbsp;<span style="font-weight:bold;color:green">response</span>&nbsp;<span style="font-weight:bold;color:fuchsia">||&nbsp;fetch(</span><span style="font-weight:bold;color:blue">e.request</span><span style="font-weight:bold;color:fuchsia">);</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;}));<br />
})
</div>
</div>

<p>&nbsp;<br /></p>

<p>それぞれの処理結果をリクエスト元のWebコンテンツ側に返します。</p>

<div style="width:100%">
<div style="width:600px;background-color:gainsboro;margin:0 auto;padding:10px;font-size:15px">
self.addEventListener(&#8216;fetch&#8217;,&nbsp;function(e)&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;<span style="font-weight:bold;color:fuchsia">e.respondWith(</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;caches.match(e.request)<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;.then(function(response&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;response&nbsp;||&nbsp;fetch(e.request);<br />
&nbsp;&nbsp;&nbsp;&nbsp;<span style="font-weight:bold;color:fuchsia">}));</span><br />
})
</div>
</div>

<p>&nbsp;<br /></p>

<p>この動作により、Webコンテンツ側ではService Workerやキャッシュを意識することなく、これまで通りの方法でページの制作を行うことができます。</p>

<p>また既存のWebコンテンツにService Workerの機能を追加する場合も、Service Workerを登録するためコードを追加する以外の作業は基本的に必要ありません。</p>

<p>de:code 2018のセッション動画で<a href="https://youtu.be/mk5qkxekFUs?t=20m3s" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">Service Workerを追加するデモ</a>を行っていますので、ぜひ実際の動作をご覧ください。</p>

<h2>Web App Manifest</h2>

<p>Web App Manifestは、デバイスのブラウザーによって[ホーム画面に追加]される際のアイコンや、ホーム画面から起動した際のスプラッシュアイコンや背景色、アプリケーションが動作するウィンドウのスタイルを定義します。</p>

<p>manifestは、json形式で記述し、以下のようなlinkタグをWebコンテンツ/アプリケーション側に追加して参照させます。</p>

<div style="width:100%">
<div style="width:600px;background-color:gainsboro;margin:0 auto;padding:10px;font-size:15px;text-align:center">
&lt;<span style="font-weight:bold;color:fuchsia">link</span> rel=&#8221;<span style="font-weight:bold;color:blue">manifest</span>&#8221; href=&#8221;<span style="font-weight:bold;color:blue">/manifest.json</span>&#8220;&gt;
</div>
</div>

<p>&nbsp;<br /></p>

<p>Web App Manifestについては、詳しい解説が<a href="https://developer.mozilla.org/ja/docs/Web/Manifest" target="_blank" title="Web App Manifest | MDN" data-wpel-link="external" rel="follow external noopener noreferrer">MDN</a>にあるので、そちらを参照することをお勧めしますが、以下に簡単な説明を兼ねたサンプルを掲示します。</p>

<div style="width:100%">
<div style="width:600px;background-color:gainsboro;margin:0 auto;padding:10px;font-size:15px">
{<br />
&nbsp;&nbsp;&nbsp;&#8220;lang&#8221;:&nbsp;&#8220;ja&#8221;,<span style="color:gray">&nbsp;←&nbsp;言語</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&#8220;name&#8221;:&nbsp;&#8220;The enemy of galaxy&#8221;,<span style="color:gray">&nbsp;←&nbsp;アプリケーションの名前</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&#8220;short_name&#8221;:&nbsp;&#8220;T.E.O.G&#8221;,<span style="color:gray">&nbsp;←&nbsp;アプリケーションのショートネーム</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&#8220;start_url&#8221;:&nbsp;&#8220;/?utm_source=pwd&#8221;,<span style="color:gray">&nbsp;←&nbsp;開始するときの&nbsp;URL&nbsp;(※1)</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&#8220;display&#8221;:&nbsp;&#8220;standalone&#8221;,<span style="color:gray">&nbsp;←&nbsp;ウィンドウのスタイル</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&#8220;background_color&#8221;:&nbsp;&#8220;black&#8221;,<span style="color:gray">&nbsp;←&nbsp;背景色</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&#8220;description&#8221;:&nbsp;&#8220;銀河に平和を取り戻すためのゲームです。&#8221;,<span style="color:gray">&nbsp;←&nbsp;説明文</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&#8220;orientation&#8221;:&nbsp;&#8220;portrait&#8221;<span style="color:gray">&nbsp;←&nbsp;画面の向き</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&#8220;icons&#8221;:&nbsp;[{<span style="color:gray">&nbsp;(※2)</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&#8220;src&#8221;:&nbsp;&#8220;images/homescreen48.png&#8221;,<span style="color:gray">&nbsp;←&nbsp;画像ファイルのパス</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&#8220;sizes&#8221;:&nbsp;&#8220;48&#215;48&#8221;,<span style="color:gray">&nbsp;←&nbsp;サイズ</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&#8220;type&#8221;:&nbsp;&#8220;image/png&#8221;<span style="color:gray">&nbsp;←&nbsp;画像の種類</span><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;},&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&#8220;src&#8221;:&nbsp;&#8220;images/homescreen72.png&#8221;,<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&#8220;sizes&#8221;:&nbsp;&#8220;72&#215;72&#8221;,<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&#8220;type&#8221;:&nbsp;&#8220;image/png&#8221;<br />
&nbsp;&nbsp;&nbsp;}]<br />
}
</div>
<div style="width:600px;margin:0 auto">
<p style="margin-top:10px">(※1)Google Analytics等を使用しているときはクエリーストリングを使用することでPWAとして起動されたのか、ブラウザからページにアクセスしたのか判断が可能</p>
<p>(※2)iPhoneやiPadでアイコンが反映されない場合は、apple-touch-iconを使用</p>
</div>
</div>

<p>de:code 2018のセッション動画で<a href="https://youtu.be/mk5qkxekFUs?t=27m24s" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">Web App Manifest を追加するデモ</a>を行っているので、ぜひ実際の動作をご覧ください。</p>

<h3>現在のWindows 10バージョン 1803のProgressive Web Appsの動作</h3>

<p>なお、2018年6月現在のWindows 10バージョン1803(OS ビルド 17134.112)では、残念ながらWeb App Manifestによるウィンドウの制御は有効になりません。</p>

<p>例えば、Windows 10のMicrosoft EdgeでProgressive Web Appsのページをタスクバーにピン留めして起動したとしてもWebブラウザーのUIを表示したまま起動してきます。</p>

<p>Windows 10でProgressive Web Appsにそれらしい動作をさせるにはUWP(Universal Windows Platform)アプリでラップする必要があります。</p>

<p>しかし、Windows 10の次のアップデートでは、Web App Manifestのdisplayの設定に対し、PWAのウィンドウは<a href="https://youtu.be/6fb-t9ffDvo?t=10m5s" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">以下のような表示になるそうです。</a></p>

<p><img src="https://msdnshared.blob.core.windows.net/media/2018/07/win10display.png" alt="" width="1283" height="449" class="aligncenter size-full wp-image-5605" />
<br />&nbsp;</p>

<h2>マイクロソフトのProgressive Web Appsへの<br />取り組み</h2>

<p>Windows 10の[Microsoft Store]から Progressive Web Appsとして作られたアプリを入手することができます。</p>

<p>Web App Manifestのところで書いたとおり、現在のWindows 10にプレーンな Progressive Web Appsをインストールしても、ネイティブアプリのような外観にはなりません。そのためMicrosoft Storeから入手できるProgressive Web AppsはUWPアプリにラップされています。</p>

<h3>MicrosoftストアでのProgressive Web Appsの公開</h3>

<p>MicrosoftストアでProgressive Web Appsを公開するには 2つの方法があります。</p>

<p>1つめがBingによる自動インデックスによる登録と、2つめがProgressive Web Appsの提供者がMicrosoftストアにProgressive Web Appsを提出するセルフパブリッシングです。</p>

<p>以下でこの 2つの方法について紹介します。</p>

<h3>Bingによる自動インデックス(BETA)</h3>

<p>現在はまだBETAの状態ですが、BingがWeb上のProgressive Web Appsを検出し、レビューを行い、自動的にMicrosoftストアに公開します。</p>

<p>Bingに自動インデックスされるための条件は、現在のところ以下のようなものがあります。</p>

<ul>
<li>HTTPS、セキュアなエンドポイント</li>
<li>高品質の manifest</li>
<li>オフラインサポート</li>
<li>ストアの審査をパス
   <ul>
      <li>デジタルグッズを販売しないこと</li>
      <li>アダルトコンテンツを含まないこと</li>
      <li>不快なコンテンツは不可</li>
</ul></li>
<li>Windowsならではの差別化がされている</li>
</ul>

<p>上記リストで、最後の&#8221;Windowsならではの差別化がされている&#8221;というのは、少しわかりづらいかもしれません。</p>

<p>Microsoft Storeで公開されるProgressive Web Appsは、UWPアプリにラップされるので<a href="https://docs.microsoft.com/ja-jp/uwp/api/" target="_blank" title="Windows UWP Namespaces" data-wpel-link="external" rel="follow external noopener noreferrer">WinRT (Windows Runtime) API</a>を使用することができます。</p>

<p>つまりWinRT APIを使用することで、Windows 10で使用した際には、純粋なProgressive Web Appsからはアクセスできないプラットプラットフォームやハードウェアリソースの機能を使用することができるため、これらを利用した機能を実装することで差別化を図ることができます。</p>

<h3>Microsoftストア向けProgressive Web Appsの検証</h3>

<p>決して、Progressive Web Appsの検証に限ったものでなく、ましてやMicrosoftストア向けのWebコンテンツ用に限定するものではありませんが、WebサイトやProgressive Web Appsの品質をチェックするための<a href="https://sonarwhal.com/" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">sonarwhal</a>というサービスが公開されています。</p>

<p><a href="https://sonarwhal.com/" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer"><img src="https://msdnshared.blob.core.windows.net/media/2018/07/sonarwhal.png" alt="" width="400" class="aligncenter size-full wp-image-5625" /></a></p>

<p>sonarwhalはProgressive Web AppsがMicrosoftストア向けの準備がてきているか、だけでなく、一般的なWebサイトやアプリケーションのパフォーマンスやユーザーエクスペリエンス、セキュリティの向上の役に立つ情報を提供してくれます。</p>

<h3>Bingに検出されないようにするには</h3>

<p>インターネットで公開しているProgressive Web AppsをMicrosoftストアから配布されたくない場合、Progressive Web Appsが以下のいずれかの条件に合致していた場合、Bingは自動インデックスを行いません。</p>

<ul>
<li>manifest なし</li>
<li>https なし、もしくは オフラインで動作しない</li>
<li>Robot.txtファイルが設定されている</li>
<li>manifestのdisplay設定がbrowser</li>
</ul>

<p>既に Microsoftストアで公開されているものを取り消したい場合は、アプリを削除するよう <a href="reportapp@microsoft.com" data-wpel-link="internal">reportapp@microsoft.com</a>宛にサービスリクエストを出します。</p>

<h3>セルフパブリッシングによるMicrosoftストアでの公開</h3>

<p>Bingの自動インデックスに期待するのではなく、明示的にMicrosoftストアでProgressive Web Appsを公開するには<a href="https://developer.microsoft.com/" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">dev.microsoft.com</a>で開発者アカウントを取得し、パッケージ(appx)化してMicrosoftストアに提出します。アプリの内容がMicrosoftストアの審査をパスすればMicrosoftストアで公開されます。</p>

<p>Progressive Web Appsをパッケージ化するには以下の方法があります。</p>

<ul>
<li><a href="https://visualstudio.microsoft.com/ja/" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">Visual Studio 2018</a> を使用する</li>
<li><a href="https://www.pwabuilder.com/" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">PWABuilder</a> を使用する</li>
<li><a href="https://github.com/pwa-builder/ManifoldJS/releases" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">pwa-builder CLI</a> を使用する</li>
</ul>

<h3 style="font-weight:bold">PWABuilder</h3>

<p><a href="https://www.pwabuilder.com/" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer"><img src="https://msdnshared.blob.core.windows.net/media/2018/07/PWABuilder.png" alt="" width="400" class="aligncenter size-full wp-image-5635" /></a></p>

<p>PWABuilderはインターネットにホストされているサービスで、同じくインターネットにホストされているWebアプリケーションのためのmanifestファイルやService Workerのコード、Service Workerを登録するためのコードを生成します。</p>

<p>また、UWPのHostedアプリのパッケージも生成できるので、Progressive Web AppsをMicrosoftストアで公開する際にはこれをストアに提出します。</p>

<p>PWABuilderを使用した既存のWebアプリケーション用のmanifestファイルやService Workerのコードを生成する手順については de:code 2018 の
<a href="https://youtu.be/mk5qkxekFUs?t=35m19s" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">セッション動画のデモ</a>をご覧ください。</p>

<h3 style="font-weight:bold">Windows ストア(UWP) アプリの価値</h3>

<p>前述しましたが、Progressive Web AppsをUWPアプリ(<a href="https://msdn.microsoft.com/en-us/magazine/mt797652.aspx" target="_blank" title="UWP Apps - Develop Hosted Web Apps for UWP" data-wpel-link="external" rel="follow external noopener noreferrer">Hosted Web アプリ</a>)としてラップすることで WinRT が提供するすべての機能を利用することができます。</p>

<p>これは予定表やアドレス帳といったプラットフォームが提供するソフトウェア的なリソースはもちろん、USBやセンサー類といったハードウェアリソースも含まれます。</p>

<p>アプリケーションが通常のWebブラウザー上で動作しているのか、UWPで動作しているかは JavaScriptのif文で簡単に判断できるので、WinRT APIにアクセスするためのコードをインターネットでホストされているコンテンツに含めておくことができます。</p>

<div style="width:100%">
<div style="width:600px;background-color:gainsboro;margin:0 auto;padding:10px;font-size:15px">
//UWP として動作しているかどうか<br />
if(window.Windows){<br />
&nbsp;&nbsp;&nbsp;//セカンダリータイルをピン留めする<br />
&nbsp;&nbsp;&nbsp;tile&nbsp;=&nbsp;new&nbsp;Windows.UI.StartScreen.SecondaryTile(<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;tileId,&nbsp;text,&nbsp;text,&nbsp;&nbsp;activationArguments,<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;newTileDesiredSize,&nbsp;logoUri);<br />
}
</div>
</div>

<p><br /></p>

<p>JavaScriptで記述したUWPから,WinRT APIを使用するサンプルコードは以下に豊富に用意されているので、Windows 10限定となってしまいますが、純粋なProgressive Web Appsの機能だけでは実現できない要件がある場合は、こういった方法もあるということを覚えておくとよいかもしれません。</p>

<p><a href="https://developer.microsoft.com/ja-jp/windows/samples" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">・コードサンプル &#8211; Windows アプリの開発</a></p>

<p>Visual Studio 2018を使用してProgressive Web AppsをUWPでラップし、WindowsRT APIを使用する方法については、de:code 2018 の<a href="https://youtu.be/mk5qkxekFUs?t=42m9s" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">セッション動画のデモ</a>をご覧ください。</p>

<p>その他、Progressive Web AppsをUWPとしてMicrosoftストアに公開することで、Microsoftストアの提供するマーケティングデータ (ダウンロード数、ユーザーの性別、年齢層、国、etc.)や、決済のためのストアのAPIが利用できたり、人気があればMicrosoftストアのトップページに掲示されたり、とさまざまなメリットもあります。</p>

<p>しかも、Hostedアプリであれば、共通のコンテンツでありながら純粋なままのProgressive Web Appsがインターネットでホストされているので、アプリストアを通さない場合のメリットも享受できます。</p>

<p>【参考】<br />
<a href="https://developer.microsoft.com/ja-jp/windows/pwa" target="_blank" style="font-size:15px" data-wpel-link="external" rel="follow external noopener noreferrer">・Windows デベロッパーセンター &#8211; PWAとWindows10</a>
</p>

<p>この記事では、UWPのHostedアプリについて紹介しましたが、Apache Cordovaでもさまざまなモバイル プラットプラットフォーム向けに Hostedアプリを開発することができます。興味のある方は以下の記事をご覧ください。</p>

<p><a href="https://docs.microsoft.com/en-us/visualstudio/cross-platform/tools-for-cordova/first-steps/create-a-hosted-app?view=toolsforcordova-2015" target="_blank" style="font-size:15px" data-wpel-link="external" rel="follow external noopener noreferrer">・Create a hosted web app using Apache Cordova</a></p>

<h2>Progressive Web Appsの価値を高める API</h2>

<p>ここまでWindows 10で動作するUWPアプリでラップされたProgressive Web Appsの紹介をしていましたが、ここからは再びプレーンな Progressive Web Appsについてです。</p>

<p>Progressive Web Appsは、ネイティブアプリのような体験を提供しますが、WebアプリケーションなのでWebブラウザーの機能を超えて動作することはできません。</p>

<p>例えば、表示速度や動作速度はネイティブアプリと比較するとどうしても劣ります。</p>

<p>しかし、Webブラウザーで新しくサポートされた以下のような機能を使用すれば、その能力的な差を縮めることができます。これまでの Webコンテンツにはなかった新しい体験をユーザーに提供したり、決済の仕組みもWeb標準のAPIを使用してこれまでよりも少ない工数で組み込むことができるので、これらを使用してアプリケーションの価値を高めることができます。</p>

<ul>
<li><span style="font-weight:bold;font-size:15px"><a href="https://www.w3.org/TR/css3-namespace/" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">CSS3</a></span>
3D Transform関連等GPUを使用する設定があり、これを意識的に利用することで描画速度を向上させることができます。
</li>
<li><span style="font-weight:bold;font-size:15px"><a href="https://webassembly.org/" target="_blank" data-wpel-link="external" rel="follow external noopener noreferrer">WebAssembly</a></span>
C言語などで記述し、あらかじめコンパイルしたものをWebブラウザーで動作させるため高速な計算が可能となります。
</li>
<li><span style="font-weight:bold;font-size:15px"><a target="_blank" data-wpel-link="internal">WebVR</a></span>
VRデバイスの提供するAPIをサポートし、ユーザーによりImmersiveな体験を提供できます。
</li>
<li><span style="font-weight:bold;font-size:15px"><a href="https://developer.microsoft.com/en-us/microsoft-edge/testdrive/demos/paymentrequest/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Web Payments</a></span>
決済画面その他、支払いに関する機能を提供します。これとPaypalやStripeなどが提供する決済用のAPIを繋げれば少ない工数で独自に決済期の機能を実装できます。自前の決済の仕組みがあれば、もう何かを販売するたびにアプリ ストアに使用料を支払う必要はありません</li>
</ul>

<p>この他にも、最新のWebブラウザーにはこれまでできなかった機能が日々搭載されてきます。そういった新機能をいち早く利用できるのも、逆にレガシーなWebブラウザーには引き算した機能を提供できるのもProgressive Web Appsの良いところです。(もちろん、そう作れば、ですが)</p>

<h1>Progressive Web Appsの位置づけ</h1>

<p>PWAことProgressive Web Appsは「ネイティブアプリのような体験を提供する」という言葉が独り歩きしてしまい、まるで「ネイティブアプリを置き換える」といった印象を持っている人も多いようです。とはいえ、「ネイティブアプリを置き換えることはできない」というのも真ではないでしょう。</p>

<p>例えば、スマートフォンアプリには、アプリとしてインストールさせるためだけに外側をWeb Viewでラップしたいわゆる「ガワアプリ」と呼ばれるものがあります。こういったものは今後はProgressive Web Appsに置き換えられていくでしょう。</p>

<p>実際にイベントサイトや、カンファレンスのセッションスケジュールの部分をPWA化してユーザーが持ち歩けるようにしているサイトはすでにいくつか存在します。</p>

<ul>
<li><a href="https://abc.android-group.jp/2018s/" target="_blank" style="font-size:15px" data-wpel-link="external" rel="follow external noopener noreferrer">ABC 2018 Spring | 日本最大級のAndroidデベロッパーの祭典</a></li>
<li><a href="https://ngjapan.org/" target="_blank" style="font-size:15px" data-wpel-link="external" rel="follow external noopener noreferrer">ng-japan &#8211; The Angular conference in Tokyo, Japan (2018/6/16)</a></li>
</ul>

<p>こういったものもProgressive Web Appsが出てくる以前はWebページとは別にスマートフォン用のアプリを、場合によってはプラットフォームごとに別々の言語で開発し、それぞれのアプリストアに提出して審査を受けるといったことをしていました。</p>

<p>Webページとアプリの開発が共通化でき、配布の手間もかからないということは予算や時間的にも非常に大きなメリットがあります。</p>

<p>しかし、Webブラウザー内のJavaScriptからアクセスできないプラットフォームの機能を利用するアプリケーションや、高速な動作や描画を必要とする用途には今後もネイティブ アプリが選択されることでしょう。</p>

<p><img src="https://msdnshared.blob.core.windows.net/media/2018/07/PWA_position.png" alt="" width="1141" height="292" class="aligncenter size-full wp-image-5645" /></p>

<p>Progressive Web AppsはWebアプリケーションとネイティブアプリの体験的なメリットを受け継ぎ、ちょうどその中間に位置し、その隙間を埋めるものです。</p>

<p>Webのページではpoorだったもの、逆にネイティブアプリではtoo muchだったものが、Progressive Web Appsには最適なのかもしれません。</p>

<p>少し作って試すだけの実装ならそれほど難しくありませんので、気になる人はぜひProgressive Web Appsを作ってみることをお勧めします。</p>

<p>&nbsp;</p>
]]></content:encoded>
			<wfw:commentRss>/osamum_ms/25709/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>「改めまして、Progressive Web Appsと申します」── Web UXの新たな基準を考える</title>
		<link>/uskay/25391/</link>
		<pubDate>Thu, 19 Apr 2018 03:23:45 +0000</pubDate>
		<dc:creator><![CDATA[宇都宮佑亮]]></dc:creator>
				<category><![CDATA[最新動向]]></category>
		<category><![CDATA[PWA]]></category>
		<category><![CDATA[Progressive Web Apps]]></category>
		<category><![CDATA[Service Worker]]></category>
		<category><![CDATA[UX]]></category>
		<category><![CDATA[Webアプリ]]></category>

		<guid isPermaLink="false">/?p=25391</guid>
		<description><![CDATA[Progressive Web Appsというワードが世に出て約2年半が経ちました。2015年10月に開催されたChrome Dev SummitにてFlipkartの事例をもってお披露目となったそのコンセプトは、201...]]></description>
				<content:encoded><![CDATA[<p><strong>Progressive Web Appsというワードが世に出て約2年半が経ちました。</strong>2015年10月に開催されたChrome Dev Summitにて<a href="https://www.youtube.com/watch?v=StdKz32M1RM" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Flipkartの事例</a>をもってお披露目となったそのコンセプトは、2018年現在までに徐々に成功事例を増やしながらWeb界隈の注目を集め、ついに先日（忘れもしない2018年3月30日！）iOS 11.3からiOSデバイスでも一部の機能が利用できるようになるまで成長しました。これは、まるで進学する我が子を見ているかのような、新年度にふさわしい晴れやかなニュースですし、<strong>いい機会なので PWAとは何かを改めて振り返ってみようと思います。</strong></p>

<p><img src="/wp-content/uploads/2018/04/pwa01.jpg" alt="pwa01" /></p>

<h2>Webに足りなかったもの</h2>

<p>私はWebが大好きです。リンクを1つクリックしたら（インストールなど煩わしい手続きなしで）すぐに新しいコンテンツを読めるのは最高の体験だなと常日頃感じています。ただし、<strong>今までのWebアプリのユーザー体験は決して最適なものではないとも思っています。</strong>たとえば以下のような体験をした覚えはありませんか？</p>

<ul>
<li>朝の満員電車でも情報収集に勤しむ現代のビジネスパーソンのあなた。つり革に捕まりながらなんらかの記事コンテンツを見ようとリンクをタップする</li>
<li>途端にHTMLのレスポンス待ちで真っ白な画面。レスポンスが返ってきたと思ったら今度はWebアプリ側のローディングGIFが動き出す。</li>
<li>しばらく待つとテキストが表示され読み始めるものの、時間差で画像が差し込まれレイアウトがずれる（どこを読んでいたんだっけ？）。極めつけは全画面広告が画面を占拠し、嫌になってそっとブラウザを閉じる（代わりにソーシャルアプリやゲームアプリを起動する）。</li>
</ul>

<p>このように世に出回っている多くのWebアプリはパフォーマンスに課題があります。特に、さまざまなスペックなデバイスがあらゆる環境で利用されるモバイルにおいては、Webアプリは単純に耐えられないくらい遅く、実測で3G回線と同等な環境における平均ページロード時間は19秒もかかっているという統計もあります[1]。 <strong>Webアプリはとにかくこのパフォーマンスをどうにかしなければなりません。</strong></p>

<p>一方ですでにネイティブ アプリに慣れ親しんでいるユーザーの期待値を満たすためには、パフォーマンスを改善するだけでは十分ではありません。</p>

<ul>
<li>ホーム画面にアイコンを追加したり</li>
<li>プッシュ通知がタイミングよく送られてきたり</li>
<li>オフラインでも動作する信頼性だったり</li>
</ul>

<p>「ユーザー体験の基準」がどんどん上がっていくなか、Webアプリだけがこのような体験を提供できずに取り残されている状況がずっと続いていました。<strong>時代に合わせて、ユーザーが求める機能性もWebアプリで実現できなければならないのです。</strong></p>

<h2>PWAはベストプラクティス集である</h2>

<p>そこで満を持して登場したのがPWAです。PWAと聞くと何かを特定の技術を指すものと思われるかもしれませんが、実はそのコンセプトは幅広く、どちらかというと前述した <strong>Webアプリ本来の課題を解決し、よりよいユーザー体験を実現するためのベストプラクティス集のようなものです。</strong>Googleが提供している<a href="https://developers.google.com/web/progressive-web-apps/checklist" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Progressive Web App Checklist</a>
なるものがあり詳細な内容はそちらを確認いただければと思いますが、PWAのコンセプトをざっくり申し上げると以下となります。</p>

<ul>
<li><strong>とにかくパフォーマンスいいWebアプリを作りましょう。</strong>SPAでもSPAじゃなくても、どんなフレームワーク使っても使わなくても、なんでもいいので、表示が速くて画面遷移がスムーズなものにしてユーザーに好かれましょう。</li>
<li>また、最新のWeb APIを積極的に使って、機能性も身につけましょう。たとえば、Webアプリをホーム画面に追加できるようにしたり、オフラインでも動作するようにしたり。<strong>そんな今までWebアプリでは実現できなかった機能が実装できるようになったんだから、やらない手はないでしょう？</strong></li>
</ul>

<p>少し上記でも触れていますが、PWAはあくまでベストプラクティス集なので特定のサードパーティーフレームワークやライブラリには依存しません。世の中には ReactやAngularを使ったPWAもありますし、Vanilla JavaScriptなPWAもあります。また、サーバーサイドの構成には一切言及しません。<strong>したがって技術スタックを変えずとも適用可能なベストプラクティスでありますし、一度に全部を適用するのではなく段階的に（= Progressively）最適化することもできます。</strong></p>

<p>以下の項目でPWAの特徴をより具体的に見ていきます。</p>

<h3>ハイパフォーマンス</h3>

<p>何より重要であり最も難しいテーマとなります。パフォーマンスの基準としては、2015年にChromeチームが発表したユーザー中心に考えるパフォーマンスモデルである<strong>RAIL</strong>[2]を前提に考えていただければ間違いありませんが、各Webアプリの特性や環境に応じてそれぞれのサービスごとに <strong>Performance Budget</strong>（ロード時間やファイルサイズ等の上限を決め、それを「予算」として管理する）を設けることをオススメします。たとえば、2017年のChrome Dev SummitでChromeチームが推奨したPerfromance Budgetは、<strong>TTI（Time to interactive)を5秒以内</strong>、それを実現するために<strong>クリティカルパスのファイルサイズを170KB以内に収める</strong>というものです。[3][4]</p>

<p>何事もまずは現状分析から。Webアプリのパフォーマンス測定やプロファイリングをする<a href="https://developers.google.com/web/tools/lighthouse/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Lighthouse</a>や<a href="https://developers.google.com/web/tools/chrome-devtools/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Chrome DevTools</a>等のツールを使って現状の評価とボトルネックを確認し、対応すべきポイントを決めて最適化を進めます。その後は<a href="https://calibreapp.com/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Calibre</a>や<a href="https://speedcurve.com/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">SpeedCurve</a>といったパフォーマンスモニタリングツールを利用して定点観測するといいでしょう。</p>

<p><img src="/wp-content/uploads/2018/04/pwa02.png" alt="pwa02" /></p>

<p>ハイパフォーマンスなPWAを作るためのデザインパターンである<a href="https://developers.google.com/web/fundamentals/performance/prpl-pattern/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">PRPLパターン</a>もまた参考にすべき方式です。PRPLパターンは<strong>Push 、Render、Pre-cache、Lazy-load</strong>の頭文字を取ったもので、主に<a href="https://developers.google.com/web/fundamentals/architecture/app-shell?hl=ja" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">App Shell モデル</a>やSPAを採用したWebアプリのために用意されたパターンですが、それ以外の構成に関してもパフォーマンス向上のヒントとなる点が多くあります。</p>

<ul>
<li><strong>Push/Render:</strong> 初期描画に必須なリソースを<a href="https://tools.ietf.org/html/rfc7540#section-8.2" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">HTTP/2 Server Push</a>するか <a href="https://developer.mozilla.org/en-US/docs/Web/HTML/Preloading_content" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">link rel=preload</a> を利用することで先行取得します。そしてそのリソースを利用し、特にAbove the fold [5] (スクロールしないで見える範囲)を優先してレンダリングします。</li>
<li><strong>Pre-cache:</strong> 次ページ（またはルート）で使用するであろうリソースを先読みします。これには<a href="https://developer.mozilla.org/en-US/docs/Web/API/Service_Worker_API" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Service Worker</a>を利用するといいでしょう。</li>
<li><strong>Lazy-load:</strong> 初期描画に必要なもの以外はすべてLazy-loadします。SPAであれば別ルートの取得がこちら該当しますし、スクロールに合わせたフラグメントの取得であれば<a href="https://developer.mozilla.org/en-US/docs/Web/API/Intersection_Observer_API" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Intersection Observer API</a>を利用するといいでしょう。</li>
</ul>

<p>上記のとおり、Webアプリでもパフォーマンスをあげるためのツールやデザインパターンなど、使えるナレッジが多くたまってきました。ぜひともこれらを利用してハイパフォーマンスなWebアプリの開発を実現してください。</p>

<h3>ホーム画面に追加</h3>

<p><a href="https://developer.mozilla.org/en-US/docs/Web/Manifest" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Web App Manifest</a>を実装するとWebアプリをホーム画面に追加できるようになります。以下のようなJSON形式のマニフェストファイルを用意し、</p>

<p></p><pre class="crayon-plain-tag">{
   "name": “PWA Demo Application”,    
   "short_name": “PWA demo”,
    "icons": [{
        "src": “/assets/icon.png”, "sizes": “192x192”
     }],
    "start_url": “/index.html”,
    "display": “standalone”,
    "scope": “/”
}</pre><p></p>

<p>link タグでページにひも付けた上で、Service Workerをインストールすると、</p>

<p></p><pre class="crayon-plain-tag">&lt;link rel="manifest" href="/manifest.json"&gt;

&lt;script&gt;
if ('serviceWorker' in navigator)    
   navigator.serviceWorker.register('/sw.js')
&lt;/script&gt;</pre><p></p>

<p>このようにホーム画面へ追加を促すプロンプトが上がってきます。ホームに追加した後はフルスクリーンで起動しますし、Androidの場合は1つの アプリとしてみなされディープリンクまで可能です。</p>

<p><img src="/wp-content/uploads/2018/04/pwa03.gif" alt="pwa03" /></p>

<p>1点補足すると、この「ホーム画面に追加」は強力ですが、よりユーザーに気持ちよく追加してもらえるように、<strong>意味のあるタイミングでプロンプトを表示するとさらにいいでしょう。</strong>これを実現するためにぜひとも<code>onbeforeinstallprompt</code>イベントハンドラーを利用してください。</p>

<p><code>onbeforeinstallprompt</code>イベントハンドラーはプロンプトが表示される直前に呼び出されます。そのタイミングでプロンプト表示が適切でなければ表示をキャンセルし、別の任意のタイミングで表示することができます。たとえば、画面上に独自の「ホームに追加アイコン」を作成し、それの <code>onclick</code>のイベントハンドラ内でプロンプトを表示させることも可能です。</p>

<p></p><pre class="crayon-plain-tag">var deferredPrompt;
window.addEventListener('beforeinstallprompt', (e) =&gt; {
  console.log('beforeinstallprompt Event fired');
  // デフォルトのタイミングでは表示させない
  e.preventDefault();
  // Eventオブジェクトを保存しておく
  deferredPrompt = e;
  return false;
});
// どっか別のタイミングで...
deferredPrompt.prompt();</pre><p></p>

<p>また執筆時点での最新の<a href="https://www.google.com/chrome/browser/canary.html" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Chrome Canary</a> (67.0.3394.0)ではプロンプト表示形式を以下のように変更して、ホームに追加の体験をより良くすべく機能検証をしています。ポイントはページ下部に表示されるバナーがコンテンツの邪魔にならないサイズに調整され、PWAが追加済であればそのディープリンク用のバナーに差し替えます。また、前述の<code>prompt</code>関数を利用してユーザーのインタラクションに応じてプロンプトを表示する際は、モーダル表示に切り替わります。</p>

<p><img src="/wp-content/uploads/2018/04/pwa04.png" alt="pwa04" /></p>

<h3>オフライン対応</h3>

<p>ハイパフォーマンスの解説でも少し登場した<strong>Service Worker</strong> と <a href="https://developer.mozilla.org/en-US/docs/Web/API/Cache" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Cache API</a>を利用することでWebアプリをオフラインで実行することが可能になります。Service Workerとはオリジン単位（またはパスの粒度）でインストール可能なJavaScriptで書かれたクライントサイドプロキシのことで、ページ内で発生するHTTPリクエスト / レスポンスを監視したり、それを書き換えたり、必要に応じてリソースを先読みすることもできます。これとCache APIというHTTPリクエスト/ レスポンスオブジェクトをKey-Value形式で保存できるAPIを組み合わせることで、<strong>オフライン時でもあらかじめキャッシュしておいたレスポンスを利用すことができるようになるわけです。</strong></p>

<p><img src="/wp-content/uploads/2018/04/pwa05.png" alt="pwa05" /></p>

<p>このService WorkerとCache APIの組み合わせは<a href="https://developers.google.com/web/fundamentals/primers/service-workers/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">「Service Worker の紹介」</a>に記載の方法で使用することができますが、より汎用的な使い方をまとめた<a href="https://developers.google.com/web/tools/workbox/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Workbox</a>というライブラリを利用するのも1つの手です。Workbox を利用すると実装が煩雑になる、特定のファイルのラインタイムキャッシュも以下のようにシンプルに記述することができます。</p>

<p></p><pre class="crayon-plain-tag">workbox.routing.registerRoute(
  new RegExp('.*\.js'),
  workbox.strategies.networkFirst()
);</pre><p></p>

<p>Workboxはその他にも、<a href="https://developers.google.com/web/tools/workbox/modules/workbox-precaching" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">ファイル単位でリビジョンを付与したPrecaching</a>、<a href="https://developers.google.com/web/tools/workbox/modules/workbox-cache-expiration" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Expirationの定義</a>や<a href="https://developers.google.com/web/tools/workbox/guides/enable-offline-analytics" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Background Syncを利用したオフライン時の Google Analytics の計測</a>などの機能も提供します。</p>

<p>オフラインの戦略はWebアプリの特性に応じて変わってくるものです。例えばニュースサイトであれば、トップページに出ている注目記事をすべて先読みし、オフラインでも読めるようにすることも考えられますし、ECサイトであればまた別の戦略が適している場合もあるでしょう。電波状況が比較的良好な日本においても、速度制限や電車内などオフライン（または不安定なネットワーク）になりえるユースケースは存在しますし、最低限でも <a href="https://www.google.co.jp/search?q=Offline+Dinosaur&amp;source=lnms&amp;tbm=isch&amp;sa=X&amp;ved=0ahUKEwiHmNqQ05vaAhUDOJQKHUkjBh0Q_AUICigB&amp;biw=1745&amp;bih=1003" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Offline Dinasour</a>はユーザーに見せないようにフォールバックページを用意すべきです。</p>

<h3>その他のWeb API</h3>

<p>以下にPWAを実装する上で注目すべきその他のWeb APIを列挙します。</p>

<ul>
<li><strong>プッシュ通知:</strong> <a href="https://developer.mozilla.org/en-US/docs/Web/API/Push_API" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Push API</a> / <a href="https://developer.mozilla.org/en-US/docs/Web/API/notification" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Notification API</a></li>
<li><strong>自動ログイン:</strong> <a href="https://developer.mozilla.org/en-US/docs/Web/API/Credential_Management_API" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Credential Management API</a></li>
<li><strong>支払いフォームの強化:</strong> <a href="https://developer.mozilla.org/en-US/docs/Web/API/Payment_Request_API" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Payment Request API</a></li>
</ul>

<p>上記のAPIも組み合わせることによって、リエンゲージメント性の向上や商品の購買フローを最適化することができます。</p>

<h2>クロスプラットフォームなイニシアチブ</h2>

<p>上記にあげたPWAで利用するWeb APIは、すべてのブラウザで利用できることを目標に標準化を進めています。個別の対応状況を確認するには<a href="https://caniuse.com/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Can I Use</a>をご参照いただければと思いますが、冒頭に述べましたとおりiOSでもiOS 11.3からSafariでService Workerが動作するようになりましたし、Microsoft はWindowsストアにPWAを載せはじめました[6]。このように、さまざまな形でPWAのクロスプラットフォーム利用の機運が高まっていると言えます。</p>

<p>また、PWAは<a href="https://en.wikipedia.org/wiki/Progressive_enhancement" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Progressive Enhancement</a>な実装を推奨している点も改めて触れておきます。たとえばService Workerを利用したオフラインキャッシュの実装は、Service Workerが未実装なブラウザでもWebアプリ本来の動作には影響がないように以下のように条件分岐をいれることを推奨しています。</p>

<p></p><pre class="crayon-plain-tag">// ServiceWorkerが利用できる場合にのみ
if ('serviceWorker' in navigator) {
  // ServiceWorkerを登録する
  navigator.serviceWorker.register('/sw.js')
}</pre><p></p>

<p>特定の環境のみでしか動かないWebアプリを作るのではなく、どの環境でもハイパフォーマンスに動作するようにし、最新のWeb APIが利用できる環境では進んでそれをレバレッジするとよいでしょう。</p>

<h2>Webアプリ vs ネイティブ アプリ</h2>

<p>PWAはその機能性や実現したいコンセプトから「PWAはネイティブ アプリを潰そうとしているのか？」「Webアプリがネイティブアプリにかなうはずない！」など当該テーマにおいては辛辣なコメントを見かける場面もあります。ここで1つお伝えしたいことは、多くのユーザーはネイティブアプリなのかWebアプリなのかというプラットフォームの選択には関心がないということです。大事なのは「うまい、やすい、はやい」ユーザー体験の良いサービスを提供することです。PWAの出現によりWebアプリでもそれを実現する準備が整いつつあり、<strong>事業者やソフトウェア エンジニアとしては理想のサービスを提供できるプラットフォームのオプションが増えるという意味で、このWebの進化を前向きに捉えていただければ幸いです。</strong></p>

<p>実際Webアプリにはまだできないこともあります[7]。ただし、Webアプリには他のプラットフォームよりも優れたリーチがあり、<strong>その意味でWeb アプリとネイティブ アプリがむしろ手を取ることは、戦略としても決して矛盾しません。</strong></p>

<p><img src="/wp-content/uploads/2018/04/pwa06.png" alt="pwa06" /></p>

<p>使用したい機能がWebプラットフォームが提供するもので十分であり、技術スタックをWebのみに集約するメリットも感じるのであれば「PWA のみ」という選択肢も今後は出てくるかと思います。しかし、前述した「まだできないこともある」という前提を加味するとネイティブアプリが活躍する場面は引き続きありますし、<strong>PWAは「Webアプリ vs ネイティブアプリ」という二者択一を迫る提案ではない点をご理解ください。</strong></p>

<h2>ユーザー体験の新たな基準</h2>

<p>このようにPWAはWebアプリの「ユーザー体験の新たな基準」を満たすために作られたベストプラクティスでしかありません。重要なのはそれぞれの機能を単純に実装するのではなく、Webアプリの特性に応じて使い分け、ときにはカスタマイズし、ユーザーによりよいサービスを提供することです。今回ご紹介しました各種ツールやデザインパターン、iOSでもService Workerが利用できるようになった点など、舞台は整いつつあります。<strong>ぜひとも最高のWeb体験を実現すべくPWAの考え方を取り入れてみてください。</strong></p>

<ul>
<li>[1] <a href="https://www.doubleclickbygoogle.com/articles/mobile-speed-matters/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">The need for mobile speed: How mobile latency impacts publisher revenue</a></li>
<li>[2] <a href="https://developers.google.com/web/fundamentals/performance/rail" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">RAILモデルでパフォーマンスを計測する</a></li>
<li>[3] <a href="https://youtu.be/_srJ7eHS3IM" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Fast By Default: Modern Loading Best Practices (Chrome Dev Summit 2017)</a></li>
<li>[4] <a href="https://infrequently.org/2017/10/can-you-afford-it-real-world-web-performance-budgets/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Can You Afford It?: Real-world Web Performance Budgets</a></li>
<li>[5] <a href="https://support.google.com/adsense/answer/132618" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Above the fold</a></li>
<li>[6] <a href="https://www.windowscentral.com/first-batch-windows-10-progressive-web-apps-here" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">First Windows 10 Progressive Web Apps (PWA) published by Microsoft hit the Store</a></li>
<li>[7] <a href="https://whatwebcando.today/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">What Web Can Do Today</a></li>
</ul>
]]></content:encoded>
			</item>
		<item>
		<title>超詳解！Service Worker Deep Dive ── HTML5 Conference 2016セッションレポート</title>
		<link>/horo/21360/</link>
		<pubDate>Mon, 24 Oct 2016 01:05:13 +0000</pubDate>
		<dc:creator><![CDATA[保呂毅]]></dc:creator>
				<category><![CDATA[最新動向]]></category>
		<category><![CDATA[プログラミング]]></category>
		<category><![CDATA[Service Worker]]></category>
		<category><![CDATA[Web Push]]></category>

		<guid isPermaLink="false">/?p=21360</guid>
		<description><![CDATA[連載： HTML5 Conference 2016 特集 (5)はじめまして。GoogleでChromeの開発をしている保呂毅です。 Chromeの中では特にService Worker周りを担当してまして、最近はNav...]]></description>
				<content:encoded><![CDATA[<div class="seriesmeta">連載： <a href="https://html5experts.jp/series/html5-conf2016/" class="series-403" title="HTML5 Conference 2016 特集" data-wpel-link="internal">HTML5 Conference 2016 特集</a> (5)</div><p>はじめまして。GoogleでChromeの開発をしている保呂毅です。<br>
Chromeの中では特にService Worker周りを担当してまして、最近は<a href="https://crbug.com/649558" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Navigation Preload</a>という新機能をがんばって実装しています。
先日開催された<a href="http://events.html5j.org/conference/2016/9/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">HTML5 Conference 2016</a>でService Worker周辺の最近（ここ1年くらい）の動向に関する発表をさせていただいきました。<br>今回は、この発表の内容を振り返りたいと思います。</p>

<p><img src="/wp-content/uploads/2016/10/DSC06505.jpg" alt="" width="640" height="414" class="alignnone size-full wp-image-21547" srcset="/wp-content/uploads/2016/10/DSC06505.jpg 640w, /wp-content/uploads/2016/10/DSC06505-300x194.jpg 300w, /wp-content/uploads/2016/10/DSC06505-207x134.jpg 207w" sizes="(max-width: 640px) 100vw, 640px" /></p>

<h2>Service Workerとは</h2>

<p>まず本題に入る前に簡単にService Workerの説明します。
Service Workerとはどういうものかと言いますと、
下のコードのようにnavigator.serviceWorker.registerというAPIで登録する、バックグラウンドで動作するJavaScript実行環境です。</p>

<p></p><pre class="crayon-plain-tag">navigator.serviceWorker.register('./sw.js', {scope: './'});</pre><p></p>

<p>登録されたServiceWorkerでは、ページからのネットワークリクエストを横取りすることができます。
下のコードでは、ページからのネットワークリクエストを横取りしてHello Worldという文字列をページに返しています。</p>

<p></p><pre class="crayon-plain-tag">self.addEventListener('fetch', event =&gt; {
  event.respondWith(new Response('Hello World!'));
});</pre><p></p>

<p>また、ページからのネットワークリクエストの横取りをする以外にも、ページを開いていなくても Push通知を受け取って Notification （通知）を表示する、Push Notificationsという機能も備えています。</p>

<p>Service Workerは新しいAPIでして、まだ利用できるブラウザが限られています。
<a href="https://jakearchibald.github.io/isserviceworkerready/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Is ServiceWorker ready?</a>というページに各ブラウザの対応状況が書かれています。ChromeとFirefox、Opera、Samsungのブラウザは対応していて、Edgeは開発中、Safariは検討中となっています。</p>

<h2>Push Notifications</h2>

<h3>Push Notificationsの仕組み</h3>

<p><img src="/wp-content/uploads/2016/10/push_flow0-640x284.png" alt="Push Notificationsの仕組み" width="640" height="284" class="aligncenter size-large wp-image-21408" srcset="/wp-content/uploads/2016/10/push_flow0.png 640w, /wp-content/uploads/2016/10/push_flow0-300x133.png 300w, /wp-content/uploads/2016/10/push_flow0-207x92.png 207w" sizes="(max-width: 640px) 100vw, 640px" /></p>

<p>Push Notificationはどういう仕組みかといいますと、WebサービスのサーバーからFCM (Firebase Cloud Messaging)のサーバーにメッセージをポストすると、それがブラウザに届いて通知を表示できるというものです。FCMは、最近GCM (Google Cloud Messaging)から名前が変わりました。</p>

<h3>Push Event</h3>

<p></p><pre class="crayon-plain-tag">self.addEventListener('push', event =&gt; {  
  event.waitUntil(  
    self.registration.showNotification(
      'Hello',
      {
        body: 'We have received a push message.',
        icon: 'message.png',
        tag: 'tag'
      }));  
});</pre><p></p>

<p>このコードはService Worker上の<code>Push</code>イベントハンドラです。WebサービスのサーバーからFCMのサーバーにPOSTリクエストすると、このイベントハンドラが呼びだされます。このコードでは、<code>ServiceWorkerRegistration</code> の <code>showNotification</code> というAPIで下のような通知を表示しています。
タイトルが&#8217;Hello&#8217;で中身が&#8217;We have received a push message.&#8217;で、アイコンにmessage.pngを指定しています。</p>

<p><img src="/wp-content/uploads/2016/10/push00-300x71.png" alt="push00" width="300" height="71" class="aligncenter size-medium wp-image-21410" srcset="/wp-content/uploads/2016/10/push00-300x71.png 300w, /wp-content/uploads/2016/10/push00.png 640w, /wp-content/uploads/2016/10/push00-207x49.png 207w" sizes="(max-width: 300px) 100vw, 300px" /></p>

<h3>Notificationclick Event</h3>

<p></p><pre class="crayon-plain-tag">self.addEventListener('notificationclick', event =&gt; {
  event.notification.close();
  clients.openWindow('/messages');
});</pre><p> 
このコードは<code>NotificationClick</code>イベントハンドラで、先ほどの通知をクリックした際の処理を書いています。このコードでは、通知を閉じて、messagesというページをopenWindowで開いています。</p>

<h3>Payload Data</h3>

<p>ここまでの機能は2015年4月にリリースされたChrome 42から使えます。
ところが、Chrome 49までは、Pushメッセージにデータを含めることができないという制限がありました。例えばチャットアプリなどでチャットの内容を通知に表示する場合は、Fetch APIを使ってWebサーバーに問い合わせる必要がありました。
それが、今年の4月にリリースされたChrome 50からは、Pushメッセージ自体に暗号化したデータを含めることができるようになりました。</p>

<p></p><pre class="crayon-plain-tag">self.addEventListener('push', event =&gt; {
  if (event.data) {
    console.log(event.data.json());
  }
});</pre><p></p>

<p>Service Worker側ではこちらのコードのようにPushEventにdataというプロパティがついて、データを読めるようになります。
サーバーからどう送信するかの詳細はこちらの<a href="https://developers.google.com/web/updates/2016/03/web-push-encryption" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">URL</a>で説明されているので、興味のある方はぜひ読んでみてください。Node.js用のライブラリ <a href="https://github.com/web-push-libs/web-push" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Web Push library for Node.js</a> もありますので、Node.jsをサーバーで使っている方はこちらを使うといいかと思います。</p>

<h3>Notification Actions</h3>

<p><img src="/wp-content/uploads/2016/10/push_actions-300x109.png" alt="Notification actions" width="300" height="109" class="aligncenter size-medium wp-image-21412" srcset="/wp-content/uploads/2016/10/push_actions-300x109.png 300w, /wp-content/uploads/2016/10/push_actions.png 640w, /wp-content/uploads/2016/10/push_actions-207x75.png 207w" sizes="(max-width: 300px) 100vw, 300px" /></p>

<p>続きまして、Chrome 48から使える機能、Notification Actionsです。この機能を使うと上の絵のように、ボタンを追加できるようになります。
</p><pre class="crayon-plain-tag">self.registration.showNotification(
  'Hello',
  {
    body: 'We have received a push message.',
    icon: 'message.png',
    tag: 'tag',
    data: 1234,
    actions: [
      {action: 'like', title: 'Like', icon: 'like.png'},
      {action: 'reply', title: 'Reply', icon: 'reply.png'}]
  });</pre><p>
コードを見ると、Actionsという項目で指定しているのが分かるかと思います。各ボタンに対応する、actionの文字列と、titleの文字列と、iconの画像を指定しています。</p>

<h3>Notificationclick Event</h3>

<p>このボタンをクリックすると、<code>NotificationClick</code>イベントハンドラが実行され、NotificationClickEventのactionを見ることで、どのボタンがクリックされたのかがわかります。
</p><pre class="crayon-plain-tag">self.addEventListener('notificationclick', event =&gt; { 
  var messageId = event.notification.data;
  event.notification.close();
  if (event.action == 'like') {
    silentlyLikeItem();
  } else if (event.action == 'reply') {
    clients.openWindow('/messages?reply=' + messageId);
  } else {
    clients.openWindow('/messages?reply=' + messageId);
  }
});</pre><p>
このコードの場合、likeがクリックされたら<code>silentlyLikeItem()</code>というメソッドを呼び出しています。Replyがクリックされた場合は、該当するメッセージのページを開いています。ボタン以外の部分がクリックされた場合も同様です。
ちなみに、メッセージを表示するURLのメッセージIDに<code>notification.data</code>の値を使っていますが、こちらはさきほどの通知を表示する際に指定したdataが渡ってきています。
この値は、<code>Push</code>イベントのdataから取得するといいかと思います。</p>

<h3>Standard Web Push Protocol</h3>

<p>Chrome 51までは、Web Pushを実現するためには、以下のような手順が必要でした。この認証方法はFCM独自のプロトコルです。</p>

<ol>
<li>事前にFCMに登録をして<code>gcm_sender_id</code>というIDを取得</li>
<li><code>manifest.json</code>にそのIDを記述</li>
<li>FCMサーバーへのPOSTリクエストの際に<code>Authorization</code>ヘッダにAPIキーを指定する</li>
</ol>

<p>ところが、Chrome 52からVoluntary Application Server Identification (VAPID) という標準化されたプロトコルでサーバー認証することでFCMへの事前登録が必要なくなりました。
詳しくはこちらの<a href="https://developers.google.com/web/updates/2016/07/web-push-interop-wins" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">URL</a>を御覧ください。</p>

<h2>Stream</h2>

<p>サーバーにFetchリクエストをして、帰ってきたレスポンスに手を加えてページに返すService Workerを考えます。</p>


<!-- iframe plugin v.4.3 wordpress.org/plugins/iframe/ -->
<iframe src="https://docs.google.com/presentation/d/1iYCCnASg-xCpwVUB2I--gGsm-eElVMexoRXY1YZTbA4/embed?start=false&amp;loop=true&amp;delayms=3000" frameborder="0" width="640" height="389" allowfullscreen="true" mozallowfullscreen="true" webkitallowfullscreen="true" class="aligncenter" scrolling="yes"></iframe>


<p>Chrome 51までは、このような処理をした場合、上のアニメにあるようにレスポンスを最後まで読んでからでないと、ページに返すことができませんでした。
しかし、今年の７月にリリースされたChrome 52から、Streamの新しい機能を使うことで、サーバーからのレスポンスを逐次処理しながらページに返すことができるようになりました。</p>

<p>具体的な例で解説します。</p>

<p>ページのヘッダとフッタを事前にキャッシュに入れておいて、ページを読み込む際はコンテンツだけをサーバに問い合わせて、Service Worker内で文字列連結してページに返す。ということを考えます。</p>

<p></p><pre class="crayon-plain-tag">self.addEventListener('install', event =&gt; {
    event.waitUntil(
      caches.open('cache_name')
        .then(cache =&gt; cache.addAll(['./header.txt',
                                     './footer.txt'])));
});</pre><p></p>

<p>まず、ヘッダとフッタをキャッシュに入れるには、<code>Install</code>イベントハンドラで上のコードのようにCacheStorage APIを使います。この例では<code>header.txt</code>と<code>footer.txt</code>をキャッシュに保存しています。
<code>caches.open()</code>でキャッシュを開き、取得した<code>cache</code>に対して、<code>cache.addAll()</code>を呼んでヘッダとフッタをサーバーから取ってきて保存しています。</p>

<p></p><pre class="crayon-plain-tag">self.addEventListener('fetch', event =&gt; {
  var url = event.request.url;
  if (!url.endsWith('.html'))
    return;

  event.respondWith(
    Promise.all(
        [caches.match('./header.txt'),
         fetch(url +  '.txt'),
         caches.match('./footer.txt')])
      .then(responseList =&gt; Promise.all(responseList.map(res =&gt; res.text())))
      .then(textList =&gt;
            new Response(textList.join(''),
                         {headers:[['content-type', 'text/html']]})));
});</pre><p></p>

<p>次に、<code>Fetch</code>イベントハンドラで、先ほど保存したヘッダとフッタをキャッシュから取り出して、サーバーから取得したコンテンツと文字列連結します。
先ほど保存した<code>header.txt</code>と<code>footer.txt</code>をキャッシュから読み込みつつ、Fetch APIで元のURLに”.txt”をつなげたURLにネットワークリクエストを投げています。
そして、それらすべてのレスポンスから、text()でコンテンツ本文を取得し、その後、joinで連結しています。
これで目的通り、ヘッダとフッタを連結してページに返すことができるのですが、問題があります。
先ほど書いたとおり、サーバーからのコンテンツ全体を取得完了するまでページに返せないのです。
そこで、解決方法がStreamです。Streamを使えば、サーバーから取得したコンテンツを徐々にページに返していくことができます。</p>

<p></p><pre class="crayon-plain-tag">self.addEventListener('fetch', event =&gt; {
    var url = event.request.url;
    if (!url.endsWith('.html'))
      return;

    var stream = new ReadableStream({
        【次コード参照】
    });

    event.respondWith(new Response(
        stream,
        {headers:[['content-type', 'text/html']]}));
  });</pre><p></p>

<p>具体的にコードを見ていきましょう。
<code>Install</code>イベントハンドラは先ほどと同じです。
<code>Fetch</code> イベントハンドラは上のコードのようになります。
<code>ReadableStream</code>という新しいクラスが登場します。中身は下で説明しますが、ここで<code>ReadableStream</code>をつくって、それを<code>Response</code>オブジェクトを作る際に渡し、<code>respondWith()</code>でレスポンスをページに返しています。</p>

<p></p><pre class="crayon-plain-tag">var stream = new ReadableStream({
start(controller) {
  var promises = [caches.match('./header.txt'),
                  fetch(url + '.txt'),
                  caches.match('./footer.txt')];
  function pushStream(body) {
    var reader = body.getReader();
    return reader.read().then(function proc(result) {
      if (result.done)
        return;
      controller.enqueue(result.value);
      return reader.read().then(proc);
    });
  }
  promises[0]
    .then(response =&gt; pushStream(response.body))
    .then(() =&gt; promises[1]).then(response =&gt; pushStream(response.body))
    .then(() =&gt; promises[2]).then(response =&gt; pushStream(response.body))
    .then(() =&gt; controller.close());
}});</pre><p>
<code>ReadableStream</code>の中身は、このようになります。
<code>start()</code>というメソッドの中でまず、<code>header.txt</code>と<code>footer.txt</code>のキャッシュからの読み込みと、サーバへのFetchリクエストのPromiseを作っています。
下の方で、まず、ひとつ目のPromise、つまりHeader読み込みのPromiseからレスポンスを取得し、そのレスポンスボディを<code>pushStream()</code>という関数に渡しています。
<code>pushStream()</code>では<code>body</code>から<code>reader</code>を取得して、順番に読んでいって、<code>controller.enqueue()</code>で詰め込んでいきます。
それが終わったら２つ目のPromise、つまりFetchのPromiseからレスポンスを取得して、あとは同様にbodyの中身を徐々にenqueueしていきます。その後、３つ目のPromise、つまりfooter.txtのPromiseも同様です。最後に、controller.close()を呼んでStreamを閉じています。</p>

<p>このようにすると、サーバーからのレスポンスを最後まで待たずに、徐々にページにデータを返していくことができます。</p>

<h2>Unified Media Pipeline</h2>

<p>実は、Android Chromeは51まで、audioエレメントやvideoエレメントではAndroidのメディアスタックを利用していました。それが、7月にリリースされたChrome 52からデスクトップ版Chromeと共通化されました。
そのおかげで、</p>

<ul>
<li>Service Workerを使ったキャッシュ</li>
<li>Blob URLからの再生</li>
<li><code>playbackRate</code>の指定</li>
<li><code>MediaRecorder</code>からの<code>MediaStreams</code>をWeb Audioに送信する</li>
</ul>

<p>といったことができるようになり、クロスデバイスの開発が容易になりました。
詳細はこちらの<a href="https://developers.google.com/web/updates/2016/06/ump" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">URL</a>を参照してください。</p>

<h2>Background Sync</h2>

<p>Background Syncはどういうものかと言いますと、その名の通り、バックグラウンドでデータを送受信できる機能です。
例えば、「オフライン時にメッセージを書いて、オンラインになったときに自動で送信」といったことができるようになります。AndroidだとChromeアプリを閉じていても動作します。
この機能はChrome 49 から利用可能です。</p>

<p>ちょうどいい動画がYouTubeにあったので<a href="https://www.youtube.com/watch?v=2ugAVXRkr9U" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">こちら</a>をご覧ください。
この動画は、Service Workerを使ってオフラインでもWikipediaの記事を読むことができるというWebアプリのデモです。このアプリで、キャッシュに入っていない記事を読もうとしたら、エラーになるのですが、Background Syncを使ってオンラインになった時に自動で記事をダウンロードし、ダウンロードが完了したら先ほど紹介したNotificationのAPIで通知を表示しています。</p>

<h2>Foreign Fetch</h2>

<h3>普通の Fetchイベント</h3>

<p><img src="/wp-content/uploads/2016/10/normal_fetch_1-640x320.png" alt="normal_fetch_1" width="640" height="320" class="aligncenter size-large wp-image-21460" srcset="/wp-content/uploads/2016/10/normal_fetch_1.png 640w, /wp-content/uploads/2016/10/normal_fetch_1-300x150.png 300w, /wp-content/uploads/2016/10/normal_fetch_1-207x104.png 207w" sizes="(max-width: 640px) 100vw, 640px" /></p>

<p>Foreign Fetchの説明をする前に通常の<code>Fetch</code>イベントを復習します。
<code>a.com</code>のサイトにService Workerが登録されているとします。
この場合、</p>

<ul>
<li><code>a.com</code>のページ自体のHTMLファイル（メインリソースと言います）へのリクエスト</li>
<li>このページで読み込まれる、<code>a.com</code>上にある画像等のサブリソースへのリクエスト</li>
<li>このページで読み込まれる、<code>b.com</code>上にあるサブリソースへのリクエスト</li>
</ul>

<p>これらはすべて、Service Workerの<code>Fetch</code>イベントハンドラで横取りすることができます。</p>

<p><img src="/wp-content/uploads/2016/10/normal_fetch_2-640x320.png" alt="normal_fetch_2" width="640" height="320" class="aligncenter size-large wp-image-21459" srcset="/wp-content/uploads/2016/10/normal_fetch_2.png 640w, /wp-content/uploads/2016/10/normal_fetch_2-300x150.png 300w, /wp-content/uploads/2016/10/normal_fetch_2-207x104.png 207w" sizes="(max-width: 640px) 100vw, 640px" /></p>

<p>しかし、例えば、<code>b.com</code>のサイトのページから<code>a.com</code>上にある画像などのサブリソースへのリクエストは<code>Fetch</code>イベントハンドラで横取りすることができません。</p>

<h3>Foreign Fetchを使うと</h3>

<p><img src="/wp-content/uploads/2016/10/foreign_fetch-640x320.png" alt="foreign_fetch" width="640" height="320" class="aligncenter size-large wp-image-21458" srcset="/wp-content/uploads/2016/10/foreign_fetch.png 640w, /wp-content/uploads/2016/10/foreign_fetch-300x150.png 300w, /wp-content/uploads/2016/10/foreign_fetch-207x104.png 207w" sizes="(max-width: 640px) 100vw, 640px" /></p>

<p>これを可能にするのがForeign Fetchです。
Foreign Fetchを使うと、他のサイト(この図の場合は<code>b.com</code>)からの自分のサイト(この図の場合は<code>a.com</code>)へのサブリソースのリクエストを<code>a.com</code>のService Workerで横取りすることができるようになります。</p>

<h3>普通のFetchイベント vs Foreign Fetchイベント</h3>

<p>まとめますとこうなります。
「ネットワークリクエストを横取りする」という点は共通です。
相違点としては、
普通の<code>Fetch</code>イベントは自分のサイトのページからのネットワークリクエストを横取りします。自分のサイトのHTMLページと、そこからの画像などのサブリソースリクエストも横取りできます。</p>

<p>それに対して<code>ForeignFetch</code>イベントでは、他のサイトから自分のサイトへのサブリソースのリクエストを横取りすることができるようになります。Web FontサーバやCDN等で活用できるのではと考えられます。</p>

<h3>Foreign Fetch の登録方法</h3>

<p>具体的にコードを見ていきましょう
</p><pre class="crayon-plain-tag">self.addEventListener('install', event =&gt; {
    event.registerForeignFetch({
        scopes: ['/myscope/'],
        origins: ['https://b.com/']
      });
  });</pre><p>
Foreign Fetchの登録方法ですが、先程の例のように、<code>a.com</code>上のService Workerが、 <code>b.com</code> からの <code>a.com/myscope/</code> 以下へのリクエストを横取りしたい場合は、<code>Install</code>イベントハンドラで上のコードのように<code>registerForeignFetch()</code>というAPIを叩きます。
<code>scopes</code>に横取りしたいリクエストの範囲、<code>origins</code>にリクエスト元のオリジンを指定します。すべてのオリジンからのリクエストを横取りしたい場合は、<code>origins</code>にアスタリスクを指定します。</p>

<p>上のコードのようにForeign Fetchを登録すると、 <code>a.com/myscope/</code> 以下へのネットワークリクエストの度に<code>ForeignFetch</code>イベントハンドラが実行されるようになります。</p>

<h3>Foreign Fetch Event</h3>

<p><code>ForeignFetch</code>イベントハンドラでは普通の<code>Fetch</code>イベントと同じように<code>respondWith()</code>でページにレスポンスを返すことができるのですが、普通の<code>Fetch</code>イベントと違って、辞書に包む（以下のコードの<code>{response: res}</code>）必要があります。
</p><pre class="crayon-plain-tag">self.addEventListener('foreignfetch', event =&gt; {
    event.respondWith(
        fetch(event.request).then(res =&gt; ({response: res}));
  });</pre><p>
注意点として、このようにしてページに返したレスポンスはページではOpaqueになります。つまり、&lt;img&gt;タグで画面に表示はできるのですが、JavaScriptから中身を読むことができません。
これを読めるようにするにはCORSの設定が必要になります。</p>

<h3>CORS (Cross-Origin Resource Sharing)</h3>

<p></p><pre class="crayon-plain-tag">self.addEventListener('foreignfetch', event =&gt; {
  event.respondWith(
     fetch(event.request)
       .then(response =&gt;({response: response,
                          origin: event.origin,
                          headers: ['...']})));
});</pre><p>
具体的にはこのように、<code>respondWith()</code>に渡す辞書でページの<code>origin</code>を指定します。<code>headers</code>を指定すると、指定したHTTPヘッダもページ側で読めるようになります。</p>

<p>このForeign Fetchですが、まだ実験中の機能でして、試す場合は、Chrome 54以降でchrome://flagsのenable-experimental-web-platform-featuresを有効にしてください。</p>

<h2>Header-based Installation</h2>

<p>今まで、Service Workerを登録する場合、JavaScriptで<code>navigator.serviceWorker.register()</code>を呼ぶ必要がありました。
</p><pre class="crayon-plain-tag">navigator.serviceWorker.register('/sw.js', {scope: '/'});</pre><p></p>

<p>Header-based Installationを使うと，これが、下のようなHTTPのLink Headerで登録したり、
</p><pre class="crayon-plain-tag">Link: &lt;/sw.js&gt;; rel=serviceworker; scope=/</pre><p> 
下のような、HTMLのLink Elementで登録できるようになります。
</p><pre class="crayon-plain-tag">&lt;link rel="serviceworker" scope="/" href="/sw.js"&gt;</pre><p></p>

<p>ここでポイントとなるのが、サブリソースのリクエストに対するHTTPレスポンスのLink HeaderでもService Workerをインストールできるということです。この機能を使うと、Iframe等でHTMLを読み込ませる必要なく、Foreign FetchのService Workerをインストールできます。CDNなどでサブリソースをサーブしている場合でもCDNのドメインに対してForeign FetchのService Workerを登録できるようになるのです。</p>

<p>この機能もForeign Fetchと同様に実験中の機能でして、試す場合は、Chrome 54以降でchrome://flagsのenable-experimental-web-platform-featuresを有効にしてください。</p>

<h2>Origin Trials</h2>

<p>Origin TrialとはChromeの実験的な新しいAPIを、申請のあった特定のオリジン(ドメイン)で一定期間だけ使ってもらってフィードバックをもらう仕組みです。
いま(2016年10月現在)はPersistent Storage, Web Bluetooth, Web USB, Foreign Fetch (Header-based Installation含む)が対象になっています。</p>

<p>申請の方法など詳しくはこちらの<a href="https://github.com/jpchase/OriginTrials/blob/gh-pages/developer-guide.md" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">URL</a>を参照してください</p>

<h2>AppCache</h2>

<p>Application Cache (通称AppCache)という、Webサイトのオフライン対応のためにかつて提案された機能があります。</p>

<p>しかし、FirefoxはApplication Cacheの非サポート化を<a href="https://www.fxsitecompat.com/en-CA/docs/2016/application-cache-support-will-be-removed/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">明言</a>してまして、Chromeも非セキュアなコンテクストではAppCacheを<a href="https://www.chromestatus.com/feature/5714236168732672" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">サポートしなくする</a>としています。
ですので、Webサイトのオフライン対応はAppCacheではなく、Service Workerを使おうという流れにあります。</p>

<p>Service Workerに移行する際の参考として紹介したいのが、sw-appcache-behaviorというAppCacheの動作を Service Workerを使ってシミュレーションするライブラリです。
もし興味がありましたら、この<a href="https://github.com/GoogleChrome/sw-helpers/tree/master/projects/sw-appcache-behavior" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">URL</a>を参照してください。</p>

<h2>DevTools</h2>

<p>DevToolsは進化が早くてよくUIが変わるのですが、今は、”Application”というタブの中にService Workersというのがありまして、こちらでService Workerの状態を確認したり、デバッグしたりすることができます。</p>

<p><img src="/wp-content/uploads/2016/10/devtools-640x360.png" alt="devtools" width="640" height="360" class="aligncenter size-large wp-image-21465" srcset="/wp-content/uploads/2016/10/devtools.png 640w, /wp-content/uploads/2016/10/devtools-300x169.png 300w, /wp-content/uploads/2016/10/devtools-207x116.png 207w" sizes="(max-width: 640px) 100vw, 640px" /></p>

<p>この画面で注目していただきたいのが、このメッセージです。
<code>Service Worker termination by a timeout timer was canceled because DevTools is attached.</code>
このメッセージは最近表示するようにしたのですが、「DevToolsがアタッチされてるので、タイムアウトによるService Workerの停止がキャンセルされました」と書かれてます。</p>

<p>Service Workerは一定期間イベントが実行されないと停止し、次回イベントで再起動するようになっています。ただし、DevToolsを開いている間は停止しません。
つまり、通常の状態だと、しばらくほっておくと勝手に停止し、FetchEventなどのイベントハンドラを呼ぶ必要が出ると再起動します。
ですので、Globalスコープに変数を置いて保存していると、再起動時に消えるので、Service Workerのコードを書く際は気をつけてください。</p>

<h2>まとめ</h2>

<p>駆け足になりましたが、Service Worker周辺の最近（ここ１年くらい）の動向について説明いたしました。下に関連リンクを貼り付けていますので、もっと詳細を確認したい方はご参照ください。</p>

<h2>関連リンク集</h2>

<h3>Introduction to Service Worker</h3>

<ul>
<li><a href="https://developers.google.com/web/fundamentals/primers/service-worker/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://developers.google.com/web/fundamentals/primers/service-worker/</a></li>
</ul>

<h3>Push</h3>

<ul>
<li><a href="https://developers.google.com/web/updates/2015/03/push-notifications-on-the-open-web" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://developers.google.com/web/updates/2015/03/push-notifications-on-the-open-web</a></li>
<li><a href="https://developers.google.com/web/updates/2016/01/notification-actions" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://developers.google.com/web/updates/2016/01/notification-actions</a></li>
<li><a href="https://developers.google.com/web/updates/2016/03/web-push-encryption" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://developers.google.com/web/updates/2016/03/web-push-encryption</a></li>
<li><a href="https://developers.google.com/web/updates/2016/03/notifications" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://developers.google.com/web/updates/2016/03/notifications</a></li>
<li><a href="https://developers.google.com/web/updates/2016/07/web-push-interop-wins" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://developers.google.com/web/updates/2016/07/web-push-interop-wins</a></li>
<li><a href="https://github.com/web-push-libs/web-push/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://github.com/web-push-libs/web-push/</a></li>
<li><a href="https://developers.google.com/cloud-messaging/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://developers.google.com/cloud-messaging/</a></li>
</ul>

<h3>Stream</h3>

<ul>
<li><a href="https://developers.google.com/web/updates/2016/06/sw-readablestreams" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://developers.google.com/web/updates/2016/06/sw-readablestreams</a></li>
<li><a href="https://www.youtube.com/watch?v=Pii-LaWOyuo" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://www.youtube.com/watch?v=Pii-LaWOyuo</a></li>
<li><a href="https://jakearchibald.com/2016/streams-ftw/#creating-your-own-readable-stream" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://jakearchibald.com/2016/streams-ftw/#creating-your-own-readable-stream</a></li>
</ul>

<h3>Unified Media Pipeline</h3>

<ul>
<li><a href="https://developers.google.com/web/updates/2016/06/ump" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://developers.google.com/web/updates/2016/06/ump</a></li>
</ul>

<h3>Background Sync</h3>

<ul>
<li><a href="https://developers.google.com/web/updates/2015/12/background-sync" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://developers.google.com/web/updates/2015/12/background-sync</a></li>
<li><a href="https://github.com/WICG/BackgroundSync/blob/master/explainer.md" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://github.com/WICG/BackgroundSync/blob/master/explainer.md</a></li>
<li><a href="https://ponyfoo.com/articles/backgroundsync" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://ponyfoo.com/articles/backgroundsync</a></li>
</ul>

<h3>Foreign Fetch</h3>

<ul>
<li><a href="https://github.com/slightlyoff/ServiceWorker/blob/master/foreign_fetch_explainer.md" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://github.com/slightlyoff/ServiceWorker/blob/master/foreign_fetch_explainer.md</a></li>
<li><a href="https://slightlyoff.github.io/ServiceWorker/spec/service_worker/#on-foreign-fetch-request-algorithm" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://slightlyoff.github.io/ServiceWorker/spec/service_worker/#on-foreign-fetch-request-algorithm</a></li>
<li><a href="https://crbug.com/540509" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://crbug.com/540509</a></li>
<li><a href="https://github.com/slightlyoff/ServiceWorker/pull/751" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://github.com/slightlyoff/ServiceWorker/pull/751</a></li>
</ul>

<h3>Header-based installation</h3>

<ul>
<li><a href="https://crbug.com/582310" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://crbug.com/582310</a></li>
<li><a href="https://github.com/slightlyoff/ServiceWorker/issues/685#issuecomment-176473764" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://github.com/slightlyoff/ServiceWorker/issues/685#issuecomment-176473764</a></li>
</ul>

<h3>Origin Trials</h3>

<ul>
<li><a href="https://github.com/jpchase/OriginTrials/blob/gh-pages/developer-guide.md" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://github.com/jpchase/OriginTrials/blob/gh-pages/developer-guide.md</a></li>
</ul>

<h3>Web Bluetooth</h3>

<ul>
<li><a href="https://developers.google.com/web/updates/2015/07/interact-with-ble-devices-on-the-web?hl=en" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://developers.google.com/web/updates/2015/07/interact-with-ble-devices-on-the-web?hl=en</a></li>
</ul>

<h3>WebUSB</h3>

<ul>
<li><a href="https://developers.google.com/web/updates/2016/03/access-usb-devices-on-the-web" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://developers.google.com/web/updates/2016/03/access-usb-devices-on-the-web</a></li>
</ul>

<p>当日の講演資料と動画は下記で公開されていますので、こちらも参照してください。</p>

<ul>
<li><a href="https://docs.google.com/presentation/d/19x3yi7Jn-6In5igGYfEiK0tBfNI290BAclT0AiqDj4Q/pub?slide=id.p" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">講演資料</a></li>
</ul>


<!-- iframe plugin v.4.3 wordpress.org/plugins/iframe/ -->
<iframe width="560" height="315" src="https://www.youtube.com/embed/MjA3XIT-IH4" frameborder="0" 0="allowfullscreen" scrolling="yes" class="iframe-class"></iframe>

]]></content:encoded>
		
		<series:name><![CDATA[HTML5 Conference 2016 特集]]></series:name>
	</item>
		<item>
		<title>プログレッシブウェブアプリ詳解 ─ 過去・現在・未来</title>
		<link>/agektmr/20527/</link>
		<pubDate>Wed, 14 Sep 2016 00:00:03 +0000</pubDate>
		<dc:creator><![CDATA[えーじ]]></dc:creator>
				<category><![CDATA[最新動向]]></category>
		<category><![CDATA[プログラミング]]></category>
		<category><![CDATA[システム開発]]></category>
		<category><![CDATA[Progressive Web Apps]]></category>
		<category><![CDATA[Service Worker]]></category>

		<guid isPermaLink="false">/?p=20527</guid>
		<description><![CDATA[連載： Web技術でアプリ開発2016 (5)Web技術でアプリ開発2016特集・第5弾は、プログレッシブウェブアプリ (Progressive Web Apps)をご紹介させていただきます。 はじめに プログレッシブウ...]]></description>
				<content:encoded><![CDATA[<div class="seriesmeta">連載： <a href="https://html5experts.jp/series/web-based-apps-2016/" class="series-391" title="Web技術でアプリ開発2016" data-wpel-link="internal">Web技術でアプリ開発2016</a> (5)</div><p>Web技術でアプリ開発2016特集・第5弾は、プログレッシブウェブアプリ (Progressive Web Apps)をご紹介させていただきます。</p>

<p><style>
.youtube {
    height: 0;
    position: relative;
    padding-bottom: 56.25%;
    overflow: hidden;
    margin-bottom: 24px;
}
.youtube > iframe {
    width: 100%;
    height: 100%;
    position: absolute;
    top: 0;
    left: 0;
}
</style></p>

<h2>はじめに</h2>

<p>プログレッシブウェブアプリ(Progressive Web Apps)という言葉が初めて登場したのは2015年8月のAlex Russellによる記事<a href="https://medium.com/@slightlylate/progressive-apps-escaping-tabs-without-losing-our-soul-3b93a8561955" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Progressive Web Apps: Escaping Tabs Without Losing Our Soul</a>です。当時オフラインやプッシュ通知など、モバイルウェブを飛躍的に進化させる画期的な機能が次々と追加されていた状況において、このムーブメントを呼称するための言葉が求められていました。Google社内でいくつもの候補が挙げられましたが、Service Workerの発案者としてAlex Russellが推したのが、この「プログレッシブウェブアプリ」でした。</p>

<p>また、当初Googleで始まったプログレッシブウェブアプリの動きではありましたが、先日の<a href="https://www.youtube.com/playlist?list=PLNYkxOF6rcIAWWNR_Q6eLPhsyx6VvYjVb" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Progressive Web App Dev Summit</a>ではMicrosoft、Mozilla、Opera、Samsungの代表者も登壇するなど、各ブラウザベンダーとも足並みが揃いつつあります。これはもはやウェブ全体の動きと言うことができます。</p>

<p>若干古い情報になりますが、以降この (長い) 記事を読む以外に手早く情報を得るための方法として、下記を挙げておきます。</p>

<h3>スマートフォン体験を一歩先へ &#8211; プログレッシブウェブアプリの作り方</h3>

<div class="youtube">

<!-- iframe plugin v.4.3 wordpress.org/plugins/iframe/ -->
<iframe style="width: 100%;
    height: 100%;
    position: absolute;
    top: 0;
    left: 0;" src="https://www.youtube.com/embed/VHN2wJWi0WE?controls=2&amp;modestbranding=1&amp;showinfo=0&amp;utm-source=crdev-wf" class="devsite-embedded-youtube-video" allowfullscreen data-video-id="hmqZxP6iTpo" data-autohide="1" data-modestbranding="1" data-controls="2" data-utm-source="crdev-wf" data-showinfo="0" frameborder="0" width="100%" height="500" scrolling="yes"></iframe>

</div>

<p><a href="https://www.youtube.com/watch?v=VHN2wJWi0WE" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">講演動画</a></p>

<div class="youtube">

<!-- iframe plugin v.4.3 wordpress.org/plugins/iframe/ -->
<iframe src="https://docs.google.com/presentation/d/1VcXsKDaCUpf2SS35WNcrKslkK6PcXxWsnhcKiLfWCXs/embed?start=false&amp;loop=false" width="100%" height="500" scrolling="yes" class="iframe-class" frameborder="0"></iframe>

</div>

<p><a href="https://docs.google.com/presentation/d/1VcXsKDaCUpf2SS35WNcrKslkK6PcXxWsnhcKiLfWCXs/pub?start=false&amp;loop=false&amp;delayms=3000" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">プレゼン資料</a></p>

<h2>プログレッシブウェブアプリとは</h2>

<p>プログレッシブウェブアプリとは、最新の標準ウェブ技術を用いることで、オフラインやプッシュ通知といったこれまでよりも一次元上のユーザー体験をもたらすウェブアプリのことです。必ずしもデスクトップのウェブアプリや、ページ遷移を使ったサーバーサイドでレンダリングされるアプリを除外するものではありません。</p>

<p>理屈で説明するよりも、実際に体験した方が早いかと思います。いくつか具体例を挙げましょう。</p>

<ul>
<li><a href="https://www.flipkart.com/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Flipkart</a></li>
<li><a href="http://www.washingtonpost.com/pwa/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Washington Post</a></li>
<li><a href="https://m.aliexpress.com/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">AliExpress</a></li>
<li><a href="http://smp.suumo.jp/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">suumo</a></li>
</ul>

<p>※ 他にも様々なプログレッシブウェブアプリが既に存在していますが、<a href="https://pwa.rocks/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">pwa.rocks</a>というサイトにいくつかまとまっていますので、ぜひご覧頂下さい。</p>

<p>実際に使ってみるとわかりますが、これらのアプリケーションにおいてこれまでにない特徴的な点がいくつかあります。</p>

<ul>
<li>ホーム画面に追加してワンタップで起動</li>
<li>キャッシュを使ってオフライン、もしくは高速に動作</li>
<li>プッシュ通知を送信</li>
<li>自動でログイン</li>
</ul>

<h2>プログレッシブとは何か</h2>

<p>「プログレッシブ」という言葉自体は直訳すると「進歩的な」といった意味がありますが、ピンとこない方は多いと思います。</p>

<p>プログレッシブウェブアプリの「プログレッシブ」のひとつの意味は、ウェブページとして使い始めたサービスが、ホーム画面に追加し、オフラインで動作し、プッシュ通知を送り始めるといった流れの中で、「徐々にアプリへと変化していく」様を指しています。</p>

<p>もう一つ意味があります。</p>

<p>フロントエンドエンジニアであれば、プログレッシブ・エンハンスメントという言葉を聞いたことがあると思います。プログレッシブ・エンハンスメントとは、HTML5 時代に続々と新しい機能が登場する中で、最新の標準技術をまだ盛り込んでいないブラウザでも、情報を届けるといった目的をゴールに、必要最低限の機能をベースとして、可能であれば最新技術を使うことでよりリッチな体験を提供する、という設計コンセプトのことを言いました。(逆に、リッチな体験を提供できるブラウザをベースに、最新の技術が使えないブラウザでも最低限の目的は達成できるように設計する手法をグレースフル・デグラデーションと呼びました)</p>

<p>プログレッシブウェブアプリも同様に捉えることができます。つまり、必要最低限の機能を提供できる状態をベースとし、オフラインやプッシュ通知などをプログレッシブな (進歩的な) 機能として提供する、というものです。</p>

<p>これは言い方を変えれば、対応していないブラウザがあるからといって対応を待つべきではなく、それを前提としてウェブを前に進めていこう、というメッセージでもあるのです。実際、プログレッシブウェブアプリ対応への副作用として、複数の機能に未対応のブラウザでも<a href="https://developers.google.com/web/showcase/2016/aliexpress" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">コンバージョンレートが改善したという事例</a>も出ています。</p>

<h2>アプリとウェブのギャップ</h2>

<p>Web2.0の時代を経てコンピューターの世界はネイティブアプリケーションからウェブアプリケーションの時代へと変化を遂げました。地図アプリケーションを思い出して下さい。その昔CD-ROMからせっせとインストールする必要のあった地図が、URLひとつでブラウザから見られるようになったのです。他にも例を挙げればキリがありません。</p>

<p>ウェブはその後もHTML5やその周辺技術の発展により大きく進化していきました。しかしその恩恵を満足の行く形で受けられたのはデスクトップでの話。モバイル・スマートフォンの時代となり、ユースケースが変化していく中で、スマートフォン上で快適に目的を達成するためには、ウェブブラウザは力不足と言われるようになってしまいました。その象徴的な出来事の一つがFacebookのCEOマーク・ザッカーバーグによるこの発言でした。</p>

<p><a href="http://jp.techcrunch.com/2012/09/14/20120911mark-zuckerberg-our-biggest-mistake-with-mobile-was-betting-too-much-on-html5/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Disrupt：ザッカーバーグ、「HTML5を過大評価したのは、われわれ最大の失敗」と認める | TechCrunch Japan</a></p>

<p>これはひとえに標準化を必要とするウェブという環境の特性ゆえと言わざるを得ません。標準化を進めつつ実装し、浸透を待たなければならなかったウェブと異なり、ネイティブアプリはモバイルに最適化した仕様や機能を次々と取り込み、技術革新を起こしていきました。</p>

<p>しかしウェブには、ネイティブアプリでは実現できない特性も持ち合わせています。それが<strong>SLICE</strong>と呼ばれるものです。</p>

<ul>
<li>Secure (安全)</li>
<li>Linkable (リンク可能)</li>
<li>Indexable (インデックス可能)</li>
<li>Composable (再構成可能)</li>
<li>Ephemeral (一時的な利用)</li>
</ul>

<p>ウェブもネイティブも、お互いの短所を埋める進化を遂げていく中で、ウェブが着実に前進していることの現れが、プログレッシブウェブアプリと言えるでしょう。一度はHTML5化を断念したFacebookがプッシュ通知を採用したのはその最たる例です。</p>

<p><a href="http://jp.techcrunch.com/2015/09/15/20150914facechrome/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Facebook、Googleと協力してモバイルウェブユーザーにプッシュ通知を送信</a></p>

<p>プログレッシブウェブアプリはWeb 2.0、HTML5と続いてきた大きな流れの、次の一歩と捉えて間違いありません。</p>

<h2>プログレッシブウェブアプリを構成する機能</h2>

<p>プログレッシブウェブアプリはそれ自体が機能を表しているわけではなく、様々な先進的機能をまとめたひとつのムーブメントです。具体的にどんな機能があるのか、ひとつひとつ見ていきましょう。</p>

<ul>
<li><a href="#pw-https" data-wpel-link="internal">HTTPS</a></li>
<li><a href="#pw-manifest" data-wpel-link="internal">Web App Manifest</a></li>
<li><a href="#pw-sw" data-wpel-link="internal">Service Worker</a></li>
<li><a href="#pw-cache" data-wpel-link="internal">Cache API</a></li>
<li><a href="#pw-push" data-wpel-link="internal">Web Push / Push Notifications</a></li>
<li><a href="#pw-bg" data-wpel-link="internal">Background Sync</a></li>
<li><a href="#pw-budget" data-wpel-link="internal">Web Budget API</a></li>
<li><a href="#pw-credential" data-wpel-link="internal">Credential Management API</a></li>
<li><a href="#pw-payment" data-wpel-link="internal">Payment Request API</a></li>
</ul>

<h3 id="pw-https">HTTPS</h3>

<p>プログレッシブウェブアプリに対応する上で大前提になるのがサイトの完全なHTTPS対応です。ログイン画面や商品購入画面だけではなく、すべてのページをHTTPS化する必要があります。</p>

<p>一般的に、HTTPSに対応することでウェブサイトは</p>

<ul>
<li>ウェブサイトの一意性を守るアイデンティティ(Identity)</li>
<li>サーバー・ブラウザ間の通信を秘匿するコンフィデンシャリティ(Confidentiality)</li>
<li>送受信されるデータを改ざんさせないインテグリティ(Integrity)</li>
</ul>

<p>の3つを手に入れることができます。</p>

<p>プログレッシブウェブアプリ対応において重要なのが、以下でご紹介する機能のほとんどがHTTPS上でのみ動作 (<a href="https://www.w3.org/TR/secure-contexts/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">セキュアコンテキスト</a>)するという点です。こういった機能はPowerful Features と呼ばれ、以前はHTTP上でも動作したgetUserMediaやgeolocationといったAPIもそこに加えられ、最近HTTPSのみで動作する変更が加えられました。</p>

<p>HTTPSに対応することなしにプログレッシブウェブアプリを実現することは不可能ですので、対応を検討する方はここからスタートして下さい。(ちなみにChromeでは、localhostとfile://もセキュアコンテキストに含まれますので、開発時は少し役に立つかもしれません)</p>

<div class="youtube">

<!-- iframe plugin v.4.3 wordpress.org/plugins/iframe/ -->
<iframe src="https://www.youtube.com/embed/YMfW1bfyGSY" frameborder="0" 0="allowfullscreen" width="100%" height="500" scrolling="yes" class="iframe-class"></iframe>

</div>

<h3 id="pw-manifest">Web App Manifest</h3>

<p>ネイティブアプリとウェブサイトを比較した時に、最も大きく異なるのがアクセスのしやすさです。もちろん、アプリのようにインストールしなくても、リンクを踏むだけでアクセスできるという意味で、ウェブは手に届きやすい存在です。しかしスマートフォンの場合、デスクトップのようにキーボードで手軽に検索ができるわけではありません。ホーム画面からタップひとつで起動できるだけで、ユーザーにとっての身近さは大きく改善します。</p>

<p><img src="/wp-content/uploads/2016/09/Screenshot_20160901-153533-169x300.png" alt="screenshot_20160901-153533" style="width: 169px; height: 300px;" class="aligncenter size-medium wp-image-20903" /></p>

<p>そこで登場するのがWeb App Manifestです。Web App Manifestを使うことで、該当ページに関する様々な情報を定義することができます。</p>

<p><a href="https://developer.mozilla.org/ja/docs/Web/Manifest" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Web App Manifest | MDN</a></p>

<p>最近は<code>lang</code>や<code>scope</code>、<code>screenshots</code>といった項目も新しく追加されていますので、その辺りも注目して下さい。</p>

<p>これらの情報を踏まえ、「ホーム画面に追加」をすると、ウェブアプリはホーム画面に鎮座することができるようになります。しかし、この機能がiOSでも古くからある<code>meta</code>タグを使ったものに近いと思われた方もいるかもしれません。</p>

<p>Chromeの場合、このWeb App Manifestといくつかの条件が加わることで、ブラウザが自動的にホーム画面に追加を促してくれる、という点が大きく異なります。詳しくは下記の記事を参照して下さい。</p>

<p><a href="http://qiita.com/horo/items/ff665e4a6613e7684f8f" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Web App ManifestでWebアプリをインストール可能に &#8211; Qiita</a></p>

<h3 id="pw-sw">Service Worker</h3>

<p>Service Workerはプログレッシブウェブアプリを構成する機能の中でも、コンセプトを成立させた一番の立役者で、中核を成すものです。</p>

<p>従来のウェブにおいて、ページはサーバーから送られたHTMLとそれに付随するリソースで構成される機能がすべてであり、ページが閉じられると同時に忘れ去られる存在でした。Service Workerは、ページとは独立して動作するスクリプトを、ページを閉じた後でもブラウザ上で動かし続けることができる機能です。このService Workerを活用することで、様々な可能性が開かれます。</p>

<p><a href="https://html5experts.jp/wp-content/uploads/2016/08/serviceworker.png" data-wpel-link="internal"><img src="/wp-content/uploads/2016/08/serviceworker-640x99.png" alt="serviceworker" width="640" height="99" class="alignnone size-large wp-image-20543" srcset="/wp-content/uploads/2016/08/serviceworker.png 640w, /wp-content/uploads/2016/08/serviceworker-300x46.png 300w, /wp-content/uploads/2016/08/serviceworker-207x32.png 207w" sizes="(max-width: 640px) 100vw, 640px" /></a></p>

<p><strong>オフライン</strong>：Service Workerはページが送信するあらゆるリクエストを横取りすることができます。実質的なローカルプロキシーとなるため、強力なキャッシュを活用すれば、ネットワークに接続していない状態でもレスポンスを返すことができるようになります。詳しくはCache APIの項で説明します。</p>

<p><strong>プッシュ通知</strong>：Service Workerはページが閉じた状態でも、また、ブラウザによっては起動してない状態であっても、イベントを検知することができます。そのため、外部サーバーが送信したプッシュリクエストをイベントとして受け取り、何かしらのアクションを起こすことができるのです。詳しくはWeb Push / Web Notificationsの項で説明します。</p>

<p>それでは、Service Workerの使い方を雰囲気だけ見てみましょう。</p>

<p>Service Workerは独立したJavaScriptとして動作するため、ファイルも分けて記述する必要があります。例えばページ上で動作するスクリプト<code>app.js</code>が、<code>service-worker.js</code>というService Workerを使いたいとします。</p>

<p></p><pre class="crayon-plain-tag">navigator.serviceWorker.register('service-worker.js', {scope: './'});</pre><p></p>

<p>これだけでService Workerが登録されます。簡単ですね。</p>

<p><code>service-worker.js</code>の中身はイベント・ドリブンのスクリプトになります。例えば下記のように記述しておけば、あらゆるリクエストをService Workerが拾い上げて、サーバーにリクエストを転送することなく&#8221;Hello World!&#8221;とレスポンスを返します。</p>

<p></p><pre class="crayon-plain-tag">self.addEventListener('fetch', function(event) {  
  event.respondWith('Hello World!');  
});</pre><p></p>

<p>Service Workerは実際いつ動作を開始するのでしょう？影響範囲はどこまででしょう？こういった細かい話になるとセキュリティへの懸念も出てくるため、若干複雑な内容を理解する必要が出てきます。scopeとライフサイクルについては抑えておきましょう。</p>

<h4>scope</h4>

<p>Service Workerが登録されると、以後半永久的にスクリプトを動かし続けることになります。仮にService Workerがドメインの範囲内すべてに影響をおよぼすことができるとすると、ブログなどのユーザーの権限がパスで区切られたサービスでは、自分の預かり知らないスクリプトが埋め込まれてしまう可能性がでてきます。そこでService Workerでは、scopeという概念を使って影響範囲を制限します。詳しくはこちらの記事が参考になります。</p>

<p><a href="http://qiita.com/nhiroki/items/eb16b802101153352bba" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">ServiceWorkerのスコープとページコントロールについて &#8211; Qiita</a></p>

<h4>ライフサイクル</h4>

<p>Service Workerが登録されたあと、スクリプトを手動で更新したい場合もあります。新しいスクリプトはすぐに有効になるのでしょうか？その場合古いスクリプトはどうなってしまうのでしょうか？Service Workerは放っておいたら永久に更新されないのでしょうか？</p>

<p>Service Workerのライフサイクルを正しく理解することはとても重要です。下記の記事をぜひご覧ください。</p>

<ul>
<li><a href="http://www.html5rocks.com/ja/tutorials/service-worker/introduction/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Service Workerの紹介: Service Workerの使い方 &#8211; HTML5 Rocks</a> (一部古い内容が含まれます)  </li>
<li><a href="https://blog.jxck.io/entries/2016-04-24/service-worker-tutorial.html" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">中級者向け Service Worker Tutorial | blog.jxck.io</a>  </li>
<li><a href="http://blog.nhiroki.jp/2015/07/05/service-worker-registration" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Service WorkerのRegistration</a>  </li>
</ul>

<h4>デバッグ</h4>

<p>Chromeではプログレッシブウェブアプリのデバッグをしやすくするために、最近DevToolsにResourcesパネルの代わりに Applicationパネルが追加されました。Manifestや各種ストレージなどを、ひとつのパネルからまとめて確認することができます。中でもService Worker専用の項目では、現在起動中のスクリプトだけでなく</p>

<ul>
<li>Offline &#8211; オフラインのエミュレーションボタン</li>
<li>Update on reload &#8211; リロードするだけでService Workerを更新</li>
<li>Bypass for network &#8211; ネットワークリクエストの横取りを一時的に停止</li>
</ul>

<p>といったオプションが利用できます。ここからService Workerの単独のDevToolsを開いてデバッグを行ってください。</p>

<p><a href="https://html5experts.jp/wp-content/uploads/2016/08/devtools.png" data-wpel-link="internal"><img src="/wp-content/uploads/2016/08/devtools-640x238.png" alt="devtools" width="640" height="238" class="alignnone size-large wp-image-20542" srcset="/wp-content/uploads/2016/08/devtools.png 640w, /wp-content/uploads/2016/08/devtools-300x112.png 300w, /wp-content/uploads/2016/08/devtools-207x77.png 207w" sizes="(max-width: 640px) 100vw, 640px" /></a></p>

<h4>少し未来の話</h4>

<p>今後利用可能になるService Worker周辺の重要な機能についても少し触れておきましょう。</p>

<p><strong>Link rel=serviceworker</strong>
Chrome 54からOrigin Trial (特定のドメインに実験的にホワイトリストで機能を提供) ベースでService Workerの登録が linkタグやHTTPヘッダーで行えるようになりました。将来的にこれが利用できるようになれば、JavaScriptを記述することなく、linkタグやHTTPヘッダーのみでService Workerを登録できるようになります。</p>

<p></p><pre class="crayon-plain-tag">&lt;link rel="serviceworker" href="/js/sw.js" scope="/"&gt;</pre><p></p>

<p><a href="https://www.chromestatus.com/feature/5682681044008960" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://www.chromestatus.com/feature/5682681044008960</a></p>

<p><strong>Foreign fetch</strong>
また、Web Fontsのような複数ドメインから共有されるリソースをCDN経由で提供するサーバーのリソースも、現在は利用する側のService Workerがハンドリングすることが可能ですが、これをCDN側のService Workerがハンドリングできれば効率が向上します。これを可能にする機能はForeign Fetchと呼ばれ、Chrome 54からOrigin Trialベースで提供が開始されています。</p>

<p><a href="https://www.chromestatus.com/feature/5684130679357440" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://www.chromestatus.com/feature/5684130679357440</a></p>

<p>その他、Service Workerのさらに詳しい情報はHTML5 Conference 2016で行われたChromeエンジニアの保呂さんのスライドが参考になります。<br />
<a href="https://goo.gl/YxL2L7" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Service Worker Deep Dive</a></p>

<h3 id="pw-cache">Cache API</h3>

<p>先程説明したように、Service Workerを使ってあらゆるリクエストを横取りすることで、オフラインの機能が実現可能になります。そこで強力なキャッシュ機能を提供するのがCache APIです。</p>

<p>Cache APIは取得されたリソースだけでなく、リクエストやレスポンスの内容も含めて保存されます。下記の例では、キャッシュが残っていればキャッシュされていたレスポンスを、なければサーバーにリクエストを送って、返ってきたレスポンスを返します (キャッシュに保存はしません)。</p>

<p></p><pre class="crayon-plain-tag">self.addEventListener('fetch', function(event) {
  event.respondWith(
    caches.match(event.request).then(function(response) {
      return response || fetch(event.request);
    })
  );
});</pre><p></p>

<p>Cache APIを使った実例について、詳しくはこちらの記事が参考になります。</p>

<p><a href="http://qiita.com/horo/items/175c8fd7513138308930" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">ServiceWorkerとCache APIを使ってオフラインでも動くWebアプリを作る &#8211; Qiita</a></p>

<p>なお、Cache APIを扱う上で fetch()も避けては通れませんが、ググらビリティの低さから、まとまった資料が見つかりにくいという問題があります。こちらのページがよくまっていたのでリンクしておきます。</p>

<p><a href="http://qiita.com/tomoyukilabs/items/9b464c53450acc0b9574" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">お疲れさまXMLHttpRequest、こんにちはfetch &#8211; Qiita</a></p>

<p>ここまで述べてきたように便利なService Worker + Cache APIですが、多くの人が似たようなコードを書くであろうことは容易に想像ができます。そこで便利なライブラリがいくつかあります。</p>

<h4>sw-toolbox</h4>

<p>リソースをキャッシュする際に必要になるのが、リソースの優先順位の判定です。例えば共通で利用されるJavaScriptファイルは、更新がない限りネットワークにリクエストを送るまでもなく常にキャッシュから読み込むことでスピードを優先したいと思うはずです。逆に例えば、天気アプリであれば、開く度に最新の天気予報を見たいですが、オフラインだった場合はオフラインだと言われるよりは、キャッシュからでもその後の天気予報を見たいと思うはずです。</p>

<p>このように、同じキャッシュを活用するでも、場面によってキャッシュを優先したいのか、ネットワークを優先したいのかが異なってきます。そこで便利なのが<a href="https://github.com/GoogleChrome/sw-toolbox" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">sw-toolbox</a>です。</p>

<p></p><pre class="crayon-plain-tag">toolbox.router.get('/myapp/index.html', toolbox.networkFirst);</pre><p></p>

<p><a href="https://html5experts.jp/wp-content/uploads/2016/08/appshell.png" data-wpel-link="internal"><img src="/wp-content/uploads/2016/08/appshell-640x258.png" alt="appshell" width="640" height="258" class="alignnone size-large wp-image-20539" srcset="/wp-content/uploads/2016/08/appshell.png 640w, /wp-content/uploads/2016/08/appshell-300x121.png 300w, /wp-content/uploads/2016/08/appshell-207x83.png 207w" sizes="(max-width: 640px) 100vw, 640px" /></a></p>

<p>sw-toolboxではネットワークを優先するnetworkFirst、キャッシュを優先するcacheFirstの他にもfastest、networkOnly、cacheOnlyといった様々なオプションが利用可能です。</p>

<p><a href="https://googlechrome.github.io/sw-toolbox/docs/master/index.html" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://googlechrome.github.io/sw-toolbox/docs/master/index.html</a></p>

<h4>sw-precache</h4>

<p>Service Workerを使ったキャッシュ機能以前にオフラインを実現する機能として登場した<a href="https://developer.mozilla.org/ja/docs/Web/HTML/Using_the_application_cache" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Application Cache</a>は、ページを初めて開いた時点で、予め定義されたリソースをすべてキャッシュするものでした。Service Worker + Cache APIにおいても同様のニーズはあるでしょう。例えばアプリの外枠や必要最低限のCSSやJavaScriptファイルを予めロードしておくことで、コンテンツ部分のみをダイナミックにロードすればよいという<a href="https://github.com/GoogleChrome/application-shell" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Application Shell</a>というアーキテクチャを利用する場合です。</p>

<p>これを実現するのに便利なライブラリが<a href="https://github.com/GoogleChrome/sw-precache" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">sw-precache</a>です。Gulp やGruntといったタスクランナーを使うことで、リストされたリソースを最初のロード時点でキャッシュしてしまうJavaScript のコードを出力してくれます。あとはこれをService Workerとしてロードすればよいだけです。また、リソースの更新もハッシュを使って自動的に管理してくれるので、どのファイルを書き換えたからリストを変更しなければならない、といったようなマニュアル作業が不要になります。</p>

<p></p><pre class="crayon-plain-tag">gulp.task('generate-service-worker', function(callback) {
  var path = require('path');
  var swPrecache = require('sw-precache');
  var rootDir = 'app';

  swPrecache.write(path.join(rootDir, 'service-worker.js'), {
    staticFileGlobs: [rootDir + '/**/*.{js,html,css,png,jpg,gif,svg,eot,ttf,woff}'],
    stripPrefix: rootDir
  }, callback);
});</pre><p></p>

<p>なお、最近追加されたruntimeCachingという機能を使うことで、<a href="https://github.com/GoogleChrome/sw-precache#runtimecaching-arrayobject" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">sw-toolbox の便利な機能を宣言的に記述</a>することもできます。</p>

<h4>sw-appcache-behavior</h4>

<p>もし既にApplication Cacheを使っていて、Service Workerに移行を検討している場合は、<a href="https://github.com/GoogleChrome/sw-helpers" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">sw-helpers</a>ライブラリの一部である<a href="https://github.com/GoogleChrome/sw-helpers/tree/master/projects/sw-appcache-behavior" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">sw-appcache-behavior</a>が便利です。既存の Manifestファイルをそのまま利用してService Workerによるオフラインを実現することができます。</p>

<h4>sw-offline-google-analytics</h4>

<p>オフラインのウェブサイトがうまく作れたとしても、ユーザーがどのアクセスページにアクセスし、どのように利用しているかが分からなければ、本当に役に立っているのかどうかを知ることはできません。これもsw-helpersの一部である<a href="https://developers.google.com/web/updates/2016/07/offline-google-analytics?hl=en" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">sw-offline-google-analytics</a>を使えば、オフライン中のアクセス解析をデータベース上に保存しておいて、オンラインになったタイミングでサーバーに送信してGoogle Analyticsでアクセス解析結果を閲覧することができるようになります。</p>

<h3 id="pw-push">Web Push / Web Notifications</h3>

<p>プッシュ通知は、プログレッシブウェブアプリを語る際、これまでできなかったことの中で、技術者ではなくても効果が理解できるという意味では、最も分かりやすい機能です。実際に既に多くのサービスがこれを利用しています。例えば Facebook です。<a href="http://m.facebook.com/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">モバイルウェブ版のFacebook</a>にアクセスしてみると、すぐにそのことに気付くでしょう。</p>

<p></p><pre class="crayon-plain-tag">navigator.serviceWorker.ready().then(function(sw) {
  sw.pushManager.subscribe({userVisibleOnly: true})
    .then(function(sub) {
      // パーミッション取得後
      sendSubToServer(sub);  // Send subscription to the server
  });
});</pre><p></p>

<p><a href="https://html5experts.jp/wp-content/uploads/2016/08/push1.png" data-wpel-link="internal"><img src="/wp-content/uploads/2016/08/push1-640x130.png" alt="push1" width="640" height="130" class="alignnone size-large wp-image-20540" srcset="/wp-content/uploads/2016/08/push1.png 640w, /wp-content/uploads/2016/08/push1-300x61.png 300w, /wp-content/uploads/2016/08/push1-207x42.png 207w" sizes="(max-width: 640px) 100vw, 640px" /></a></p>

<p></p><pre class="crayon-plain-tag">self.addEventListener('push', function(event) {
  event.waitUntil(
    self.registration.showNotification(title, {
      body: body, icon: icon, tag: tag,
      actions: [{action: 'remindMe', title: 'REMIND ME'}]
    }
  });
});</pre><p></p>

<p><a href="https://html5experts.jp/wp-content/uploads/2016/08/push2.png" data-wpel-link="internal"><img src="/wp-content/uploads/2016/08/push2-640x122.png" alt="push2" width="640" height="122" class="alignnone size-large wp-image-20541" srcset="/wp-content/uploads/2016/08/push2.png 640w, /wp-content/uploads/2016/08/push2-300x57.png 300w, /wp-content/uploads/2016/08/push2-207x39.png 207w" sizes="(max-width: 640px) 100vw, 640px" /></a></p>

<p>具体的な実装方法は若干複雑ですので、サードパーティーのサービスを使うのも手かもしれません。自分で実装する場合は、こちらのドキュメントがよくまとまっています。</p>

<p><a href="http://qiita.com/tomoyukilabs/items/217915676603fda73b0a" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Web Pushでブラウザにプッシュ通知を送ってみる &#8211; Qiita</a><br />
<a href="http://qiita.com/tomoyukilabs/items/c7268aa29447a1d0a3fb" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Firefox (Developer Edition)でW3C Push APIを使ってみる &#8211; Qiita</a></p>

<p>Chromeの場合、以前はGCM (Google Cloud Messaging、もしくはFirebase Cloud Messaging)が必須でしたが、現在は<a href="https://tools.ietf.org/html/draft-ietf-webpush-protocol-08" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">標準の Web Push</a>が利用できます。詳しくは下記をご覧下さい。</p>

<p><a href="http://qiita.com/tomoyukilabs/items/8fffb4280c1914b6aa3d" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">ChromeでW3C Push APIを使ってみた &#8211; Qiita</a></p>

<p>Web Pushについては、<a href="https://github.com/web-push-libs/web-push" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">node.js 向けのライブラリ</a>も提供されています。</p>

<h3 id="pw-bg">Background Sync</h3>

<p>妙なタイミングでネットワークがオフラインになってしまったために、リクエストが送れなかったという経験は誰しもあると思います。そんな時にリクエストをキューしておいて、オンラインになったタイミングで送ってくれるとありがたいですよね。それを実現してくれるのが、Background Syncです。</p>

<p><a href="http://qiita.com/horo/items/28bc624b8a26ffa09621" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">ServiceWorkerのBackground Syncでオンライン復帰時にデータ送信 &#8211; Qiita</a></p>

<h3 id="pw-budget">Web Budget API</h3>

<p>現状Service Workerはプッシュを受信すると、通知を出さなければなりません。これはブラウザリソースの無駄遣いを防ぐためです。しかし、例えばバックグラウンドで購読しているフィードを更新するといったユースケースは容易に思いつきます。</p>

<p>そこで検討されているのが<a href="https://beverloo.github.io/budget-api/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Web Budget API</a>という機能です。これを使うことで、ページは与えられたバジェットを適切に配分してバックグラウンドでの処理を行うことが可能になりますので、リソースの無駄遣いを防ぎつつ必要な機能を実現できるようになるはずです。</p>

<p><a href="https://www.chromestatus.com/features/5691190548627456" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://www.chromestatus.com/features/5691190548627456</a></p>

<h3 id="pw-credential">Credential Management API</h3>

<p>Credential Management APIはユーザーのクレデンシャル (認証情報) を制御するためのAPIです。</p>

<div class="youtube">

<!-- iframe plugin v.4.3 wordpress.org/plugins/iframe/ -->
<iframe src="https://www.youtube.com/embed/O3mBdKYMsMY" frameborder="0" 0="allowfullscreen" width="100%" height="500" scrolling="yes" class="iframe-class"></iframe>

</div>

<p>これを使うことにより</p>

<ul>
<li>id / passwordをブラウザに覚えさせる</li>
<li>ソーシャルログインの選択肢をブラウザに覚えさせる</li>
<li>ログイン時にアカウント選択ダイアログを提供する</li>
<li>ユーザーを自動的にログインさせる</li>
</ul>

<p>といったことが可能になります。</p>

<p>典型的なユースケースとしては、ニュースサイトやコマースサイトといった、必ずしもログインしなくてもよい、しかし提供者側としてはログインしてもらいたいサービスにおいて役立ちます。実際に利用されている例としては<a href="https://m.aliexpress.com/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">AliExpress</a>の実装が非常によくできていますので、是非一度お試し下さい。</p>

<p>実装は基本的にフロントエンド側だけですので、比較的気軽に導入することができます。</p>

<ul>
<li><a href="https://credential-management-sample.appspot.com/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">デモアプリ</a> (<a href="https://github.com/GoogleChrome/credential-management-sample" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">ソースコード</a>)</li>
<li><a href="https://developers.google.com/web/updates/2016/04/credential-management-api" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">翻訳済みのドキュメント</a></li>
<li><a href="http://g.co/codelabs/cmapi" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">実際に実装を体験できるコードラボ</a></li>
<li><a href="http://w3c.github.io/webappsec-credential-management/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">仕様</a></li>
<li><a href="https://developer.mozilla.org/en-US/docs/Web/API/Credential_Management_API" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">MDN リファレンス</a></li>
</ul>

<h3 id="pw-payment">Payment Request API</h3>

<p>Payment Request APIは支払い時のチェックアウトに必要なフォームを置き換えるUIを提供します。Chromeではバージョン 53から利用できるようになる予定です。</p>

<div class="youtube">

<!-- iframe plugin v.4.3 wordpress.org/plugins/iframe/ -->
<iframe src="https://www.youtube.com/embed/hmqZxP6iTpo" frameborder="0" 0="allowfullscreen" width="100%" height="500" scrolling="yes" class="iframe-class"></iframe>

</div>

<p>Payment Request APIが提供するのは、ユーザーインターフェースです。ブラウザのAutofill機能を活用することで、クレジットカードの情報や住所などを、すばやく簡単に入力できるようになります。</p>

<p>支払い方法は現状クレジットカードのみですが、まもなくAndroid Payも利用可能 (残念ながら当初日本は対象外) になる予定です。将来的には、さらに様々な支払い方法をオープンかつ柔軟に組み込めるようになっていく予定です。</p>

<ul>
<li><a href="https://developers.google.com/web/updates/2016/07/payment-request" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">アナウンス・概要</a></li>
<li><a href="https://developers.google.com/web/fundamentals/primers/payment-request/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">実装ガイド</a></li>
<li><a href="https://developers.google.com/web/fundamentals/primers/payment-request/android-pay" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Android Pay 実装ガイド</a></li>
<li><a href="https://www.w3.org/TR/payment-request/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">仕様</a></li>
<li><a href="https://developer.mozilla.org/en-US/docs/Web/API/Payment_Request_API" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">MDN リファレンス</a></li>
</ul>

<h3>その他の機能</h3>

<p>プログレッシブウェブアプリを語る上で数えるべき機能や考え方は他にもいくつかありますが、キリがありませんのでこの辺にしておきましょう。</p>

<ul>
<li>Web Bluetooth</li>
<li>Web MIDI</li>
<li>Web Animation</li>
</ul>

<p>興味のある方は、この辺りもぜひ抑えておいていただければと思います。</p>

<h2>Lighthouse</h2>

<p>あるウェブアプリを指してそれがプログレッシブウェブアプリかどうかを議論するのはあまり意味のないことですが、それを示す指標を得ることはできます。それが<a href="https://chrome.google.com/webstore/detail/lighthouse/blipmdconlkpinefehnmjammfjpmpbjk" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">Lighthouse</a>というChrome拡張機能です。ウェブサイトを開いた状態でこの拡張を起動すると、プログレッシブウェブアプリの対応度合いをスコアで示してくれます。</p>

<p><img src="https://lh3.googleusercontent.com/69AN0502kzahk9AmzmLGzRvsVy_6zq2gUnPB8ZH2NIVKMF8DSf9qw44bIh2v9CDVy1FQHFMq1r1US9dey9mjUjA7jg_lLnLcXseLHMVGDDyjGVGetdVCJFu4YlZmQPW18jAcgdg-" alt="" /></p>

<p>Chrome拡張機能としてだけではなく、node moduleとしても利用が可能です。</p>

<p><a href="https://github.com/GoogleChrome/lighthouse" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://github.com/GoogleChrome/lighthouse</a></p>

<p>Lighthouseはまだアルファ版の段階のため、評価できない機能が多数ありますが、今後対応していく予定です。</p>

<h2>リソース</h2>

<p>プログレッシブウェブアプリに興味を持たれた方で、最新情報を追いかけたいという方は下記のページをご覧下さい。RSSフィードも配信しています。</p>

<p><a href="https://developers.google.com/web/updates/" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">https://developers.google.com/web/updates/</a></p>

<p>資料が英語で読めない？英語が分かる人はぜひ、<a href="http://qiita.com/yoichiro6642/items/3afc2f01706398a5a458" data-wpel-link="external" target="_blank" rel="follow external noopener noreferrer">翻訳にご協力下さい</a>。</p>

<h2>最後に</h2>

<p>プログレッシブウェブアプリについて長々と書いてきましたが、重要なのは決してこれらの機能すべてに対応することではありません。自分が提供するサービスがユーザーに最大の価値を提供するために必要であれば、ネイティブアプリを提供するべき場合もあるでしょう。オフライン機能だけが必要な場合もあるかもしれません。これらの機能をうまく組み合わせて、ユーザーが本当に求めている価値を提供できるアプリを開発していただければと思います。</p>
]]></content:encoded>
		
		<series:name><![CDATA[Web技術でアプリ開発2016]]></series:name>
	</item>
	</channel>
</rss>
