2020-08-06 19:26:15 +02:00
|
|
|
intllib = {
|
|
|
|
getters = {},
|
|
|
|
strings = {},
|
|
|
|
}
|
|
|
|
|
2021-12-23 10:54:09 +01:00
|
|
|
local path = core.get_builtin_path() .. "intllib" .. DIR_DELIM
|
2020-08-06 19:26:15 +02:00
|
|
|
dofile(path .. "lib.lua")
|
|
|
|
|
|
|
|
local LANG = core.settings:get("language")
|
|
|
|
if not (LANG and (LANG ~= "")) then LANG = os.getenv("LANG") end
|
|
|
|
if not (LANG and (LANG ~= "")) then LANG = "en" end
|
|
|
|
|
|
|
|
|
|
|
|
local INS_CHAR = intllib.INSERTION_CHAR
|
|
|
|
local insertion_pattern = "("..INS_CHAR.."?)"..INS_CHAR.."(%(?)(%d+)(%)?)"
|
|
|
|
|
|
|
|
local function do_replacements(str, ...)
|
|
|
|
local args = {...}
|
|
|
|
-- Outer parens discard extra return values
|
|
|
|
return (str:gsub(insertion_pattern, function(escape, open, num, close)
|
|
|
|
if escape == "" then
|
|
|
|
local replacement = tostring(args[tonumber(num)])
|
|
|
|
if open == "" then
|
|
|
|
replacement = replacement..close
|
|
|
|
end
|
|
|
|
return replacement
|
|
|
|
else
|
|
|
|
return INS_CHAR..open..num..close
|
|
|
|
end
|
|
|
|
end))
|
|
|
|
end
|
|
|
|
|
|
|
|
local function make_getter(msgstrs)
|
|
|
|
return function(s, ...)
|
|
|
|
local str
|
|
|
|
if msgstrs then
|
|
|
|
str = msgstrs[s]
|
|
|
|
end
|
|
|
|
if not str or str == "" then
|
|
|
|
str = s
|
|
|
|
end
|
|
|
|
if select("#", ...) == 0 then
|
|
|
|
return str
|
|
|
|
end
|
|
|
|
return do_replacements(str, ...)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
local function Getter(modname)
|
|
|
|
modname = modname or core.get_current_modname()
|
|
|
|
if not intllib.getters[modname] then
|
|
|
|
local msgstr = intllib.get_strings(modname)
|
|
|
|
intllib.getters[modname] = make_getter(msgstr)
|
|
|
|
end
|
|
|
|
return intllib.getters[modname]
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
function intllib.Getter(modname)
|
|
|
|
local info = debug and debug.getinfo and debug.getinfo(2)
|
|
|
|
local loc = info and info.short_src..":"..info.currentline
|
|
|
|
core.log("deprecated", "intllib.Getter is deprecated."
|
|
|
|
.." Please use intllib.make_gettext_pair instead."
|
|
|
|
..(info and " (called from "..loc..")" or ""))
|
|
|
|
return Getter(modname)
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
local langs
|
|
|
|
|
|
|
|
local function split(str, sep)
|
|
|
|
local pos, endp = 1, #str+1
|
|
|
|
return function()
|
|
|
|
if (not pos) or pos > endp then return end
|
2021-12-23 10:54:09 +01:00
|
|
|
local s, e = str:find(sep, pos, true)
|
|
|
|
local part = str:sub(pos, s and s-1)
|
2020-08-06 19:26:15 +02:00
|
|
|
pos = e and e + 1
|
|
|
|
return part
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
function intllib.get_detected_languages()
|
|
|
|
if langs then return langs end
|
|
|
|
|
|
|
|
langs = { }
|
|
|
|
|
|
|
|
local function addlang(l)
|
|
|
|
local sep
|
|
|
|
langs[#langs+1] = l
|
2021-12-23 10:54:09 +01:00
|
|
|
sep = l:find(".", 1, true)
|
2020-08-06 19:26:15 +02:00
|
|
|
if sep then
|
2021-12-23 10:54:09 +01:00
|
|
|
l = l:sub(1, sep-1)
|
2020-08-06 19:26:15 +02:00
|
|
|
langs[#langs+1] = l
|
|
|
|
end
|
2021-12-23 10:54:09 +01:00
|
|
|
sep = l:find("_", 1, true)
|
2020-08-06 19:26:15 +02:00
|
|
|
if sep then
|
2021-12-23 10:54:09 +01:00
|
|
|
langs[#langs+1] = l:sub(1, sep-1)
|
2020-08-06 19:26:15 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local v
|
|
|
|
|
|
|
|
v = core.settings:get("language")
|
|
|
|
if v and v~="" then
|
|
|
|
addlang(v)
|
|
|
|
end
|
|
|
|
|
|
|
|
v = os.getenv("LANGUAGE")
|
|
|
|
if v then
|
|
|
|
for item in split(v, ":") do
|
|
|
|
langs[#langs+1] = item
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
v = os.getenv("LANG")
|
|
|
|
if v then
|
|
|
|
addlang(v)
|
|
|
|
end
|
|
|
|
|
|
|
|
langs[#langs+1] = "en"
|
|
|
|
|
|
|
|
return langs
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
local gettext = dofile(path .. "gettext.lua")
|
|
|
|
|
|
|
|
|
|
|
|
local function catgettext(catalogs, msgid)
|
|
|
|
for _, cat in ipairs(catalogs) do
|
|
|
|
local msgstr = cat and cat[msgid]
|
|
|
|
if msgstr and msgstr~="" then
|
|
|
|
local msg = msgstr[0]
|
|
|
|
return msg~="" and msg or nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-12-23 10:54:09 +01:00
|
|
|
local floor = math.floor
|
2020-08-06 19:26:15 +02:00
|
|
|
local function catngettext(catalogs, msgid, msgid_plural, n)
|
2021-12-23 10:54:09 +01:00
|
|
|
n = floor(n)
|
2020-08-06 19:26:15 +02:00
|
|
|
for _, cat in ipairs(catalogs) do
|
|
|
|
local msgstr = cat and cat[msgid]
|
|
|
|
if msgstr then
|
|
|
|
local index = cat.plural_index(n)
|
|
|
|
local msg = msgstr[index]
|
|
|
|
return msg~="" and msg or nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return n==1 and msgid or msgid_plural
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
local gettext_getters = { }
|
|
|
|
function intllib.make_gettext_pair(modname)
|
|
|
|
modname = modname or core.get_current_modname()
|
|
|
|
if gettext_getters[modname] then
|
|
|
|
return unpack(gettext_getters[modname])
|
|
|
|
end
|
|
|
|
local modpath = core.get_modpath(modname) and core.get_modpath(modname)
|
|
|
|
local localedir = modpath and modpath.."/locale"
|
|
|
|
local catalogs = localedir and gettext.load_catalogs(localedir) or {}
|
|
|
|
local getter = Getter(modname)
|
|
|
|
local function gettext_func(msgid, ...)
|
|
|
|
local msgstr = (catgettext(catalogs, msgid)
|
|
|
|
or getter(msgid))
|
|
|
|
return do_replacements(msgstr, ...)
|
|
|
|
end
|
|
|
|
local function ngettext_func(msgid, msgid_plural, n, ...)
|
|
|
|
local msgstr = (catngettext(catalogs, msgid, msgid_plural, n)
|
|
|
|
or getter(msgid))
|
|
|
|
return do_replacements(msgstr, ...)
|
|
|
|
end
|
|
|
|
gettext_getters[modname] = { gettext_func, ngettext_func }
|
|
|
|
return gettext_func, ngettext_func
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
local function get_locales(code)
|
|
|
|
local ll, cc = code:match("^(..)_(..)")
|
|
|
|
if ll then
|
|
|
|
return { ll.."_"..cc, ll, ll~="en" and "en" or nil }
|
|
|
|
else
|
|
|
|
return { code, code~="en" and "en" or nil }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
function intllib.get_strings(modname, langcode)
|
|
|
|
langcode = langcode or LANG
|
|
|
|
modname = modname or core.get_current_modname()
|
|
|
|
local msgstr = intllib.strings[modname]
|
|
|
|
if not msgstr then
|
|
|
|
local modpath = core.get_modpath(modname) and core.get_modpath(modname)
|
|
|
|
msgstr = { }
|
|
|
|
if modpath then
|
|
|
|
for _, l in ipairs(get_locales(langcode)) do
|
|
|
|
local t = intllib.load_strings(modpath.."/locale/"..modname.."."..l..".tr")
|
|
|
|
or intllib.load_strings(modpath.."/locale/"..l..".txt") or { }
|
|
|
|
for k, v in pairs(t) do
|
|
|
|
msgstr[k] = msgstr[k] or v
|
|
|
|
end
|
|
|
|
end
|
|
|
|
intllib.strings[modname] = msgstr
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return msgstr
|
|
|
|
end
|