2021-01-05 16:05:40 +03:00
// Copyright 2020 The Gitea Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package translation
import (
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/options"
"code.gitea.io/gitea/modules/setting"
"github.com/unknwon/i18n"
"golang.org/x/text/language"
)
// Locale represents an interface to translation
type Locale interface {
Language ( ) string
Tr ( string , ... interface { } ) string
2022-01-02 06:33:57 +03:00
TrN ( cnt interface { } , key1 , keyN string , args ... interface { } ) string
2021-01-05 16:05:40 +03:00
}
2021-01-26 18:36:53 +03:00
// LangType represents a lang type
type LangType struct {
Lang , Name string
}
2021-01-05 16:05:40 +03:00
var (
2021-04-14 21:52:01 +03:00
matcher language . Matcher
allLangs [ ] LangType
supportedTags [ ] language . Tag
2021-01-05 16:05:40 +03:00
)
2021-01-26 18:36:53 +03:00
// AllLangs returns all supported langauages
func AllLangs ( ) [ ] LangType {
return allLangs
}
2021-01-05 16:05:40 +03:00
// InitLocales loads the locales
func InitLocales ( ) {
2021-03-21 18:11:36 +03:00
i18n . Reset ( )
2021-01-05 16:05:40 +03:00
localeNames , err := options . Dir ( "locale" )
if err != nil {
log . Fatal ( "Failed to list locale files: %v" , err )
}
2021-01-26 18:36:53 +03:00
localFiles := make ( map [ string ] [ ] byte )
2021-01-05 16:05:40 +03:00
for _ , name := range localeNames {
localFiles [ name ] , err = options . Locale ( name )
if err != nil {
log . Fatal ( "Failed to load %s locale file. %v" , name , err )
}
}
2021-04-14 21:52:01 +03:00
supportedTags = make ( [ ] language . Tag , len ( setting . Langs ) )
2021-01-05 16:05:40 +03:00
for i , lang := range setting . Langs {
2021-04-14 21:52:01 +03:00
supportedTags [ i ] = language . Raw . Make ( lang )
2021-01-05 16:05:40 +03:00
}
2021-01-26 18:36:53 +03:00
2021-04-14 21:52:01 +03:00
matcher = language . NewMatcher ( supportedTags )
2021-01-26 18:36:53 +03:00
for i := range setting . Names {
key := "locale_" + setting . Langs [ i ] + ".ini"
2021-02-06 00:41:45 +03:00
if err = i18n . SetMessageWithDesc ( setting . Langs [ i ] , setting . Names [ i ] , localFiles [ key ] ) ; err != nil {
2021-03-21 18:11:36 +03:00
log . Error ( "Failed to set messages to %s: %v" , setting . Langs [ i ] , err )
2021-01-26 18:36:53 +03:00
}
}
i18n . SetDefaultLang ( "en-US" )
allLangs = make ( [ ] LangType , 0 , i18n . Count ( ) - 1 )
langs := i18n . ListLangs ( )
names := i18n . ListLangDescs ( )
for i , v := range langs {
allLangs = append ( allLangs , LangType { v , names [ i ] } )
2021-01-05 16:05:40 +03:00
}
}
// Match matches accept languages
2021-04-14 21:52:01 +03:00
func Match ( tags ... language . Tag ) language . Tag {
_ , i , _ := matcher . Match ( tags ... )
return supportedTags [ i ]
2021-01-05 16:05:40 +03:00
}
// locale represents the information of localization.
type locale struct {
Lang string
}
// NewLocale return a locale
func NewLocale ( lang string ) Locale {
return & locale {
Lang : lang ,
}
}
func ( l * locale ) Language ( ) string {
return l . Lang
}
// Tr translates content to target language.
func ( l * locale ) Tr ( format string , args ... interface { } ) string {
return i18n . Tr ( l . Lang , format , args ... )
}
2022-01-02 06:33:57 +03:00
// Language specific rules for translating plural texts
var trNLangRules = map [ string ] func ( int64 ) int {
// the default rule is "en-US" if a language isn't listed here
"en-US" : func ( cnt int64 ) int {
if cnt == 1 {
return 0
}
return 1
} ,
"lv-LV" : func ( cnt int64 ) int {
if cnt % 10 == 1 && cnt % 100 != 11 {
return 0
}
return 1
} ,
"ru-RU" : func ( cnt int64 ) int {
if cnt % 10 == 1 && cnt % 100 != 11 {
return 0
}
return 1
} ,
"zh-CN" : func ( cnt int64 ) int {
return 0
} ,
"zh-HK" : func ( cnt int64 ) int {
return 0
} ,
"zh-TW" : func ( cnt int64 ) int {
return 0
} ,
"fr-FR" : func ( cnt int64 ) int {
if cnt > - 2 && cnt < 2 {
return 0
}
return 1
} ,
}
// TrN returns translated message for plural text translation
func ( l * locale ) TrN ( cnt interface { } , key1 , keyN string , args ... interface { } ) string {
var c int64
if t , ok := cnt . ( int ) ; ok {
c = int64 ( t )
} else if t , ok := cnt . ( int16 ) ; ok {
c = int64 ( t )
} else if t , ok := cnt . ( int32 ) ; ok {
c = int64 ( t )
} else if t , ok := cnt . ( int64 ) ; ok {
c = t
} else {
return l . Tr ( keyN , args ... )
}
ruleFunc , ok := trNLangRules [ l . Lang ]
if ! ok {
ruleFunc = trNLangRules [ "en-US" ]
}
if ruleFunc ( c ) == 0 {
return l . Tr ( key1 , args ... )
}
return l . Tr ( keyN , args ... )
}