GitHub - intlify/vue-i18n-locale-message: :globe_with_meridians: i18n locale messages management tool for vue-i18n (original) (raw)
๐ vue-i18n-locale-message
i18n locale messages management tool / library for vue-i18n
โ ๏ธ NOTICE
v2.x or later
is for vue-i18n-next (Vue 3.x)
If you want to use for vue-i18n@v8.x (Vue 2.x), use the v1.x
๐ Motivations
The big motivation is as follows.
- ๐ซ Hard to integrate locale messages for localization services
- ๐ซ Hard to maintain consistency of locale message keys (
eslint-plugin-vue-i18n
need it!) - ๐ Requested by 3rd vendor tools (
i18n-ally
and etc ...)
๐ฟ Installation
npm
npm install --save-dev vue-i18n-locale-message
If you can globally use CLI only, you need -g
option the below command.
npm install -g vue-i18n-locale-message
yarn
yarn add -D vue-i18n-locale-message
If you can globally use CLI, you need global
option the below command.
yarn global add vue-i18n-locale-message
โญ Features
- API
- squeeze the meta of locale messages from
i18n
custom block - infuse the meta of locale messages to
i18n
custom block - get translation status from localization service
- diff locale messages between local and localization service
- push the locale messages to localization service
- squeeze the meta of locale messages from
- CLI
- squeeze: squeeze the locale messages from
i18n
custom block - infuse: infuse the locale messages to
i18n
custom block - push: push the locale messages to localization service
- pull: pull the locale mesagees from localization service
- diff: diff locale messages between local and localization service
- status: indicate translation status from localization service
- import: import locale messages to localization service
- export: export locale messages from localization service
- list: list undefined fields in locale messages
- squeeze: squeeze the locale messages from
๐ Usages
API
const fs = require('fs') const { squeeze, infuse } = require('vue-i18n-locale-message')
// read single-file component contents
// NOTE: about scheme of target contents, see the SFCInfo
type at types/index.d.ts const files = [ { path: '/path/to/src/components/Modal.vue', content:
...
{
...
}
`
},
// ...
]
// squeeze meta locale message from single-file components
// NOTE: about scheme of meta locale message, see the MetaLocaleMessage
type at types/index.d.ts
const meta = squeeze('/path/to/src', files)
// write squeezed meta locale messages fs.writeFileSync('/path/to/src/meta.json', JSON.stringify(meta))
// after update meta locale message with 3rd vendor tools or your code, it read meta locale messsage const updatedMeta = require('/path/to/src/updated-meta.json')
// infuse meta locale message to single-file components const updatedFiles = infuse('/path/to/src', files, updatedMeta)
// write updated single-file component updateFiles.forEach(file => { fs.writeFileSync(file.path, file.content) })
CLI
Squeeze
vue-i18n-locale-message squeeze --target=./src --output=./messages.json
Infuse
vue-i18n-locale-message infuse --target=./src --locales=./translated.json
Push
vue-i18n-locale-message push --provider=l10n-service-provider
--conf=110n-service-provider-conf.json
--target-paths=./src/locales/.json
--filename-match=^([\w])\.json
Pull
vue-i18n-locale-message pull --provider=l10n-service-provider
--conf=110n-service-provider-conf.json
--output=./src/locales
Diff
vue-i18n-locale-message diff --provider=l10n-service-provider
--conf=110n-service-provider-conf.json
--target-paths=./src/locales/.json
--filename-match=^([\w])\.json
Status
vue-i18n-locale-message status --provider=l10n-service-provider
--conf=110n-service-provider-conf.json
Import
$ vue-i18n-locale-message import --provider=l10n-service-provider
--conf ./l10n-service-provider-conf.json
--target=./src/locales/ja.json
--format=json
Export
$ vue-i18n-locale-message export --provider=l10n-service-provider
--conf ./l10n-service-provider-conf.json
--output=./src/locales
list
vue-i18n-locale-message list --locale=en
--target-paths=./src/locales/.json
--filename-match=^([\w])\.json
๐ Exit codes
Codes | Description |
---|---|
4 | Not completed localization |
5 | There are undefined fields in locale messages |
64 | difference between local and localization services |
๐ API: Specifications
squeeze (basePath: string, files: SFCFileInfo[]): MetaLocaleMessage
- Arguments:
{string} basePath
: The base path that single-file components are located in project{SFCFileInfo[]} files
: The target single-file components information
- Return:
MetaLocaleMessage
Squeeze the meta of locale messages from i18n custom block at single-file components.
In about structure of the meta information that is returned with this function, You can see the here.
infuse (basePath: string, sources: SFCFileInfo[], meta: MetaLocaleMessage): SFCFileInfo[]
- Arguments:
{string} basePath
: The base path that single-file components are located in project{SFCFileInfo[]} sources
: The target single-file components information{MetaLocaleMessage}
: The meta of locale message
- Return:
SFCFileInfo[]
Infuse the meta of locale messages to i18n custom block at single-file components.
infuse
function will return new single-file components information that is updated with the single-file components information specified as sources
and the meta of locale message as meta
.
status (options: TranslationStatusOptions): Promise<TranslationStatus[]>
- Arguments:
- `{options}
*provider
: The target localization service provider, required, sameprovider
option ofstatus
command
*conf
: The json file configration of localization service provider, sameconf
option ofstatus
command
*locales
: For some locales of translation status, samelocales
option ofstatus
command
- `{options}
- Return:
Promise<TranslationStatus[]>
๐ Provider: Specifications
You can use the push
or pull
commands to push the locale message to the localization service as a resource for that service, and also to pull resources from the l10n service as the locale message.
When you run the following commands,
push
pull
diff
status
import
export
you need the provider that implements the following.
- export provider factory function
- provider factory function must return a provider object that have the following I/F:
push
methodpull
methodstatus
methodimport
methodexport
method
The type definition with TypeScript is as follows:
/**
- Provider factory function */ type ProviderFactory<T = {}> = (configration: ProviderConfiguration) => Provider
/**
- Translation Status */ export type TranslationStatus = { locale: Locale // target locale percentage: number // translation percentage }
/**
- Raw Locale Message */ export type RawLocaleMessage = { locale: Locale // target locale format: string // locale message format data: Buffer // data of locale message }
/**
- Provider interface / interface Provider { /*
- push the locale messsages to localization service / push (args: PushArguments): Promise /*
- pull the locale messages from localization service / pull (args: PullArguments): Promise /*
- indicate translation status from localization service / status (args: StatusArguments): Promise<TranslationStatus[]> /*
- import the locale messsages to localization service / import (args: ImportArguments): Promise /*
- export the locale message buffer from localization service */ export (args: ExportArguments): Promise<RawLocaleMessage[]> }
type CommonArguments = { dryRun: boolean // whether the CLI run as dryRun mode normalize?: string // normalization ways for locale messages or resource }
/**
- Provider Push Arguments */ type PushArguments = { messages: LocaleMessages // the locale messages that push to localization service } & CommonArguments
/**
- Provider Pull Arguments */ type PullArguments = { locales: Locale[] // locales that pull from localization service, if empty, you must pull the all locale messages } & CommonArguments
/**
- Provider Status Arguments */ export type StatusArguments = { locales: Locale[] // locales that indicate translation status from localization service, if empty, you must indicate translation status all locales }
/**
- Provider Import Arguments */ export type ImportArguments = { messages: RawLocaleMessage[] // the raw locale messages that import to localization service } & CommonArguments
/**
- Provider Export Arguments */ export type ExportArguments = { locales: Locale[] // locales that export from localization service, if empty, you must export all locale messages format: string // locale messages format } & CommonArguments
/**
- ProviderConfiguration provider fields structure
- e.g.
- {
"provider": {
"token": "xxx"
}
- }
*/ interface ProviderConfiguration<T = {}> { provider: { [key in keyof ProviderConfigurationValue]: ProviderConfigurationValue[key] } }
type ProviderConfigurationValue<T = {}> = T & { [prop: string]: unknown }
As an implementation example of Provider, there is poeditor-service-provider implemented as localization service provider of poeditor.
๐ CLI: Locale message squeezing rules
The structure of locale messages to be squeezed is layered with the directory structure and single-file component (.vue
) filename.
This repotitory demo
project directory structure:
cd demo tree src src โโโ App.vue โโโ components โย ย โโโ Modal.vue โย ย โโโ nest โย ย โโโ RankingTable.vue โโโ i18n.js โโโ locales โย ย โโโ en.json โย ย โโโ ja.json โโโ main.js โโโ pages โโโ Login.vue
4 directories, 8 files
You use vue-cli-locale-message
CLI, run squeeze
command as follows:
vue-i18n-locale-message squeeze --target=./src --output=./messages.json cat ./messages.json
You will get the following JSON structure (the following output results are commented To make it easier to understand):
{
"ja": { // for ja
locale "App": { // src/App.vue "title": "ใขใใชใฑใผใทใงใณ", "lang": "่จ่ชๅใๆฟใ" }, "components": { // src/components "Modal": { // src/components/Modal.vue "ok": "OK", "cancel": "ใญใฃใณใปใซ" } }, "pages": { // src/pages "Login": { // src/pages/Login.vue "id": "ใฆใผใถใผID", "password": "ใในใฏใผใ", "confirm": "ใในใฏใผใใฎ็ขบ่ชๅ
ฅๅ", "button": "ใญใฐใคใณ" } } }, "en": { // for
en` locale
"App": { // src/App.vue
"title": "Application",
"lang": "Change languages"
},
"components": { // src/components
"Modal": { // src/components/Modal.vue
"ok": "OK",
"cancel": "Cancel"
},
"nest": { // src/components/nest
"RankingTable": { // src/components/nest/RankingTable.vue
"headers": {
"rank": "Rank",
"name": "Name",
"score": "Score"
}
}
}
}
}
}
๐ Changelog
Details changes for each release are documented in the CHANGELOG.md.
โ Issues
Please make sure to read the Issue Reporting Checklist before opening an issue. Issues not conforming to the guidelines may be closed immediately.
โ TODO
Managed with GitHub Projects