6
4

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

Vite + Vue3 + Pinia + TypeScriptで拡張機能を作成した

Last updated at Posted at 2022-09-04

環境

  • pinia: 2.0.16
  • vue: 3.2.37
  • typescript: ~4.7.4
  • vite: 3.0.1
  • eslint: 8.5.0
  • prettier: 2.5.1

はじめに

タイトルにある通りVue3+TypeScriptでchrome拡張を作成しました。
VueとTypeScriptの相性はあまり良くないイメージがあるのですが、Vue3では割と気持ちよく使えるなという印象です。
参考までに実際に作成したものを貼っておきます。

Viteについて

「ヴィート」と読むらしいです。ずっと「ばいと」って呼んでました...人に言う前に気づけて良かった...
ちなみに語源はフランス語の「素早い」という意味の単語でして、その名の通り非常に高速なビルドツールです。
概要について以下公式まま引用:

Viteは、現代の Web プロジェクトのために、より速く無駄のない開発体験を提供することを目的としたビルドツールです。2 つの主要な部分で構成されています:

  • 非常に高速な Hot Module Replacement (HMR) など、ネイティブ ES モジュールを利用した豊富な機能拡張を提供する開発サーバ。
  • Rollup でコードをバンドルするビルドコマンド。プロダクション用に高度に最適化された静的アセットを出力するように事前に設定されています。

詳しくは公式ドキュメント見るのが一番だと思います。詳しく説明する自信がないわけではないです。断じて。

Vue3について

20年にリリースされ、Vue2から結構書き方が変わりました。
ここではVue3についての記事ではないので詳しい説明は省きますが、一番大きな変更点はsetupだと思います。
ざっくり説明すると、setup内に全ての処理を書いていくイメージです。これがTypeScriptと相性が良いです。

公式例(Composition API):
<script setup lang="ts">
// 変数
const msg = 'Hello!'

// 関数
function log() {
  console.log(msg)
}
</script>

<template>
  <div @click="log">{{ msg }}</div>
</template>

こちらも以下公式ドキュメント参照下さい。
個人的に、Vueの公式ドキュメントってかなり分かりやすいと思っています。

Piniaについて🍍

Vueにおける状態管理と言えばVuexが有名だと思います。
PiniaはVuex5の内容が盛り込まれた状態管理ライブラリでVuex5で統合されると言われています。

公式によると、Vueの公式の状態管理ライブラリはPiniaに変更されたようです。

Vueの公式の状態管理ライブラリがPiniaに変更されました。PiniaはVuex 5のRFCに記載されているVuex 5とほぼ同じ、または強化されたAPIを持っています。PiniaはVuex 5の名前を変えたものだと思えばよいでしょう。PiniaはVue 2.xでも動作します。

以下がVuex4以前との大きな違いになります。

  • mutations が無くなった。
  • TypeScriptのフルサポート
  • モジュールのネストを禁止
  • ネームスペースの廃止

簡単な使い方については以下のとおりです。具体的な使用方法は今回は省きます。

pinia
import { defineStore } from 'pinia'

export const useCounterStore = defineStore('counter', {
  state: () => {
    return { count: 0 }
  },
  // could also be defined as
  // state: () => ({ count: 0 })
  actions: {
    increment() {
      this.count++
    },
  },
})
vue
<script setup lang="ts">
import { useCounterStore } from '@/stores/counter'

const counter = useCounterStore()
counter.count++
counter.$patch({ count: counter.count + 1 })
counter.increment()
</script>

こちらも公式サイトがありますので確認してみてください。

手順

Viteで開発テンプレート作成

早速Viteさんを使います。以下のコマンドで高速にテンプレートを作成できます。
実行後は画面表示に従って作成していけばOKです。

# npm 6.x
$ npm create vite@latest my-vue-app --template vue-ts

# npm 7+ は追加で 2 つのダッシュが必要:
$ npm create vite@latest my-vue-app -- --template vue-ts

# yarn
$ yarn create vite my-vue-app --template vue-ts

一旦テンプレートの作成ができたら、npm install npm run devで動作確認しておきましょう。
ここでViteの素早さを実感できると思います。ストレスフリーです。

Piniaのインストール🍍

まずはプロジェクトにpiniaを入れます。

# npmの場合
$ npm install pinia
# yarnの場合
$ yarn add pinia

続いてmain.tsでPiniaインスタンスを生成します。

src/main.ts
import App from "./App.vue";
import { createApp } from "vue";
import { createPinia } from "pinia";

const pinia = createPinia();
const app = createApp(App);
app.use(pinia);

Did you forget to install pinia?

createPiniaをしていないとこのエラーを吐きます。

しかし、間違いなく書いているのにいつかこのエラーを見ることになるかもしれません。
原因としてはいくつかありますが、私が陥った一例を共有します。

TypeScriptファイルで直接Storeを呼び出していた

どういうことかというと下記をみてください。

src/ts/sample.ts
import { useCounterStore } from "@/stores/counter";

// エラーの原因
const store = useCounterStore();

const sampleFunc = ():string => {
    ...適当な処理
    return "hogehoge";
}

コンポーネントのsetupの外からuseCounterStoreを呼んでいるのが原因でした。
Piniaがセットアップされる前に呼び出されてしまっているということですね。
同じエラーに遭遇したらこの線も考えてみてください。

ストアをリアクティブに変更・参照する

Piniaはちゃんとコンポーネント内で使用してあげれば仲良くなれます。が、リアクティブ性を保つには少し注意が必要になります。
以下の例を見てください。

App.vue
import { toRefs } from "vue";
import { useCounterStore } from "@/stores/counter";

const store = useCounterStore();
// リアクティブ性が失われている
const { count, increment } = toRefs(store);

上記の様に分割代入してしまうとリアクティブ性が失われてしまいます。なのでリアクティブ性を保持したまま分割代入をしたい場合は、storeToRefsを使用します。

App.vue
import { storeToRefs } from "pinia";

const store = useCounterStore();
// リアクティブ性は保持される
const { count, increment } = storeToRefs(store);

なんやかんや書いていく

ここまで準備できれば、後は実現したい機能に向けてVue&TypeScriptで書いていくだけです。
VueとTypeScriptの詳しい書き方等はここでは書きません、というか書く自信がありませんので、とても参考になる記事をいくつか貼っておきます。

後はやっぱり公式ドキュメント見るのが一番だとは思います。

Viteでビルド

実現したい機能が完成したらnpm run buildします。
ビルドの設定などはvite.config.tsで出来ます。参考までに私のを張っておきます。

vite.config.ts
import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue'

// https://vitejs.dev/config/
import { fileURLToPath, URL } from "node:url";

import { defineConfig } from "vite";
import vue from "@vitejs/plugin-vue";

// import vueJsx from "@vitejs/plugin-vue-jsx";

import path from "path";

// https://vitejs.dev/config/
export default defineConfig({
  plugins: [vue()],
  server: {
    host: true,
    port: 3000,
  },
  define: {
    "process.env": {},
  },
  build: {
    rollupOptions: {
      output: {
        entryFileNames: `[name].js`,
        chunkFileNames: `[name].js`,
        assetFileNames: `[name].[ext]`,
      },
    },
    outDir: "./extension/js",
    lib: {
      entry: path.resolve(__dirname, "src/main.ts"),
      name: "main",
      formats: ["umd"],
    },
    cssCodeSplit: false,
  },
  resolve: {
    alias: {
      "@": fileURLToPath(new URL("./src", import.meta.url)),
    },
  },
});

manifest.jsonの作成

拡張機能開発において必須のファイルになります。名の通り拡張機能のマニフェストをJson形式で記載します。
manifest.jsonにはV2とV3がありますが、これから作成するのであればV3で必ず書きましょう。
またすでに公開しているものでもV2で書かれているいるものは2023年6月には動かなくなるみたいです。未対応の方はお早めに!

以下サンプルを記載しておきます。

manifest.json
{
  "manifest_version": 3,
  "name": "Vue3 - Chrome拡張サンプル",
  "version": "1.0",
  "description": "Vue3でChrome拡張を作成したサンプルです。",
  "permissions": ["activeTab", "scripting", "tabs", "storage"],
  "host_permissions": [
    "https://sample/*"
  ],
  "background": {
    "service_worker": "./background.js",
    "type": "module"
  },
  "content_scripts": [
    {
      "matches": [
        "https://sample.com/*",
      ],
      "css": ["./js/style.css"]
    }
  ],
  "icons": {
    "16": "./icon/icon16.png",
    "48": "./icon/icon48.png",
    "128": "./icon/icon128.png"
  }
}

作成出来たら、こんな感じで先ほどのビルド先にコピーして下さい。

├── dist
│   ├── icon
│   └── js
|   └── manifest.json

ここまでできたら、7Zip等のツールを使って、ビルドファイルをZipに圧縮しておいてください。ストアへの公開で必要になります。

manifest.jsonの詳しい書き方は以下。

拡張機能を読み込む

ここまできたらあと少しです。以下ステップを踏めばとりあえず拡張機能を動作することができます。

  1. 自分の拡張機能ページを開きデベロッパーモードをオンにする。
  2. 「パッケージ化されていない拡張機能を読み込む」をクリックする。
  3. 先ほどビルドしたファイルを選択する(Vueデフォルトだとdistファイルになります)

拡張機能が読み込めなかった場合は、manifest.jsonの内容を見直し修正後、再度同じ操作を実施してください。

ストアへ公開するぞ!

後はストアに公開するだけです。
ストアに公開するには、「デベロッパー ダッシュボード」なるものに登録が必要になります。

一応登録料がかかるのですが、1度きり約500円のお支払いでずっと公開し続けることができます。かなり良心的です。

登録が完了したら、ダッシュボードへ移動し+新しいアイテム⇒Zipに圧縮済みのビルドファイルを選択。
qiita.PNG

後は、公開情報やらプライバシーへの取り組みやらの必須項目に入力をしていき、下書き保存審査のため送信を選択すればOKです。

しっかり必要事項が入力できていれば、早ければ1日以内には公開されています。やったぜ。

できたもの

ちなみにこんな感じで私自身が作成したもの。

Netflixの拡張機能で、映画やドラマを自由にカテゴライズし保存できるといったものです。
もしこの記事が少しでもお役に立てたのならば、是非使ってみてください。そして贅沢をいうとフィードバックもお待ちしております。

VSCodeの設定

ついでにVscodeの拡張機能まわりで私が若干はまったことを共有させて頂きます。
これまでVue2で開発されてきた方はVeturを使用されていたと思いますが、Vue3ではVolarなるものを使用します。

インストールしていない方は忘れずしておきましょう。

...あれ..?まだエラーがでてるぞ??となった方は、私と同じミスしている可能性があります。

VeturVolarが両方有効になっている

競合して上手く機能が働いてくれないみたいです。VeturはDisable(Workspace)しておきましょう。

最後に

ここまで読んでくださり誠にありがとうございます。
Qiita初投稿だったので、拙いところが散見されると思います。申し訳ございません。
間違っているところや仕様が変更されているものがございましたら、コメント頂けると幸いです。

では最後に、しつこいですが...これ使ってください!!!

他参考サイト

6
4
2

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
6
4

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?