3.2 KiB
Internationalization
Internationalization in NocoBase is implemented based on i18next.
How to register a multilingual package?
export class MyPlugin extends Plugin {
load() {
this.app.i18n.addResources('zh-CN', 'test', {
Hello: '你好',
World: '世界',
});
this.app.i18n.addResources('en-US', 'test', {
Hello: 'Hello',
World: 'World',
});
}
}
Two i18n instances
app.i18n
Global i18n instance, typically used in the CLI.
app.i18n.t('World') // "世界" or "World"
ctx.i18n
CloneInstance of global i18n with a completely independent context for each request, typically used to respond to multilingual messages based on the client language.
app.use(async (ctx, next) => {
ctx.body = `${ctx.i18n.t('Hello')} ${ctx.i18n.t('World')}`;
await next();
});
The client request parameters can be placed in the query string
GET /?locale=en-US HTTP/1.1
Host: localhost:13000
or in the request headers
GET / HTTP/1.1
Host: localhost:13000
X-Locale: en-US
Suggested configuration
With English text as the key and translation as the value, this has the advantage that even if multiple languages are missing, it will be displayed in English and will not cause reading barriers, e.g.
i18n.addResources('zh-CN', 'your-namespace', {
'Show dialog': '显示对话框',
'Hide dialog': '隐藏对话框'
});
To make it easier to manage multilingual files, it is recommended to create a locale
folder in the plugin and place all the corresponding language files in it:
|- /my-plugin
|- /src
|- /server
|- locale # Multi-language folder
|- en-cn.ts
|- en-US.ts
Example
Server-side error alert
For example, when a user places an order for a product in the store, if the product is not in stock, or not on the shelf, then the order interface should return the appropriate error when it is called.
const namespace = 'shop';
export default class ShopPlugin extends Plugin {
async load() {
this.app.i18n.addResources('zh-CN', namespace, {
'No such product': '商品不存在',
'Product not on sale': '商品已下架',
'Out of stock': '库存不足',
});
this.app.resource({
name: 'orders',
actions: {
async create(ctx, next) {
const productRepo = ctx.db.getRepository('products');
const product = await productRepo.findOne({
filterByTk: ctx.action.params.values.productId
productId });
if (!product) {
return ctx.throw(404, ctx.t('No such product'));
}
if (!product.enabled) {
return ctx.throw(400, ctx.t('Product not on sale'));
}
if (!product.inventory) {
return ctx.throw(400, ctx.t('Out of stock'));
}
const orderRepo = ctx.db.getRepository('orders');
ctx.body = await orderRepo.create({
values: {
productId: product.id,
quantity: 1,
totalPrice: product.price,
userId: ctx.state.currentUser.id
}
});
next();
}
}
});
}
}