WXTのcontent scriptsの書き方
ブラウザの拡張機能のフレームワークWXTでは、content scriptでUIを差し込むパターンがいくつか用意されています。本記事では、その書き方と実践的な実装を紹介します。
WXTの概要を知りたい方は以前書いた記事WXTとSvelteでChrome拡張機能の開発をご覧ください。
content scripts構成
WXTでは、Popup、Background、Content Scriptのような機能ごとにエントリーポイントを分けます。
エントリーポイントは次のようにディレクトリを切っても切らなくてもよいです。
entrypoints/content/index.ts
entrypoints/content.ts
content scriptの場合、最初からディレクトリを使った方が、構成やコードがすっきりして分かりやすくなると思います。
エントリーポイントに書く内容
content scriptのエントリーポイントにdefineContentScript
を使って差し込む内容を定義します。
先に簡単な例を見てみましょう。
import App from "./App.svelte";
export default defineContentScript({ matches: ["https://example.com/"], main(ctx) { const ui = createIntegratedUi(ctx, { position: "overlay", anchor: "body > div", onMount: (container) => { const app = new App({ target: container, }); return app; }, onRemove: (app) => { app?.$destroy(); }, }); ui.mount(); },});
matches
がcontent scriptを実行するURLです。manifestのmatchesですね。
main
がcontent scriptで実行する処理です。
UIを差し込む
UIを差し込む場合、便利な関数が3つ用意されています。
createIntegratedUi
createShadowRootUi
createIframeUi
先にこの3つの違いを説明します。共通の引数はのちほど紹介します。
createIntegratedUi
createIntegratedUi
を使うと、特に工夫もせずUIがそのまま差し込まれ、ページのCSSの影響を受けます。
content scriptでUIを追加する場合は既存のデザインと合わせることが多いため、この関数はよく使いそうです。
デフォルトではルートとなるタグはdiv
ですが、tag
オプションで変更できます。
const ui = createIntegratedUi(ctx, { position: "overlay", anchor: "body > div", onMount: (container) => { const app = new App({ target: container, }); return app; }, onRemove: (app) => { app?.$destroy(); }, tag: "wxt-content-script",});ui.mount();
例のようにカスタム要素の指定も可能です。
CSSのセレクタとしてdiv
のように指定されている場合に、その影響を減らせます。
IntegratedContentScriptUiOptions
createShadowRootUi
createShadowRootUi
を使うと、UIはShadow DOMとして差し込まれます。
Shadow DOMを使うことでカプセル化され、ページのCSSの影響を受けずにUIを書けます。逆にページもUIのCSSの影響を受けません。
とはいえ、完全に分離されているわけではなく、一部のスタイルは通常のDOMから継承されます(参考:HTML Web Components とは何か | grip on minds)。
createShadowRootUi
を使う場合はcssInjectionMode
の指定やmain
がasync main
に書き換えるなど他の2つと少し違う書き方になります。公式ドキュメントをよく読みましょう。
import './style.css';import App from './App.svelte';
export default defineContentScript({ matches: ['<all_urls>'], cssInjectionMode: 'ui',
async main(ctx) { const ui = await createShadowRootUi(ctx, { name: 'example-ui', position: 'inline', onMount: (container) => { const app = new App({ target: container, }); return app; }, onRemove: (app) => { app?.$destroy(); }, }); ui.mount(); },});
createIframeUi
createIframeUi
はUIをiframeとして挿入します。
iframeを使う場合は別途htmlファイルの準備やmanifestでweb_accessible_resources
の指定が必要です。こちらも公式ドキュメントが圧倒的に分かりやすいです。
正直使い所がパッと思いつかないため、何かあれば追記しようと思います。
UIの配置方法
UIの配置方法はposition
で決めます。anchor
はUI挿入時の起点となるセレクタや要素を指定します(後述)。
const ui = createIntegratedUi(ctx, { position: "overlay", anchor: "body > div", onMount: (container) => { const app = new App({ target: container, }); return app; }, onRemove: (app) => { app?.$destroy(); },});
position
は3種類あります。次の画像は公式ドキュメントより引用です。
data:image/s3,"s3://crabby-images/5193a/5193a3f7ba12fdac9f0c85bec433f753f5efc8cc" alt="positionの種類。公式ドキュメントより引用"
次の項目からそれぞれを詳しく解説します。
Type alias: ContentScriptPositioningOptions – WXT
inline
inline
は文字どおりインラインで配置します。一番使いそうですね。
Interface: ContentScriptInlinePositioningOptions – WXT
overlay
overlay
は重ねて表示されます。
alignment
で起点に対するUIの位置を決めます。次の画像は公式ドキュメントより引用です。
data:image/s3,"s3://crabby-images/d4081/d4081bfc430208f276bbc651dfa8dffd365f083e" alt="overlayのalignmentの種類。公式ドキュメントより引用"
画像では右下になっていますがこれはあくまでも一例です。
overlay
を使うと、「0x0ピクセルの要素」が「挿入されるUIのルート」として用意されます。これが起点です。
起点の具体的な挿入位置はanchor
やappend
で指定できます(後述)。
実装を見た所createIntegratedUi
を使った場合は挙動が違うようです(参考1、参考2)。
Interface: ContentScriptOverlayPositioningOptions – WXT
modal
modal
を使うとUIが全体に広がり、モーダルのようになります。
overlay
と同じく、createIntegratedUi
を使った場合は挙動が違うようです(参考1、参考2)。
次はShadowRoot(Shadow DOM)で使った例です。
const ui = await createShadowRootUi(ctx, { name: "example-ui", position: "modal", anchor: "body > div", onMount: (container) => { const app = new App({ target: container, }); return app; },});
data:image/s3,"s3://crabby-images/71106/7110647b9dfb3521aa84248a56dc754b4be5c2f1" alt="modalとShadowRootの例"
content scriptの内容が画面全体に広がっていますね。
Interface: ContentScriptModalPositioningOptions – WXT
タグの挿入方法
ここからはHTMLのタグがどのように挿入されるか決定するオプションを解説します。
anchor
anchor
はUI挿入時の起点となる要素を指定します。具体的にはセレクタとなる文字列や要素を書きます。
文字列の場合はdocument.querySelector(anchor)
のように処理されます。
const ui = createIntegratedUi(ctx, { anchor: "body > div", // ... });},
一方、要素を直接渡すとそのままマウントしてくれます。
async main(ctx) { const anchor = document.querySelector("body > div"); const ui = createIntegratedUi(ctx, { anchor: anchor, // ... });},
() => undefined | null | string | Element
のような関数も書けるっぽいです。
Interface: ContentScriptAnchoredOptions – WXT
append
anchor
の要素を基準に、どのように拡張機能側のタグを挿入するかをappend
で指定します。
次の画像は公式ドキュメントより引用です。
data:image/s3,"s3://crabby-images/22c86/22c869add63677b0680aa28c32b82fa5bbb28043" alt="appendの種類。公式ドキュメントより引用。"
HTMLの挿入の関数append
やinsertBefore
などを自分で調べるよりも直感的で扱いやすいです。
Interface: ContentScriptAnchoredOptions – WXT
Type alias: ContentScriptAppendMode – WXT
要素の出現を待つパターン
content scriptでよくある
「ある要素の出現を待ってからマウント」
の例を書いておきます。
関数自体はWXT以外でも使えます。
要素の出現を待つ関数
まずは要素の出現を待つ関数を実装します。MutationObserver
とsetInterval
それぞれの例を紹介します。
MutationObserverを使うパターン
MutationObserver
を使ってDOMの変更を監視して要素を見つけるパターンです。
export async function waitForElement( selector: string,): Promise<Element | undefined> { return new Promise((resolve) => { const elm = document.querySelector(selector); if (elm) { console.log("最初に見つかった"); return resolve(elm); }
const observer = new MutationObserver((mutations) => { for (const mutation of mutations) { for (const node of mutation.addedNodes) { if (node instanceof Element && node.matches(selector)) { observer.disconnect(); clearTimeout(timeout); console.log("差分で見つかった"); return resolve(node); } } } }); const timeout = setTimeout(() => { observer.disconnect(); console.log("見つからなかった"); return resolve(undefined); }, 10000); observer.observe(document.body, { childList: true, subtree: true, }); });}
少し長ったらしく見えるかもしれませんがやっていることは単純です。
- セレクタで指定した要素があれば返す
- なければDOMの変更の監視を開始
- 出現したら返す
- 10秒経っても出現しなければ
undefined
を返す
setIntervalを使うパターン
setInterval
を使い、0.5秒おきにdocument.querySelector
を実行するパターンです。
export async function waitForElement( selector: string,): Promise<Element | undefined> { return new Promise((resolve) => { let elm = document.querySelector(selector); if (elm) { console.log("最初に見つかった"); return resolve(elm); }
const timer = setInterval(() => { elm = document.querySelector(selector); if (elm) { clearInterval(timer); clearTimeout(timeout); console.log("途中で見つかった"); return resolve(elm); } }, 500); const timeout = setTimeout(() => { clearInterval(timer); console.log("見つからなかった"); return resolve(undefined); }, 10000); });}
「要素の出現を待つ関数」の実装については、Stack Overflowで10年以上議論されている伝説の質問に他の実装も載っています。
mainの実装
上記の関数を使ってdefineContentScript
のmain
を書いたのが次の例です。
async main(ctx) { const anchor = await waitForElement("selectoooooooooooooooor"); if (typeof anchor === "undefined") { return; } const ui = createIntegratedUi(ctx, { position: "inline", anchor: anchor, append: "first", onMount: (container) => { // TODO: svelteの例なので適宜書き換え const app = new App({ target: container, }); return app; }, onRemove: (app) => { app?.$destroy(); }, tag: "extension-name-hereeeeeeeeeeeeeee", }); ui.mount();},
要素が見つからなければマウントしない、という単純な構成です。残りの部分は今までで解説したとおりです。
WXTを使ったcontent scriptの実装を紹介しました。
UIの挿入はたくさんの実装方法があります。
WXTではある程度パターン化でき、「どこに何をマウントするのか」が書きやすく読みやすいので重宝しそうです(特にappend
)。
今回紹介したコードを動かしたい方向けに次のリポジトリを用意しました。参考程度にどうぞ。
eetann/wxt-content-script | GitHub