Chrome拡張機能の開発
拡張機能の構成
拡張機能には大きく三つの要素が両立する:
- background(現在service workerという):DOMにアクセスできないが、その代わり外部サーバーに接続できるので、バックエンドっぽい
- popup:拡張機能のアイコンをクリックするときに表示されるポップアップ
- contents scripts:全てのブラウザーページで実行される。ブラウザーのランタイム機能を部分的にしか使えない
そして、それぞれの要素のやり取り方法はこちら:
しかしこの度開発した拡張機能では、contents scriptsへのやりとりが必要とされず、どちらかというと次の図になった:
ポップアップ構築
ポップアップの役割は比較的に単純:ツールバーにある拡張機能アイコンがクリックされたときに表示されるポップアップのになる。そのため、ただのHTMLファイルであり、次のようにマニフェストで指定できる:
"action": {
"default_popup": "popup.html"
}
あとはポップアップの裏側に作動するpopup.js
をHTML内に<script>
タグとともに埋め込む。
background(またはService Worker)
マニフェストv3から名前はbackgroundからservice workerに変わり、動作も少し違う。
Chrome for Developersによると、現在のサービスウワーカーは「拡張機能の中心的なイベント ハンドラ」であり、イベントを長く受信しなければシャットダウンするという特徴がある。
サービスワーカー側でchrome.runtime.onMessage.addListener
でリスナーを実装することでchrome.runtime.sendMessage
からのメッセージを処理できるようになる。(参照)
よくある問題
Unchecked runtime.lastError: The message port closed before a response was received
:このエラーはsendMessage
への不適切な返事により発生する。この場合、解決方法は二つ:
-
onMessage.addListener
の最後にtrueを返す - ちゃんとしたPromiseを返す
Content Scripts
役割:
- イベントの処理
- DOM操作
例えばGoogle翻訳の場合、文字選択イベントを検知すれば、コンテンツ スクリプトは「翻訳」ボタンを表示し、今回クリックを検知すればボタンを消すという例が挙げられる。
コンテンツ スクリプトもmanifestに指定する必要がある:
"content_scripts": [
{
"js": ["content-script.js"],
"matches": ["<all_urls>"]
}
]
js
フィールドでスクリプトのパスを指定した上で、matches
にスクリプトの挿入先を指定する。普段はmatches
に<scheme>://<host>/<path>
のようなURLを渡すべきだが、こうすればURLに一致するページでのみcontent-script.js
がロードされる。その反面、全てのページに一致する<all_urls>
を指定すると「Chrome ウェブストアでの拡張機能の審査には時間がかかることがあります」。
ボタン表示
間違った方法:chrome.windows.create
は完全に新しいウインドウを作成してしまう。
代わりに、新しいdiv
を作り、背景画像にAnkiのアイコンを指定する。また、ユーザが文字を選択してマウスボタンを放した位置に表示する。
画像の指定
背景画像なので、画像へのパスを.style.backgroundImage
に代入したいが、まずはmanifestのweb_accessible_resources
に画像へのパスを追加する(assets
フォルダなら"assets/*")。
次にchrome.runtime.getURL
で正しいパスをCSSのurl
関数に渡してから代入できる。
位置
正しい位置に表示するために二つの要素を揃える:
- マウスイベントの
clientX/clientY
:複数の画面に対応する。screenX/Y
だと、どうしてもマウスの位置はメイン画面を元に計算されるため、間違った座標になってしまう。 -
clientX/Y
にscrollingElement
の高さ・幅を加える。でないと、スクロールして文字を選択してもボタンはスクロールした分間違った位置に生成される。
セットアップ
ECMAScriptモジュールへの変換
import
構文を使いたい場合は次のファイルにファイルのタイプはmodule
であることを示さないといけない:
-
manifest.jsonにbackgroundの
sw.js
をmodule化:manifest.json"background": { "service_worker": "sw.js", "type": "module" }
ただし、上記はChromeの場合のみの機能する。詳細はmanifestの違い
-
popup.htmlで<script>のtypeを
module
にする -
package.jsonに
"type": "module"
ChromeとFirefoxの違い
参照:Build a cross-browser extension
厳密に言えば、Chrome対Firefoxというよりも、FirefoxとSafari対Chrome、OperaとEdgeのは現状である。
例えば、Firefoxならchrome.runtime
ではなく、browser.runtime
になるわけだ。
manifestの違い
-
id
: Firefoxではマニフェストv3の拡張機能のidを準備する必要がある:
"browser_specific_settings": {
"gecko": {
"id": "extension@example.com"
}
}
-
background
: Chromeでは"service_worker"、Firefoxでは"page"か"scripts"("scripts"なら複数のファイルを指定できる)
Chrome:
"background": {
"service_worker": "background.js",
"type": "module"
}
Firefox:
"background": {
"page": "background.js"
// か
"scripts": ["helper.js", "background.js"],
"type": "module"
}
browser-polyfillで互換性を高める
browser.*
を使って不具合が発生しないように、WebExtension browser API Polyfillを使うといい。
Discussion