mirror of https://github.com/gogits/gogs.git
228 lines
5.1 KiB
228 lines
5.1 KiB
// Copyright 2013 Unknwon |
|
// |
|
// Licensed under the Apache License, Version 2.0 (the "License"): you may |
|
// not use this file except in compliance with the License. You may obtain |
|
// a copy of the License at |
|
// |
|
// http://www.apache.org/licenses/LICENSE-2.0 |
|
// |
|
// Unless required by applicable law or agreed to in writing, software |
|
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
|
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
|
// License for the specific language governing permissions and limitations |
|
// under the License. |
|
|
|
// Package i18n is for app Internationalization and Localization. |
|
package i18n |
|
|
|
import ( |
|
"errors" |
|
"fmt" |
|
"reflect" |
|
"strings" |
|
|
|
"gopkg.in/ini.v1" |
|
) |
|
|
|
var ( |
|
ErrLangAlreadyExist = errors.New("Lang already exists") |
|
|
|
locales = &localeStore{store: make(map[string]*locale)} |
|
) |
|
|
|
type locale struct { |
|
id int |
|
lang string |
|
langDesc string |
|
message *ini.File |
|
} |
|
|
|
type localeStore struct { |
|
langs []string |
|
langDescs []string |
|
store map[string]*locale |
|
defaultLang string |
|
} |
|
|
|
// Get target language string |
|
func (d *localeStore) Get(lang, section, format string) (string, bool) { |
|
if locale, ok := d.store[lang]; ok { |
|
if key, err := locale.message.Section(section).GetKey(format); err == nil { |
|
return key.Value(), true |
|
} |
|
} |
|
|
|
if len(d.defaultLang) > 0 && lang != d.defaultLang { |
|
return d.Get(d.defaultLang, section, format) |
|
} |
|
|
|
return "", false |
|
} |
|
|
|
func (d *localeStore) Add(lc *locale) bool { |
|
if _, ok := d.store[lc.lang]; ok { |
|
return false |
|
} |
|
|
|
lc.id = len(d.langs) |
|
d.langs = append(d.langs, lc.lang) |
|
d.langDescs = append(d.langDescs, lc.langDesc) |
|
d.store[lc.lang] = lc |
|
|
|
return true |
|
} |
|
|
|
func (d *localeStore) Reload(langs ...string) (err error) { |
|
if len(langs) == 0 { |
|
for _, lc := range d.store { |
|
if err = lc.message.Reload(); err != nil { |
|
return err |
|
} |
|
} |
|
} else { |
|
for _, lang := range langs { |
|
if lc, ok := d.store[lang]; ok { |
|
if err = lc.message.Reload(); err != nil { |
|
return err |
|
} |
|
} |
|
} |
|
} |
|
return nil |
|
} |
|
|
|
// SetDefaultLang sets default language which is a indicator that |
|
// when target language is not found, try find in default language again. |
|
func SetDefaultLang(lang string) { |
|
locales.defaultLang = lang |
|
} |
|
|
|
// ReloadLangs reloads locale files. |
|
func ReloadLangs(langs ...string) error { |
|
return locales.Reload(langs...) |
|
} |
|
|
|
// Count returns number of languages that are registered. |
|
func Count() int { |
|
return len(locales.langs) |
|
} |
|
|
|
// ListLangs returns list of all locale languages. |
|
func ListLangs() []string { |
|
langs := make([]string, len(locales.langs)) |
|
copy(langs, locales.langs) |
|
return langs |
|
} |
|
|
|
func ListLangDescs() []string { |
|
langDescs := make([]string, len(locales.langDescs)) |
|
copy(langDescs, locales.langDescs) |
|
return langDescs |
|
} |
|
|
|
// IsExist returns true if given language locale exists. |
|
func IsExist(lang string) bool { |
|
_, ok := locales.store[lang] |
|
return ok |
|
} |
|
|
|
// IndexLang returns index of language locale, |
|
// it returns -1 if locale not exists. |
|
func IndexLang(lang string) int { |
|
if lc, ok := locales.store[lang]; ok { |
|
return lc.id |
|
} |
|
return -1 |
|
} |
|
|
|
// GetLangByIndex return language by given index. |
|
func GetLangByIndex(index int) string { |
|
if index < 0 || index >= len(locales.langs) { |
|
return "" |
|
} |
|
return locales.langs[index] |
|
} |
|
|
|
func GetDescriptionByIndex(index int) string { |
|
if index < 0 || index >= len(locales.langDescs) { |
|
return "" |
|
} |
|
|
|
return locales.langDescs[index] |
|
} |
|
|
|
func GetDescriptionByLang(lang string) string { |
|
return GetDescriptionByIndex(IndexLang(lang)) |
|
} |
|
|
|
func SetMessageWithDesc(lang, langDesc string, localeFile interface{}, otherLocaleFiles ...interface{}) error { |
|
message, err := ini.Load(localeFile, otherLocaleFiles...) |
|
if err == nil { |
|
message.BlockMode = false |
|
lc := new(locale) |
|
lc.lang = lang |
|
lc.langDesc = langDesc |
|
lc.message = message |
|
|
|
if locales.Add(lc) == false { |
|
return ErrLangAlreadyExist |
|
} |
|
} |
|
return err |
|
} |
|
|
|
// SetMessage sets the message file for localization. |
|
func SetMessage(lang string, localeFile interface{}, otherLocaleFiles ...interface{}) error { |
|
return SetMessageWithDesc(lang, lang, localeFile, otherLocaleFiles...) |
|
} |
|
|
|
// Locale represents the information of localization. |
|
type Locale struct { |
|
Lang string |
|
} |
|
|
|
// Tr translates content to target language. |
|
func (l Locale) Tr(format string, args ...interface{}) string { |
|
return Tr(l.Lang, format, args...) |
|
} |
|
|
|
// Index returns lang index of LangStore. |
|
func (l Locale) Index() int { |
|
return IndexLang(l.Lang) |
|
} |
|
|
|
// Tr translates content to target language. |
|
func Tr(lang, format string, args ...interface{}) string { |
|
var section string |
|
|
|
idx := strings.IndexByte(format, '.') |
|
if idx > 0 { |
|
section = format[:idx] |
|
format = format[idx+1:] |
|
} |
|
|
|
value, ok := locales.Get(lang, section, format) |
|
if ok { |
|
format = value |
|
} |
|
|
|
if len(args) > 0 { |
|
params := make([]interface{}, 0, len(args)) |
|
for _, arg := range args { |
|
if arg == nil { |
|
continue |
|
} |
|
|
|
val := reflect.ValueOf(arg) |
|
if val.Kind() == reflect.Slice { |
|
for i := 0; i < val.Len(); i++ { |
|
params = append(params, val.Index(i).Interface()) |
|
} |
|
} else { |
|
params = append(params, arg) |
|
} |
|
} |
|
return fmt.Sprintf(format, params...) |
|
} |
|
return format |
|
}
|
|
|