GitHub上で動作するChrome拡張を作成したかったのですが、目的のページへ遷移しても動作しなくてはまったので対応メモです。
やりたいこと
GitHub上の特定のページへ遷移すると発火するChrome拡張を作りたいです。
現象
manifest.jsonのcontent_scriptsのmatchesに当てはまるURLにアクセスしても、content_scriptsに書いたJavaScriptが動作しませんでした。
原因
SPAなのかSSRなのかはわかりませんが、GitHubはページ遷移時、ページ全体を読み込むのではなく差分のみを読み込んでレンダリングしているみたいです。
content scriptだとこの差分更新を上手く検知できないようです。ページの読み込みが発生しない&非同期で差分更新をしてるので当然っちゃ当然なんですが。
対策
以下の手順を踏むと、SPAとかでもページの遷移を検知できます。
イベントページを使う
content_scriptsではなく、イベントページ/バックグラウンドページを使います。
イベントページ/バックグラウンドページを使う場合、manifest.jsonに以下のように記述すると、イベントページ/バックグラウンドページとしてbackground.js
が起動します。
"background": {
"scripts": [
"background.js"
],
"persistent": false
},
persistent
はtrueに設定するとバックグラウンドページとして、falseに設定するとイベントページとして動作します。
バックグラウンドページは常にバックグラウンドで常駐するため、メモリを常に占有してしまいます。イベントページは必要な時のみ立ち上がって、処理が完了したら閉じます。
現在はイベントページが推奨されているようなので、特別理由がない限りはpersistent
はfalseで良さそうです。
参考:
イベントページでのページ遷移の検知
chrome.tabs.onUpdated.addListener
を使うと、うまいことタブの更新を検知できます。
具体的には以下のような感じで使います。
chrome.tabs.onUpdated.addListener(function(tabId, info, tab) { if (info.status === 'complete' && tab.url.indexOf('https://github.com/') !== -1) { // いい感じの処理 } });
chrome.tabs
を使うときはmanifest.jsonのpermissionsにtabsを追加する必要があります。
"permissions": [
"tabs",
],
このchrome.tabs.onUpdated.addListener
ですが、読み込み時と読み込み完了時の2回走ります。
読み込み時はinfo.status
がloading
になり、読み込み完了時はinfo.status
がcomplete
となります。
今回は読み込み完了時だけ処理が走ればよく、またGitHub上でのみ動けばいい拡張機能を作りたかったので、上記のようにif分で条件を絞っています。
参考:
イベントページでDOMにアクセスする
Chrome拡張ではcontent_scriptsだとDOMに普通にアクセスできるのですが、イベントページやポップアップからDOMへのアクセスしようとすると一工夫必要です。
具体的には以下のような感じで、chrome.tabs.executeScript
を使用します。executeScript
を使えば、content_scriptsをイベントページに挿入できます。
chrome.tabs.onUpdated.addListener(function(tabId, info, tab) { if (info.status === 'complete' && tab.url.indexOf('https://github.com/') !== -1) { chrome.tabs.executeScript(null, { file: './contentScript.js' }, () => {}); } });
この例では./contentScript.js
が実行されます。この./contentScripts.js
はcontent_scriptsとして扱われるので普通にDOMにアクセスすることが可能です。
なお、content_scriptsとして扱われるためmanifest.jsonにURLを登録する必要があります。以下のようにpermissionsに./contentScripts.js
が走る可能性のあるURLを追加しましょう。
"permissions": [
"tabs",
"https://github.com/*"
],
ここに追加されていないURL上で./contentScripts.js
が発火しようとすると、イベントページのデベロッパーツール上でエラーが発生します。
これでSPAとかのWebページでも意図したタイミングで発火するChrome拡張を作れそうです。
参考:
コメント