Statamic Peak

Article

Traduire Laravel Jetstream et Inertia

Comment traduire un nouveau projet à base de Laravel Jetstream et InertiaJs avec vue-i18n / Matice ? Je vous explique les différentes possibilités !

Pour traduire les différents contenus de Laravel et de Jetstream, le package laravel-lang/lang fournit des fichiers clef en main à copier / coller. Il nous reste seulement à gérer la traduction des fichiers Vue.

Pour cela, nous avons 2 possibilités :

  • Baser notre traduction uniquement coté serveur. C'est le partie pris du package GENL/Matice.
  • Séparer les traductions front-end / back-end et utiliser le package vue-i18n

En fonction de votre équipe et de vos possibilités d'évolution, une solution ou l'autre est préférable. Matice s'utilise très simplement pour une équipe qui vient de Blade / Laravel parce qu'il reprend le fonctionnement des fonctions côté PHP. Le package est encore jeune et pour l'instant il ne gère pas encore le lazy loading de plusieurs langues : on choisit dans un fichier de config les langues à inclure. C'est seulement un problème si votre projet gère de nombreuses langues ou traductions

À l'inverse, vue-i18n est la solution la plus utilisée pour traduire des projets Vue mais ne propose pas d'intégration particulière pour Laravel.

Utiliser GENL/Matice

On ajoute le package composer :

composer require genl/matice

On ajoute la directive @translations dans notre fichier app.blade.php avant le chargement de nos fichiers JS.

Coté JS, un package est également disponible pour ajouter les fonctions de traductions inspirées de Laravel :

npm install matice --save-dev

Ensuite, les fonctions et leurs paramètres sont détaillés dans le README du projet et sont similaires à Laravel.

Utiliser Vue-i18n

Installation de Vue-i18n dans un projet Laravel Jetstream

Pour vue-i18n, la solution est uniquement côté front-end.

npm i --save-dev vue-i18n@next

Pour utiliser des fichiers JSON ou des traductions dans les fichiers Vue, il est nécessaire d'ajouter @intlify/vue-i18n-loader et de configurer webpack.

npm i --save-dev @intlify/vue-i18n-loader@next

Dans le fichier webpack.config.js :

module: {
        rules: [
            {
                test: /\\.vue$/,
                loader: 'vue-loader',
            },
            {
                test: /\\.(json5?|ya?ml)$/, // target json, json5, yaml and yml files
                type: 'javascript/auto',
                loader: '@intlify/vue-i18n-loader',
                include: [ // Use `Rule.include` to specify the files of locale messages to be pre-compiled
                    path.resolve(__dirname, 'src/locales')
                ]
            },
            {
                resourceQuery: /blockType=i18n/,
                type: 'javascript/auto',
                loader: '@intlify/vue-i18n-loader'
            },
        ]
    }

Utilisation de Vue-i18n dans un projet Laravel Jetstream

Il ne nous reste plus qu'à ajouter vue-i18n à notre instance de Vue.

Pour cela, on modifie notre fichier app.js.

import { createI18n } from 'vue-i18n'

const i18n = createI18n({
    locale: 'fr',
    fallbackLocale: 'fr',
    messages: {
        fr
    }
});

// ajouter après .use(InertiaPlugin)
.use(i18n)

Il est alors possible de créer des blocs i18n directement dans un composant :

<i18n locale="fr">{
    "forgot_password": "Mot de passe oublié?",
    "login": "Se connecter",
    "remember_me": "Se souvenir de moi"
}</i18n>

Sinon, on peut mettre nos traductions dans un fichier js.

export default {
    "agree": "J'accepte les {tos} et la {privacyPolicy}",
    "email": "Email",
    "name" : "Raison sociale",
    "password": "Mot de passe",
}

Notre installation rend disponible une fonction this.$t() dans nos composants qui prend en paramètre la clef de traduction et renvoie le contenu en fonction de la langue définie.

Par exemple :

<jet-label for="password_confirmation" :value="$t('confirm_password')" />

Parfois, ce fonctionnement n'est pas suffisant. Je veux afficher le message

"J'accepte les Conditions d'Utilisation et la Politique de Confidentialité" et créer des liens vers 2 pages différentes.

Pour des cas complexes, vue-i18n permet de créer une balise i18n-t qui dans notre exemple va remplacer {tos} et {privacyPolicy} par les templates fournis. Coté traduction, on a juste à indiquer

"agree": "J'accepte les {tos} et la {privacyPolicy}".

<i18n-t keypath="agree">
    <template #tos>
        <a target="_blank" :href="route('terms.show')" class="underline text-sm text-gray-600 hover:text-gray-900">{{
                $t('tos')
            }}</a>
    </template>
    <template #privacyPolicy>
        <a target="_blank" :href="route('policy.show')" class="underline text-sm text-gray-600 hover:text-gray-900">{{$t('privacy_policy')}}</a>
    </template>
</i18n-t>