お問い合わせ

ブログ

これまでに経験してきたプロジェクトで気になる技術の情報を紹介していきます。

Laravel mix vue No.9 - Vue Internationalization - Vue多言語化

okuda Okuda 4 years
Laravel mix vue No.9 - Vue Internationalization - Vue多言語化

こんにちは、あすかのkoheiです。

今回はVue側を多言語化します。

連載記事

Vue Internationalization - Vue多言語化

サンプル


フォルダ構成


└─ server
   └─ resources
      └─ js
         ├─ components
         |  └─ Header.vue
         ├─ pages
         |  ├─ Home.vue
         |  ├─ Login.vue
         |  └─ Reset.vue
+        ├─ lang
+        |  ├─ index.js
+        |  ├─ En.js
+        |  └─ Js.js
         └─ app.js

dockerスタート

gitからクローンした場合は.envの作成と設定を忘れないように!

# コンテナスタート
docker-compose start

# コンテナに入る
docker-compose exec php bash

# composerをインストール(前回からの続きで行う場合はいらない)
composer install

# npmをインストール(前回からの続きで行う場合はいらない)
npm i

# encryption keyを作成(前回からの続きで行う場合はいらない)
php artisan key:generate

# ホットリリード開始
npm run watch

多言語化モジュールをインストール

今回はvue-i18nを使用する

インストール

vue-i18nをインストール


  npm i vue-i18n

言語ファイルの準備

言語ファイルの作成

server\resources\js\langを作成しその中に
En.jsJa.jsを準備する

server\resources\js\lang\En.jsを作成


    const contents = {
        messages: {
            // 単語
            word: {
                hello: 'hello!',
            },
            // メッセージ
            sentence: {
                '{msg} world!': '{msg} world!',
            }
        },
        /*  
          日付フォーマット

          以下の定義形式で日時をローカライズ
          weekday         "narrow", "short", "long"
          era             "narrow", "short", "long"
          year            "2-digit", "numeric"
          month           "2-digit", "numeric", "narrow", "short", "long"
          day             "2-digit", "numeric"
          hour            "2-digit", "numeric"
          minute          "2-digit", "numeric"
          second          "2-digit", "numeric"
          timeZoneName    "short", "long"
        */
        dateTimeFormats: {
            full: {
                year: "numeric",
                month: "short",
                day: "numeric",
                weekday: "short",
                hour: "numeric",
                minute: "numeric"
            },
            day: {
                year: "numeric",
                month: "short",
                day: "numeric"
            },
            time: {
                hour: "numeric",
                minute: "numeric"
            },
            week: {
                weekday: "long"
            }
        },
        // ナンバーフォーマット
        numberFormats: {
            currency: {
                style: 'currency',
                currency: 'USD'
            }
        }
    };

    export {
        contents
    };

server\resources\js\lang\Ja.jsを作成


    const contents = {
        messages: {
            // 単語
            word: {
                hello: 'こんにちは!',
            },
            // メッセージ
            sentence: {
                '{msg} world!': '{msg} 世界!',
            }
        },
        /*
          日付フォーマット

          以下の定義形式で日時をローカライズ
          weekday         "narrow", "short", "long"
          era             "narrow", "short", "long"
          year            "2-digit", "numeric"
          month           "2-digit", "numeric", "narrow", "short", "long"
          day             "2-digit", "numeric"
          hour            "2-digit", "numeric"
          minute          "2-digit", "numeric"
          second          "2-digit", "numeric"
          timeZoneName    "short", "long"
        */
        dateTimeFormats: {
            full: {
                year: "numeric",
                month: "short",
                day: "numeric",
                weekday: "short",
                hour: "numeric",
                minute: "numeric",
                hour12: true
            },
            day: {
                year: "numeric",
                month: "short",
                day: "numeric"
            },
            time: {
                hour: "numeric",
                minute: "numeric",
                hour12: true
            },
            week: {
                weekday: "short"
            }
        },
        // ナンバーフォーマット
        numberFormats: {
            currency: {
                style: 'currency',
                currency: 'JPY',
                currencyDisplay: 'symbol'
            }
        }
    };

    export { contents };

言語ファイルをまとめる

各言語ファイルをまとめるserver\resources\js\lang\index.jsを作成


    // 言語ファイルをインポート
    import {contents as en} from "./En";
    import {contents as ja} from "./Ja";

    // 言語アイテムをマージ
    const messages = {
        en: en.messages,
        ja: ja.messages
    };

    // 日付フォーマットをマージ
    const dateTimeFormats = {
        en: en.dateTimeFormats,
        ja: ja.dateTimeFormats
    };

    // ナンバーフォーマットをマージ
    const numberFormats = {
        en: en.numberFormats,
        ja: ja.numberFormats
    };

    // それぞれをエクスポート
    export { messages };
    export { dateTimeFormats };
    export { numberFormats };

Vueで使う準備

server\resources\js\app.jsに追加


    ...

    // ストアをインポート
    import store from "./store";
    // ヘルパーをインポート
+   import Helper from "./helper";

    ...

    /**
     * vue-i18n
     * https://kazupon.github.io/vue-i18n/
     */
    // モジュールのインポート
+   import VueI18n from "vue-i18n";
    // 言語コンテンツのインポート
+   import { messages, dateTimeFormats, numberFormats } from "./lang/index";
    // 多言語化の宣言
+   Vue.use(VueI18n);
    // ローカルストレージから「language」を取得してセット、ない場合はブラウザーの言語をセット
+   const locale = Vue.storage.get("language", Helper.getLanguage());
    // VueI18nコンストラクタのオプションを指定
+   const i18n = new VueI18n({
        // 言語設定
+       locale: locale,
        // 選択中の言語に対応する文字列が存在しない場合はこの言語の文字列を使用する
+       fallbackLocale: "en",
        // インポートした言語コンテンツ
+       messages,
+       dateTimeFormats,
+       numberFormats
+   });

    // 非同期通信でAUTHストアのcurrentUserアクションを実行するので
    // asyncメソッドにして、awaitで通信をまつ
    const createApp = async () => {

        // AUTHストアのcurrentUserアクションでユーザの認証状態をチェック
        await store.dispatch("auth/currentUser");

        new Vue({
            // マウントする要素に「index.blade.php」の「<div id="app"></div>」のidを登録
            el: "#app",
            // ルーターの使用を宣言
            router,
            // ストアを登録
            store,
            // I18nを登録
+           i18n,
            // 使用するコンポーネントにルートコンポーネントの登録
            components: { App },
            // 描画するテンプレート
            template: "<App />"
        });
};

埋め込み

使用例


// messages word
<p>{{ $t('word.hello') }}</p>

// messages sentence
<p>{{ $t('sentence.{msg} world!', {msg: 'hello'}) }}</p>

// datetime full
<p>{{ $d(new Date(), 'full') }}</p>

// datetime day
<p>{{ $d(new Date(), 'full') }}</p>

// number currency 
<p>{{ $n(100, 'currency') }}</p>

// vueの中で取得する場合
this.$i18n.tc('sentence.sentence.{msg} world!');

コンポネントに埋め込み

server\resources\js\components\Header.vueを修正


    <template>
        <header>
            <!-- リンクを設定 -->
-           <RouterLink to="/">home</RouterLink>
+           <RouterLink to="/">{{ $t('word.home') }}</RouterLink>
-           <RouterLink v-if="!isLogin" to="/login">login</RouterLink>
+           <RouterLink v-if="!isLogin" to="/login">{{ $t('word.login') }}</RouterLink>
            <!-- ログインしている場合はusernameを表示 -->
            <span v-if="isLogin">{{username}}</span>
            <!-- クリックイベントにlogoutメソッドを登録 -->
-           <span v-if="isLogin" @click="logout">logout</span>
+           <span v-if="isLogin" @click="logout">{{ $t('word.logout') }}</span>
            <!-- 言語切替 -->
            <select v-model="selectedLang" @change="changeLang">
-               <option v-for="lang in langList" :value="lang.value" :key="lang.value">{{ lang.text }}</option>\
+               <option v-for="lang in langList" :value="lang.value" :key="lang.value">{{ $t(`word.${lang.text}`) }}</option>
            </select>
        </header>
    </template>

    <script>
    import Cookies from "js-cookie";
    import Helper from "../helper";

    export default {
        data() {
            return {
                ...
            };
        },
        computed: {
            ...
        },
        storage: {
            ...
        },
        methods: {
            // ログアウトメソッド
            async logout() {
                ...
            },
            // 言語切替
            changeLang() {
                // ローカルストレージに「language」をセット
                this.$storage.set("language", this.selectedLang);
                // Apiリクエスト 言語を設定
                axios.get(`/api/set-lang/${this.selectedLang}`);
                // Vue i18n の言語を設定
+               this.$i18n.locale = this.selectedLang;
            }
        },
        created() {
           ...
        }
    };
    </script>

ページに埋め込み

server\resources\js\pages\Home.vueを修正


    <template>
      <div class="container">

-       <h1>home</h1>
+       <h1>{{ $t('word.home') }}</h1>

      </div>
    </template>

server\resources\js\pages\Login.vueを修正


    <template>
        <div class="container">
            <!-- tabs -->
            <ul class="tab">
-               <li class="tab__item" :class="{'tab__item--active': tab === 1 }" @click="tab = 1">Login</li>
+               <li class="tab__item" :class="{'tab__item--active': tab === 1 }" @click="tab = 1">{{ $t('word.login') }}</li>
                <li
                    class="tab__item"
                    :class="{'tab__item--active': tab === 2 }"
                    @click="tab = 2"
-               >Register</li>
+               >{{ $t('word.register') }}</li>
                <li
                    class="tab__item"
                    :class="{'tab__item--active': tab === 3 }"
                    @click="tab = 3"
-               >Forgot password ?</li>
+               >{{ $t('word.forgot_password') }}</li>
            </ul>
            <!-- /tabs -->

            <!-- login -->
            <section class="login" v-show="tab === 1">
-               <h2>Login</h2>
+               <h2>{{ $t('word.login') }}</h2>

                <!-- errors -->
                <div v-if="loginErrors" class="errors">
                    <ul v-if="loginErrors.email">
                        <li v-for="msg in loginErrors.email" :key="msg">{{ msg }}</li>
                    </ul>
                    <ul v-if="loginErrors.password">
                        <li v-for="msg in loginErrors.password" :key="msg">{{ msg }}</li>
                    </ul>
                </div>
                <!--/ errors -->

                <!-- @submitで login method を呼び出し -->
                <!-- @submitイベントリスナに prevent をつけるとsubmitイベントによってページがリロードさない -->
                <form @submit.prevent="login">
-                   <div>Email</div>
+                   <div>{{ $t('word.email') }}</div>
                    <div>
                        <!-- v-modelでdataをバインド -->
                        <input type="email" v-model="loginForm.email" />
                    </div>
-                   <div>Password</div>
+                   <div>{{ $t('word.password') }}</div>
                    <div>
                        <input type="password" v-model="loginForm.password" />
                    </div>
                    <div>
-                       <button type="submit">login</button>
+                       <button type="submit">{{ $t('word.login') }}</button>
                    </div>
                </form>

-               <h2>Socialite</h2>
+               <h2>{{ $t('word.Socialite') }}</h2>
                <a class="button" href="/login/twitter" title="twitter">
                    twitter
                </a>
                <a class="button" href="/login/facebook" title="facebookfacebook">
                    facebook
                </a>
                <a class="button" href="/login/google" title="google">
                    google
                </a>
                <a class="button" href="/login/github" title="github">
                    github
                </a>
            </section>
            <!-- /login -->

            <!-- register -->
            <section class="register" v-show="tab === 2">
-               <h2>register</h2>
+               <h2>{{ $t('word.register') }}</h2>
                <!-- errors -->
                <div v-if="registerErrors" class="errors">
                    <ul v-if="registerErrors.name">
                        <li v-for="msg in registerErrors.name" :key="msg">{{ msg }}</li>
                    </ul>
                    <ul v-if="registerErrors.email">
                        <li v-for="msg in registerErrors.email" :key="msg">{{ msg }}</li>
                    </ul>
                    <ul v-if="registerErrors.password">
                        <li v-for="msg in registerErrors.password" :key="msg">{{ msg }}</li>
                    </ul>
                </div>
                <!--/ errors -->
                <form @submit.prevent="register">
-                   <div>Name</div>
+                   <div>{{ $t('word.name') }}</div>
                    <div>
                        <input type="text" v-model="registerForm.name" />
                    </div>
-                   <div>Email</div>
+                   <div>{{ $t('word.email') }}</div>
                    <div>
                        <input type="email" v-model="registerForm.email" />
                    </div>
-                   <div>Password</div>
+                   <div>{{ $t('word.password') }}</div>
                    <div>
                        <input type="password" v-model="registerForm.password" />
                    </div>
-                   <div>Password confirmation</div>
+                   <div>{{ $t('word.password_confirmation') }}</div>
                    <div>
                        <input type="password" v-model="registerForm.password_confirmation" />
                    </div>
                    <div>
-                       <button type="submit">register</button>
+                       <button type="submit">{{ $t('word.register') }}</button>
                    </div>
                </form>
            </section>
            <!-- /register -->

            <!-- forgot -->
            <section class="forgot" v-show="tab === 3">
-               <h2>forgot</h2>
+               <h2>{{ $t('word.forgot_password') }}</h2>
                <!-- errors -->
                <div v-if="forgotErrors" class="errors">
                    <ul v-if="forgotErrors.email">
                        <li v-for="msg in forgotErrors.email" :key="msg">{{ msg }}</li>
                    </ul>
                </div>
                <!--/ errors -->
                <form @submit.prevent="forgot">
-                   <div>Email</div>
+                   <div>{{ $t('word.email') }}</div>
                    <div>
                        <input type="email" v-model="forgotForm.email" />
                    </div>
                    <div>
-                       <button type="submit">send</button>
+                       <button type="submit">{{ $t('word.send') }}</button>
                    </div>
                </form>
            </section>
            <!-- /forgot -->
        </div>
    </template>

    <script>
    export default {
    // vueで使うデータ
        data() {
            ...
        },
        computed: {
            ...
        },
        methods: {
            async login() {
                ...
            },
            /*
            * register
            */
            async register() {
                // authストアのregisterアクションを呼び出す
                await this.$store.dispatch("auth/register", this.registerForm);
                // 通信成功
                if (this.apiStatus) {
                    // メッセージストアで表示
                    this.$store.commit("message/setContent", {
-                       content: "認証メールを送りました。",
                        // メッセージストアはサーバからのメッセージもあるので「i18n.tc」でコード内で翻訳しておく
+                       content: this.$i18n.tc("sentence.sent_verification_email"),
                        timeout: 10000
                    });
                    // AUTHストアのエラーメッセージをクリア
                    this.clearError();
                    // フォームをクリア
                    this.clearForm();
                }
            },
            /*
            * forgot
            */
            async forgot() {
                // authストアのforgotアクションを呼び出す
                await this.$store.dispatch("auth/forgot", this.forgotForm);
                if (this.apiStatus) {
                    // show message
                    this.$store.commit("message/setContent", {
-                       content: "パスワードリセットメールを送りました。",
                        // メッセージストアはサーバからのメッセージもあるので「i18n.tc」でコード内で翻訳しておく
+                       content: this.$i18n.tc('sentence.sent_password_reset_email'),
                        timeout: 10000
                    });
                    // AUTHストアのエラーメッセージをクリア
                    this.clearError();
                    // フォームをクリア
                    this.clearForm();
                }
            },
            clearError() {
                ...
            },
            clearForm() {
                ...
            }
        }
    }
    <script>
    ...

server\resources\js\pages\Reset.vueを修正


        <template>
            <div class="container--small">
-               <h2>password reset</h2>
+               <h2>{{ $t('word.password_reset') }}</h2>

                <div class="panel">
                    <!-- @submitで login method を呼び出し -->
                    <!-- @submitイベントリスナに reset をつけるとsubmitイベントによってページがリロードさない -->
                    <form class="form" @submit.prevent="reset">
                        <!-- errors -->
                        ...
                        <!--/ errors -->

                        <div>
                            <input type="password" v-model="resetForm.password" />
                        </div>
                        <div>
                            <input type="password" v-model="resetForm.password_confirmation" />
                        </div>
                        <div>
-                           <button type="submit">reset</button>
+                           <button type="submit">{{ $t('word.reset') }}</button>
                        </div>
                    </form>
                </div>
            </div>
        </template>
        ...

翻訳ファイルに設定したものを追加

server\resources\js\lang\En.jsを編集


    const contents = {
        messages: {
            // 単語
            word: {
                hello: 'hello!',
+               home: 'Home',
+               login: 'Login',
+               logout: 'logout',
+               english: 'English',
+               japanese: 'Japanese',
+               register: 'Register',
+               forgot_password: 'Forgot Password ?',
+               email: 'Email',
+               password: 'Password',
+               Socialite: 'Socialite',
+               name: 'Name',
+               password_confirmation: 'Password Confirmation',
+               send: 'Send',
+               password_reset: 'Password Reset',
+               reset: 'Reset',
            },
            // メッセージ
            sentence: {
                '{msg} world!': '{msg} world!',
+               sent_verification_email: 'Sent verification email.',
+               sent_password_reset_email: 'Sent password reset email.',
            }
        },
        ...

server\resources\js\lang\Ja.jsを編集


    const contents = {
        messages: {
            // 単語
            word: {
                hello: 'こんにちは!',
+               home: 'ホーム',
+               login: 'ログイン',
+               logout: 'ログアウト',
+               english: '英語',
+               japanese: '日本語',
+               register: '登録',
+               forgot_password: 'パスワードをわすれましたか?',
+               email: 'メールアドレス',
+               password: 'パスワード',
+               Socialite: 'ソーシャルログイン',
+               name: '名前',
+               password_confirmation: 'パスワード確認',
+               send: '送信',
+               password_reset: 'パスワード リセット',
+               reset: 'リセット',
            },
            // メッセージ
            sentence: {
                '{msg} world!': '{msg} world!',
+               sent_verification_email: '確認メールを送信しました。',
+               sent_password_reset_email: 'パスワード再設定メールを送信しました。',
            }
        },
        ...

これでVue側でも多言語化が完了しました。

次は、「Vue Font Awesome」、「Vue Formulate」を使ってUIをもっと使いやすくしていこうと思います。

Laravel mix vue No.10 - Vue Font Awesome, Vue Formulate, etc

Laravel mix vue No.9 - Vue Internationalization - Vue多言語化 2021-08-18 04:14:14

コメントはありません。

4638

お気軽に
お問い合わせください。

お問い合わせ
gomibako@aska-ltd.jp