需求
我想在 golang gin 同一套网站模板中同时支持中文和英文,类似 SpringBoot 中的 Thymeleaf I18N 那样,能在模板中调用翻译。
<label th:text="#{label}"></label>
改造思路
但是 gin i18n 的文档里并没有介绍如何在 template 中使用翻译。大概是很少有人用 gin 写网站吧,都是如何在 api 中使用翻译的方式。
Google 搜素了半天,几乎没找到几个模板国际化相关的参考。好不容易找到一个台湾大哥的实现方案,给了我很大启发。
其实就是使用 golang template 的自定义函数功能。
https://siongui.github.io/2016/01/19/i18n-go-web-application-by-gettext-html-template/const tmpl = `
<span>{{gettext "Home"}}</span>
<span>{{gettext "About"}}</span>
`
func main() {
setup("zh_TW", "messages", "locale")
setup("vi_VN", "messages", "locale")
funcMap := template.FuncMap{
"gettext": translate,
}
t, _ := template.New("foo").Funcs(funcMap).Parse(tmpl)
这个方案的实现非常像 IBM 的风格,从作者资料看确实在 IBM 呆过。。。
The new template package allows you to at add a function to template's function map, that would transform the given string to a localized version.
golang gin 模板 i18n 实现方案
参考 gin i18n 的这个封装库的 example 目录里的代码
https://github.com/gin-contrib/i18n
我感觉只要把 ginI18n.MustGetMessage 封装一下,映射到模板中,就基本满足我的使用需求了。
修改默认配置
https://github.com/gin-contrib/i18n/blob/master/constant.go
里面显式指定了翻译文件的文件类型,及所在目录。
所以项目里可以分离出一个 i18n 文件来写死这些配置。也方便在其他不需要此功能的项目,能快速删除代码。
开干!
安装 gin-contrib/i18n
go get github.com/gin-contrib/i18n
需要注意的是:
- gin-contrib/i18n 同时自动安装了 nicksnyder/go-i18n
- 自动升级了 gin 的版本
go: added github.com/gin-contrib/i18n v0.0.1
go: upgraded github.com/gin-gonic/gin v1.7.2 => v1.7.4
go: added github.com/nicksnyder/go-i18n/v2 v2.1.2
代码实现
i18n.go
- 将默认语言设置为了中文。因为主要客户在国内,但是对于某些业务,还是设置为英文合适
- 只判断 url 中的语言参数,而不判断 http 头里的 Accept-Language。默认配置会优先使用 Accept-Language,但是我不喜欢这种逻辑
- 选择 toml 是因为示例基本都是 toml 的,看起来对复杂场景(复数、整句带变量)支持较好。
package main
import (
"github.com/BurntSushi/toml"
ginI18n "github.com/gin-contrib/i18n"
"github.com/gin-gonic/gin"
"golang.org/x/text/language"
)
// apply i18n middleware
// router.Use(GinI18nLocalize())
func GinI18nLocalize() gin.HandlerFunc {
return ginI18n.Localize(
ginI18n.WithBundle(&ginI18n.BundleCfg{
RootPath: "./lang",
AcceptLanguage: []language.Tag{language.Chinese, language.English},
DefaultLanguage: language.Chinese,
FormatBundleFile: "toml",
UnmarshalFunc: toml.Unmarshal,
}),
ginI18n.WithGetLngHandle(
func(context *gin.Context, defaultLng string) string {
lng := context.Query("lang")
if lng == "" {
return defaultLng
}
return lng
},
),
)
}
main.go
r := gin.Default()
// apply i18n middleware
r.Use(GinI18nLocalize())
// 自定义模板函数
r.SetFuncMap(template.FuncMap{
"Localize": ginI18n.GetMessage,
})
lang 目录,存放翻译资源文件
> tree lang/
lang/
├── en.toml
└── zh.toml
> cat lang/en.toml
News = "News"
> cat lang/zh.toml
News = "新闻资讯"
模板 index.html 中使用
<h2>{{ Localize "News" }}</h2>
链接格式:
默认中文
http://localhost:9014/
英文
http://localhost:9014/?lang=en
中文
http://localhost:9014/?lang=zh
没有法语资源,所以使用默认的中文翻译
http://localhost:9014/?lang=fr
测试了一下,成功!
在线效果演示
例如,我写了个小 demo:
感慨
- golang gin 的文档真是少,即便找到了 i18n 的库,也得看代码才知道怎么用。相比 SpringBoot 详尽的文档,感觉看 go 的源码更爽一点。。。
- 缺乏标准,就得花大精力去选库,确实挺耗费精力