package main
import (
"context"
"html/template"
"log"
"net/http"
"time"
"github.com/go-playground/locales"
"github.com/go-playground/locales/currency"
"github.com/go-playground/locales/en"
"github.com/go-playground/locales/fr"
"github.com/go-playground/pure/v5"
"github.com/go-playground/pure/v5/_examples/middleware/logging-recovery"
"github.com/go-playground/universal-translator"
)
var (
tmpls *template.Template
utrans *ut.UniversalTranslator
transKey = struct {
name string
}{
name: "transKey",
}
)
// Translator wraps ut.Translator in order to handle errors transparently
// it is totally optional but recommended as it can now be used directly in
// templates and nobody can add translations where they're not supposed to.
type Translator interface {
locales.Translator
// creates the translation for the locale given the 'key' and params passed in.
// wraps ut.Translator.T to handle errors
T(key interface{}, params ...string) string
// creates the cardinal translation for the locale given the 'key', 'num' and 'digit' arguments
// and param passed in.
// wraps ut.Translator.C to handle errors
C(key interface{}, num float64, digits uint64, param string) string
// creates the ordinal translation for the locale given the 'key', 'num' and 'digit' arguments
// and param passed in.
// wraps ut.Translator.O to handle errors
O(key interface{}, num float64, digits uint64, param string) string
// creates the range translation for the locale given the 'key', 'num1', 'digit1', 'num2' and
// 'digit2' arguments and 'param1' and 'param2' passed in
// wraps ut.Translator.R to handle errors
R(key interface{}, num1 float64, digits1 uint64, num2 float64, digits2 uint64, param1, param2 string) string
// Currency returns the type used by the given locale.
Currency() currency.Type
}
// implements Translator interface definition above.
type translator struct {
locales.Translator
trans ut.Translator
}
var _ Translator = (*translator)(nil)
func (t *translator) T(key interface{}, params ...string) string {
s, err := t.trans.T(key, params...)
if err != nil {
log.Printf("issue translating key: '%v' error: '%s'", key, err)
}
return s
}
func (t *translator) C(key interface{}, num float64, digits uint64, param string) string {
s, err := t.trans.C(key, num, digits, param)
if err != nil {
log.Printf("issue translating cardinal key: '%v' error: '%s'", key, err)
}
return s
}
func (t *translator) O(key interface{}, num float64, digits uint64, param string) string {
s, err := t.trans.C(key, num, digits, param)
if err != nil {
log.Printf("issue translating ordinal key: '%v' error: '%s'", key, err)
}
return s
}
func (t *translator) R(key interface{}, num1 float64, digits1 uint64, num2 float64, digits2 uint64, param1, param2 string) string {
s, err := t.trans.R(key, num1, digits1, num2, digits2, param1, param2)
if err != nil {
log.Printf("issue translating range key: '%v' error: '%s'", key, err)
}
return s
}
func (t *translator) Currency() currency.Type {
// choose your own locale. The reason it isn't mapped for you is because many
// countries have multiple currencies; it's up to you and you're application how
// and which currencies to use. I recommend adding a function it to to your custon translator
// interface like defined above.
switch t.Locale() {
case "en":
return currency.USD
case "fr":
return currency.EUR
default:
return currency.USD
}
}
func main() {
en := en.New()
utrans = ut.New(en, en, fr.New())
setup()
tmpls, _ = template.ParseFiles("home.tmpl")
r := pure.New()
r.Use(middleware.LoggingAndRecovery(true), translatorMiddleware)
r.Get("/", home)
log.Println("Running on Port :8080")
log.Println("Try me with URL http://localhost:8080/?locale=en")
log.Println("and then http://localhost:8080/?locale=fr")
http.ListenAndServe(":8080", r.Serve())
}
func home(w http.ResponseWriter, r *http.Request) {
// get locale translator ( could be wrapped into a helper function )
t := r.Context().Value(transKey).(Translator)
s := struct {
Trans Translator
Now time.Time
PositiveNum float64
NegativeNum float64
Percent float64
}{
Trans: t,
Now: time.Now(),
PositiveNum: 1234576.45,
NegativeNum: -35900394.34,
Percent: 96.76,
}
if err := tmpls.ExecuteTemplate(w, "home", s); err != nil {
log.Fatal(err)
}
}
func translatorMiddleware(next http.HandlerFunc) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
// there are many ways to check, this is just checking for query param &
// Accept-Language header but can be expanded to Cookie's etc....
params := r.URL.Query()
locale := params.Get("locale")
var t ut.Translator
if len(locale) > 0 {
var found bool
if t, found = utrans.GetTranslator(locale); found {
goto END
}
}
// get and parse the "Accept-Language" http header and return an array
t, _ = utrans.FindTranslator(pure.AcceptedLanguages(r)...)
END:
// I would normally wrap ut.Translator with one with my own functions in order
// to handle errors and be able to use all functions from translator within the templates.
r = r.WithContext(context.WithValue(r.Context(), transKey, &translator{trans: t, Translator: t.(locales.Translator)}))
next(w, r)
}
}
func setup() {
en, _ := utrans.FindTranslator("en")
en.AddCardinal("days-left", "There is {0} day left", locales.PluralRuleOne, false)
en.AddCardinal("days-left", "There are {0} days left", locales.PluralRuleOther, false)
fr, _ := utrans.FindTranslator("fr")
fr.AddCardinal("days-left", "Il reste {0} jour", locales.PluralRuleOne, false)
fr.AddCardinal("days-left", "Il reste {0} jours", locales.PluralRuleOther, false)
err := utrans.VerifyTranslations()
if err != nil {
log.Fatal(err)
}
}