1
0

ブラウザ拡張機能をReactで開発する【WXT】

Posted at

はじめに

自分のタスク管理用として、ブラウザ拡張機能でタスク管理アプリを開発しようと考えました。React,TypeScriptで開発したいと考えていたところ以下の記事を見つけたので、一番最初に紹介されていたWXTというフレームワークを使用してみました。

個人的にはなかなか使いやすかったので、備忘録として使い方など記事にしてみようと思います。

WXTとは

公式サイトのトップには「Next-gen Web Extension Framework」と記載があります。

次世代のWeb拡張機能フレームワークということで、

  • 近年のモダンなフロントエンド開発に対応している(モダンな開発体験を目指している)
  • Chromeなど1つのブラウザに依存せず、複数のブラウザに対応している

といったことが売りなのかなと思います。

導入方法

こちらの「Bootstrap Project」を参考に導入してみます。なお、今回はnpmで構築していきます。
まず任意のディレクトリで以下のコマンドを実行します。

npx wxt@latest init <project-name>

バージョンは以下で実施しています。

  • node: v21.5.0
  • npm: 10.2.4
  • wxt: 0.19.1

コマンド実行後はいくつか指示があるので、任意の値を入力・選択して進んでください。この記事ではReactを使用するので、templateはreactを選択しています。

ℹ Initalizing new project                                                                                                                                                                     9:38:55
? Choose a template › - Use arrow-keys. Return to submit.
    vanilla
    vue
❯   react  # reactを選択
    solid
    svelte

? Package Manager › - Use arrow-keys. Return to submit.
❯   npm  # npmを選択
    pnpm
    yarn
    bun (experimental)

上記が完了したら、作成されたプロジェクトのディレクトリに移動して以下のコマンドを実行してpackegeのインストールを行います。

npm install

以上で環境構築は完了です。

基本操作

開発用ブラウザの起動

npm run dev

開発する拡張機能を実行可能なブラウザを起動できます。
image.png

立ち上げたままにしておけばコードを保存したタイミングで即座に反映(ホットリロード)してくれますので、開発が非常に楽でした。

ビルド

npm run build
npm run build:firefox # firefoxの場合

Reactで開発していた内容を、ブラウザの拡張機能として読み込ませるための形式に変換してくれます。
プロジェクトディレクトリに[.output/chrome-mv3](Firefoxの場合は[.output/firefox-mv2])というディレクトリが作成されますので、そのディレクトリを拡張機能として読み込ませることで開発した拡張機能を自分のブラウザで利用できるようになります。

拡張機能として読み込ませる方法は以下の記事などを参考にしてください。

zip化

npm run zip
npm run zip:firefox # firefoxの場合

拡張機能を一般に公開するためにはディレクトリ一式をzip圧縮する必要がありますので、そのためのコマンドです。
.outputディレクトリに出力されます。

なお、私は試せていませんがリリースの自動化をするための機能なども用意されているようなので、公開して継続的にリリースする場合などにもかなり重宝しそうです。

他にもいくつかCLIが用意されていますので、公式サイトをご確認ください。

コードの記述方法

WXT API

WXT独自のAPIが用意されています。
これらのAPIを上手に活用することで、任意の設定や独自の機能の開発をブラウザ間での差異等を考慮せずに開発できます。

ディレクトリ構成

初期のディレクトリ構成としてはおおよそ以下ようになります。

.
├── assets
├── entrypoints
│   ├── background.ts # Background scripts
│   ├── content.ts    # Content scripts
│   └── popup/        # Popups
├── public
├── package.json
├── tsconfig.json
└── wxt.config.ts  # 設定ファイル

なお、WXTではTypeScriptがデフォルトで採用されています。JavaScriptで書きたい場合には自身で拡張子を変更する必要があるようです。

All templates default to TypeScript. Rename the file extensions to to use JavaScript instead..js

ディレクトリ構成の中で、コードを書いていくのは基本的に[entrypoints]ディレクトリとなります。
初期の[entrypoints]ディレクトリには以下の3つのファイル、もしくはディレクトリが存在することとなりますが、これがブラウザ拡張機能で開発できる3つの機能に対応することとなります。

  • popup
    • 拡張機能のボタンをクリックした際に表示される画面や機能
    • 開いているWebページのDOMにアクセスすることはできない
  • content
    • Webページのコンテキストで実行される
    • WebページのDOMにアクセス可能(読み取りや変更・追加など)
  • background
    • バックエンドで実行される機能
    • 今のChromeなどの拡張機能(マニフェストv3)ではService Workerと呼ばれる機能

このあとは上記3つの機能の開発方法を見てみます。
なお、これらの機能は単独で使用することも組み合わせて使用することも可能です。

popup

導入方法で作成されたpopupデフォルトのディレクトリ構成は以下の通りとなっています。Reactを触ったことがある方であれば、見慣れたファイルかなと思います。

popup
  ├── App.css
  ├── App.tsx
  ├── index.html
  ├── main.html
  └── style.css

CSS以外の内容は以下の通りです。

index.html
<!doctype html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Default Popup Title</title>
    <meta name="manifest.type" content="browser_action" />
  </head>
  <body>
    <div id="root"></div>
    <script type="module" src="./main.tsx"></script>
  </body>
</html>
main.tsx
import React from 'react';
import ReactDOM from 'react-dom/client';
import App from './App.tsx';
import './style.css';

ReactDOM.createRoot(document.getElementById('root')!).render(
  <React.StrictMode>
    <App />
  </React.StrictMode>,
);
App.tsx
import { useState } from 'react';
import reactLogo from '@/assets/react.svg';
import wxtLogo from '/wxt.svg';
import './App.css';

function App() {
  const [count, setCount] = useState(0);

  return (
    <>
      <div>
        <a href="https://wxt.dev" target="_blank">
          <img src={wxtLogo} className="logo" alt="WXT logo" />
        </a>
        <a href="https://react.dev" target="_blank">
          <img src={reactLogo} className="logo react" alt="React logo" />
        </a>
      </div>
      <h1>WXT + React</h1>
      <div className="card">
        <button onClick={() => setCount((count) => count + 1)}>
          count is {count}
        </button>
        <p>
          Edit <code>src/App.tsx</code> and save to test HMR
        </p>
      </div>
      <p className="read-the-docs">
        Click on the WXT and React logos to learn more
      </p>
    </>
  );
}

export default App;

一般的なReactの記法、React hooksなども利用できます。
image.png

通常のReactの開発のように、コンポーネントを作成して組み合わせることももちろん可能です。

TestComponent.tsx
function TestComponent() {
  return (
    <>
      <h2>Hello World</h2>
    </>
  )
}

export default TestComponent;

App.tsx
import { useState } from 'react';
import reactLogo from '@/assets/react.svg';
import wxtLogo from '/wxt.svg';
import TestComponent from './TestComponent';
import './App.css';

function App() {
  const [count, setCount] = useState(0);

  return (
    <>
      <div>
        <a href="https://wxt.dev" target="_blank">
          <img src={wxtLogo} className="logo" alt="WXT logo" />
        </a>
        <a href="https://react.dev" target="_blank">
          <img src={reactLogo} className="logo react" alt="React logo" />
        </a>
      </div>
      <h1>WXT + React</h1>
      <div className="card">
        <button onClick={() => setCount((count) => count + 1)}>
          count is {count}
        </button>
        <TestComponent />
        <p>
          Edit <code>src/App.tsx</code> and save to test HMR
        </p>
      </div>
      <p className="read-the-docs">
        Click on the WXT and React logos to learn more
      </p>
    </>
  );
}

export default App;

image.png

content

contentはentrypoints直下にcontent.ts(x)を配置するか、contentディレクトリを作成してその中にファイルを作成します。
image.png
参考

なお、開発する際にはwxt.config.tsにruunerパラメータを追加して、どのWebサイトをもとに開発するかを指定した方が良いです。
contentはWebページ上で実行されるものなので、この指定がないと開発ブラウザでうまく動作確認ができません。

wxt.config.ts
import { defineConfig } from 'wxt';

// See https://wxt.dev/api/config.html
export default defineConfig({
  modules: ['@wxt-dev/module-react'],
  // 以下を追加
  runner: {
    startUrls: ["https://google.com"],
  },
});

ここからcontentの書き方のサンプルを紹介します。
contentでもpopupと同じようにUIをReactで構築することができます。いくつか方法があるのですが、WebサイトのCSSに左右されないShadowRootを使用した方法を紹介します。

以下のディレクトリ構成とします。

entrypoints
  └── content
      ├── index.tsx
      ├── App.tsx
      └── style.css

コードの内容です。こちらを参考にしています。

index.tsx
import "./style.css";
import ReactDOM from "react-dom/client";
import App from "./App.tsx";

export default defineContentScript({
  matches: ["*://*/*"],
  cssInjectionMode: "ui",

  async main(ctx) {
    const ui = await createShadowRootUi(ctx, {
      name: "wxt-react-example",
      position: "inline",
      anchor: "body",
      append: "first",
      onMount: (container) => {
        const wrapper = document.createElement("div");
        container.append(wrapper);

        const root = ReactDOM.createRoot(wrapper);
        root.render(<App />);
        return { root, wrapper };
      },
      onRemove: (elements) => {
        elements?.root.unmount();
        elements?.wrapper.remove();
      },
    });

    ui.mount();
  },
});

この中のcreateShadowRootUiはWXT側で用意されている関数です。
ui.mount()でShadow DOMに対してAppコンポーネントをマウントしています。

App.tsx
import { useState } from "react";
import "./style.css";

function App() {
  const [count, setCount] = useState(1);
  const increment = () => setCount((count) => count + 1);

  return (
    <div>
      <h2>Content Script Sample</h2>
      <p>This is React. {count}</p>
      <button onClick={increment}>Increment</button>
    </div>
  );
};

export default App;
style.css
* {
  padding: 0;
  margin: 0;
}

body {
  background-color: #f0f8ff;
  padding: 16px;
}

h2 {
  color: #1e90ff;
}

p {
  color: #2c3e50;
  margin: 8px;
}

button {
  background-color: #4169e1;
  color: white;
  margin: 8px;
  padding: 8px 16px;
  border: none;
  border-radius: 4px;
  cursor: pointer;
  transition: background-color 0.3s ease;
}

button:hover {
  background-color: #1e90ff;
}

image.png
画面上の方に作成されている部分がcontentで構築したものです。
popupやbackgroundと組み合わせることで、表示/非表示の制御なども可能になります。

background

backgroundもcontentやpopup同様、entrypoints直下にbackground.tsを配置するか、backgroundディレクトリを作成してその中にファイルを作成します。
image.png
参考

今回はbackground.tsを作成してみます。
あんまり使い道はないかもしれませんが、一定期間でリロードを行う機能の例です。

background.ts
export default defineBackground(() => {
  const reloadEveryMinute = () => {
    browser.tabs.query({ active: true, currentWindow: true }).then((tabs) => {
      if (tabs[0]) {
        browser.tabs.reload(tabs[0].id);
      }
    });
  };

  setInterval(reloadEveryMinute, 5000);  // 5秒ごとにリロード
});

その他

以下からWXTのサンプルを確認できますので興味のある方はご覧ください。

おわりに

WXTはブラウザ拡張機能でモダンな開発をするための1つの選択肢となるかと思います。個人的には今後も使ってみたいなと感じました。
この記事が何か皆様の参考となれば嬉しく思います。ここまでご覧いただきありがとうございました。

1
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
0