VOICEVOX のエディタは Electron・TypeScript・Vue・Vuex などが活用されており、全体構成がわかりにくくなっています。 ここではどういう構成になっているのかを紹介します。コードを読む上で参考になれば幸いです。
とにかくコードを読みたい方向けに、コードが実行される順番を簡潔に説明します。わかりやすさを優先したため正確な表現でない箇所もあることにご注意ください。
- electron がアプリケーションを起動します。このとき実行されるコードが
src/background.ts
です。 src/background.ts
から UI 表示のためのウィンドウを起動します。このとき実行されるコードがsrc/main.ts
です。main.ts
が Vue を使って UI がレンダリングされます。UI のルートコンポーネントがsrc/App.vue
です。src/App.vue
が様々な UI を呼び出します。src/views
やsrc/components
にそれぞれの UI があります。- UI のレンダリングと同時に、アプリの状態やロジックを管理する Vuex のストアが初期化されます。
src/store
ディレクトリに Vuex 用のコードがあります。 - これらのコードは全て TypeScript で書かれています。Vue や TypeScript のコードは実行・ビルド時に自動的に JavaScript へトランスパイルされます。
また electron は、最初に起動するプロセス(メインプロセス)と UI ウィンドウ用のプロセス(レンダラープロセス)で通信するために、プリロードスクリプトsrc/electron/preload.ts
を両方から読み込みます。
flowchart TB
subgraph Electron
direction TB
background.ts -.- preload.ts
end
background.ts --> main.ts
main.ts -.- preload.ts
main.ts --> Vue & Vuex
subgraph Vuex
direction TB
store/
end
subgraph Vue
direction TB
App.vue --> views/ & components/
end
ここでは Electron 周り・Vuex 周り・UI 周りの設計方針を簡単に紹介します。
TODO
VOICEVOX では動的に変更される状態と、状態を変更する関数を、Vuex のStore
を介して管理しています。
Store
は音声・エンジン・UI などそれぞれの目的に応じて分割し、src/store/index.ts
で統合しています。
Vuex の仕様を少し説明します。
状態はState
に保存し、状態の変更はMutation
で行います。
Mutation
は同期的に実行される必要がありますが、非同期処理を行いたい場合はAction
を使います。
VOICEVOX 内では、UI から実行するものはAction
を介するようにして、Action
内からMutation
を実行するようにしています。
これは後述する Undo/Redo 機構のための設計です。
関数は全て文字列を渡す形になっていますが、これも Vuex の仕様です。コードを追う際は文字列で全体検索を行うと便利です。
Vuex は型サポートが不十分なので、VOICEVOX では独自に型周りの機構を作っています(src/store/type.ts
)。
関数を定義したい場合は、まずtype.ts
に型を宣言し、対応するStore
内に実装します。
UI から Electron やエンジンにアクセスしたい場合は、必ず Vuex を介するようにしています。 これは Vuex の元の思想である Flux の考え方を引き継いでいて、外部にアクセスすることで生じる予想外な状態の変更などをなるべく一元管理するためです。
VOICEVOX では Vuex のMutation
をラップする形で Undo/Redo 機構を実現しています。
状態 A→B の変更を行った時、自動的にその逆方向 B→A の変更が Undo スタックに積まれます。
詳細はUndo/Redo を実装したい #116をご参照ください。
TODO
- src
- background.ts ・・・ 最初に実行されるコード。ウィンドウを表示したり、エンジンを起動したりする。Electron のメインプロセス。
- main.ts ・・・ ウィンドウを表示するために最初に実行されるコード。ここで Vue や Vuex を組み込む。
- App.vue ・・・ Vue のルートになるコンポーネント。他の全てのコンポーネントの親。
- views ディレクトリ ・・・ 画面全体を覆うような Vue コンポーネントのディレクトリ。
- components ディレクトリ ・・・ UI のパーツになる Vue コンポーネントディレクトリ。
- store ディレクトリ ・・・ Vuex のストアのディレクトリ。アプリのロジックの大半はここに書かれる。
- electron ディレクトリ ・・・ Electron の ipc 通信などのコードが置かれるディレクトリ。
- type ディレクトリ ・・・ TypeScript 用の型定義などが入るディレクトリ。
- styles ディレクトリ ・・・ CSS や SCSS などのディレクトリ。
- infrastructures ディレクトリ ・・・ UI 用のコードと UI 以外のコードを跨ぐときに一枚かませたいときのためのコードのディレクトリ。
- openapi ディレクトリ ・・・ エンジンの API を叩くためのコードのディレクトリ。OpenAPI で自動生成される。
- router ディレクトリ ・・・ Vue Router 用のディレクトリ。
- helpers ディレクトリ ・・・ 便利な関数を置くディレクトリ。
- public
- アプリのリソースファイルが置かれるディレクトリ。
- build
- ビルドに必要なファイルが置かれるディレクトリ。