Merge pull request #3073 from adamreichold/optimize-multilangmgr
Simplify the multiple language manager for better maintainability
This commit is contained in:
commit
0fca97b725
@ -16,111 +16,151 @@
|
|||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with RawTherapee. If not, see <http://www.gnu.org/licenses/>.
|
* along with RawTherapee. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
#include "multilangmgr.h"
|
||||||
|
|
||||||
|
#include <fstream>
|
||||||
|
#include <regex>
|
||||||
|
|
||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
// Desired auto detect function is Vista+
|
|
||||||
#if (__GNUC__ == 4 && __GNUC_MINOR__ >= 8) || __GNUC__ > 4
|
|
||||||
#define WINVER 0x0600 // switching to WINVER for gcc 4.8.1 support on Winx64
|
|
||||||
#else
|
|
||||||
#define _WIN32_WINNT 0x0600
|
|
||||||
#endif
|
|
||||||
#include <windows.h>
|
#include <windows.h>
|
||||||
#include <winnls.h>
|
#include <winnls.h>
|
||||||
#if (__GNUC__ == 4 && __GNUC_MINOR__ >= 8) || __GNUC__ > 4
|
|
||||||
#undef WINVER
|
|
||||||
#else
|
|
||||||
#undef _WIN32_WINNT
|
|
||||||
#endif
|
#endif
|
||||||
#endif
|
|
||||||
#include <glib/gstdio.h>
|
namespace
|
||||||
#include "multilangmgr.h"
|
{
|
||||||
#include <cstring>
|
|
||||||
#include "../rtengine/safegtk.h"
|
// Maps standard locales to languages, e.g. "de-DE" to "Deutsch".
|
||||||
|
struct LocaleToLang : private std::map<std::pair<Glib::ustring, Glib::ustring>, Glib::ustring>
|
||||||
|
{
|
||||||
|
static const std::pair<Glib::ustring, Glib::ustring> key (const Glib::ustring& major, const Glib::ustring& minor = Glib::ustring ())
|
||||||
|
{
|
||||||
|
return std::make_pair (major, minor);
|
||||||
|
}
|
||||||
|
|
||||||
|
LocaleToLang ()
|
||||||
|
{
|
||||||
|
emplace (key ("ca"), "Catala");
|
||||||
|
emplace (key ("cs"), "Czech");
|
||||||
|
emplace (key ("da"), "Dansk");
|
||||||
|
emplace (key ("de"), "Deutsch");
|
||||||
|
emplace (key ("es"), "Espanol");
|
||||||
|
emplace (key ("eu"), "Euskara");
|
||||||
|
emplace (key ("fr"), "Francais");
|
||||||
|
emplace (key ("el"), "Greek");
|
||||||
|
emplace (key ("he"), "Hebrew");
|
||||||
|
emplace (key ("it"), "Italiano");
|
||||||
|
emplace (key ("ja"), "Japanese");
|
||||||
|
emplace (key ("lv"), "Latvian");
|
||||||
|
emplace (key ("hu"), "Magyar");
|
||||||
|
emplace (key ("nl"), "Nederlands");
|
||||||
|
emplace (key ("nn"), "Norsk BM");
|
||||||
|
emplace (key ("nb"), "Norsk BM");
|
||||||
|
emplace (key ("pl"), "Polish");
|
||||||
|
emplace (key ("pt"), "Portugues (Brasil)");
|
||||||
|
emplace (key ("ru"), "Russian");
|
||||||
|
emplace (key ("sr"), "Serbian (Cyrilic Characters)");
|
||||||
|
emplace (key ("sk"), "Slovak");
|
||||||
|
emplace (key ("fi"), "Suomi");
|
||||||
|
emplace (key ("sv"), "Swedish");
|
||||||
|
emplace (key ("tr"), "Turkish");
|
||||||
|
emplace (key ("zh", "CN"), "Chinese (Simplified)");
|
||||||
|
emplace (key ("zh", "SG"), "Chinese (Traditional)");
|
||||||
|
}
|
||||||
|
|
||||||
|
Glib::ustring operator() (const Glib::ustring& locale) const
|
||||||
|
{
|
||||||
|
Glib::ustring major, minor;
|
||||||
|
|
||||||
|
if (locale.length () >= 2) {
|
||||||
|
major = locale.substr (0, 2).lowercase ();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (locale.length () >= 5) {
|
||||||
|
minor = locale.substr (3, 2).uppercase ();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Look for matching language and country.
|
||||||
|
auto iterator = find (key (major, minor));
|
||||||
|
|
||||||
|
if (iterator != end ()) {
|
||||||
|
return iterator->second;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Look for matching language only.
|
||||||
|
iterator = find (key (major));
|
||||||
|
|
||||||
|
if (iterator != end ()) {
|
||||||
|
return iterator->second;
|
||||||
|
}
|
||||||
|
|
||||||
|
return "default";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const LocaleToLang localeToLang;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
MultiLangMgr langMgr;
|
MultiLangMgr langMgr;
|
||||||
|
|
||||||
Glib::ustring M (std::string key)
|
MultiLangMgr::MultiLangMgr ()
|
||||||
{
|
{
|
||||||
return langMgr.getStr (key);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// fb is fallback manager if the first could not be loaded
|
MultiLangMgr::MultiLangMgr (const Glib::ustring& fname, MultiLangMgr* fallbackMgr)
|
||||||
bool MultiLangMgr::load (Glib::ustring fname, MultiLangMgr* fb)
|
|
||||||
{
|
{
|
||||||
FILE *f = safe_g_fopen (fname, "rt");
|
load (fname, fallbackMgr);
|
||||||
|
}
|
||||||
|
|
||||||
fallBack = fb;
|
bool MultiLangMgr::load (const Glib::ustring& fname, MultiLangMgr* fallbackMgr)
|
||||||
|
{
|
||||||
|
this->fallbackMgr.reset (fallbackMgr);
|
||||||
|
|
||||||
if (f == NULL) {
|
std::ifstream file (fname.c_str ());
|
||||||
|
if (!file.is_open ()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
transTable.clear ();
|
std::map<std::string, Glib::ustring> translations;
|
||||||
|
std::string entry, key, value;
|
||||||
|
|
||||||
char* buffer = new char[2048];
|
while (std::getline (file, entry)) {
|
||||||
|
|
||||||
while (fgets (buffer, 2048, f) != 0) {
|
if (entry.empty () || entry.front () == '#') {
|
||||||
// find separator
|
|
||||||
int seppos = 0;
|
|
||||||
|
|
||||||
while (buffer[seppos] != 0 && buffer[seppos] != ';') {
|
|
||||||
seppos++;
|
|
||||||
}
|
|
||||||
|
|
||||||
// no separator found
|
|
||||||
if (buffer[seppos] == 0) {
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
// cut the last \n and \r characters
|
std::istringstream line (entry);
|
||||||
int endpos = strlen(buffer) - 1;
|
|
||||||
|
|
||||||
while (buffer[endpos] == '\n' || buffer[endpos] == '\r') {
|
if (!std::getline (line, key, ';') || !std::getline (line, value)) {
|
||||||
endpos--;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
buffer[endpos + 1] = 0;
|
static const std::regex newline ("\\\\n");
|
||||||
// replace "\n" to '\n'
|
value = std::regex_replace (value, newline, "\n");
|
||||||
int j = 0;
|
|
||||||
|
|
||||||
for (int i = 0; i < endpos + 1; i++)
|
translations.emplace (key, value);
|
||||||
if (i < endpos && buffer[i] == '\\' && buffer[i + 1] == 'n') {
|
|
||||||
buffer[j++] = '\n';
|
|
||||||
i++;
|
|
||||||
} else {
|
|
||||||
buffer[j++] = buffer[i];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
buffer[j] = 0;
|
this->translations.swap (translations);
|
||||||
// cut to two parts
|
|
||||||
buffer[seppos] = 0;
|
|
||||||
transTable[buffer] = buffer + seppos + 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
fclose (f);
|
|
||||||
delete [] buffer;
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool MultiLangMgr::save (Glib::ustring fname)
|
Glib::ustring MultiLangMgr::getStr (const std::string& key) const
|
||||||
{
|
{
|
||||||
|
const auto iterator = translations.find (key);
|
||||||
|
|
||||||
FILE *f = safe_g_fopen (fname, "wt");
|
if (iterator != translations.end ()) {
|
||||||
|
return iterator->second;
|
||||||
if (f == NULL) {
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
std::map<std::string, Glib::ustring>::iterator r;
|
if (fallbackMgr) {
|
||||||
|
return fallbackMgr->getStr (key);
|
||||||
for (r = transTable.begin (); r != transTable.end(); r++) {
|
|
||||||
fprintf (f, "%s;%s\n", r->first.c_str(), safe_locale_from_utf8(r->second).c_str());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fclose (f);
|
return key;
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
bool MultiLangMgr::isOSLanguageDetectSupported ()
|
bool MultiLangMgr::isOSLanguageDetectSupported ()
|
||||||
{
|
{
|
||||||
#if defined (WIN32) || defined (__linux__) || defined (__APPLE__)
|
#if defined (WIN32) || defined (__linux__) || defined (__APPLE__)
|
||||||
@ -130,8 +170,6 @@ bool MultiLangMgr::isOSLanguageDetectSupported()
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// returns Language name mapped from the currently selected OS language
|
|
||||||
Glib::ustring MultiLangMgr::getOSUserLanguage ()
|
Glib::ustring MultiLangMgr::getOSUserLanguage ()
|
||||||
{
|
{
|
||||||
Glib::ustring langName ("default");
|
Glib::ustring langName ("default");
|
||||||
@ -148,153 +186,23 @@ Glib::ustring MultiLangMgr::getOSUserLanguage()
|
|||||||
|
|
||||||
localeName[langLen - 1] = '-';
|
localeName[langLen - 1] = '-';
|
||||||
|
|
||||||
const int countryLen = GetLocaleInfo (localeID, LOCALE_SISO3166CTRYNAME, localeName + langLen, 9);
|
const int countryLen = GetLocaleInfo (localeID, LOCALE_SISO3166CTRYNAME, &localeName[langLen], 9);
|
||||||
if (countryLen <= 0) {
|
if (countryLen <= 0) {
|
||||||
return langName;
|
return langName;
|
||||||
}
|
}
|
||||||
|
|
||||||
langName = TranslateRFC2Language (localeName);
|
langName = localeToLang (localeName);
|
||||||
|
|
||||||
#elif defined (__linux__) || defined (__APPLE__)
|
#elif defined (__linux__) || defined (__APPLE__)
|
||||||
|
|
||||||
const char* locale = setlocale(LC_CTYPE, "");
|
// Query the current locale and force decimal point to dot.
|
||||||
setlocale(LC_NUMERIC, "C"); // to set decimal point to "."
|
if (const char* locale = setlocale (LC_CTYPE, "")) {
|
||||||
|
langName = localeToLang (locale);
|
||||||
if (locale) {
|
|
||||||
langName = TranslateRFC2Language (locale);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
setlocale (LC_NUMERIC, "C");
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return langName;
|
return langName;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Translates RFC standard language code to file name, e.g. "de-DE" to "Deutsch"
|
|
||||||
Glib::ustring MultiLangMgr::TranslateRFC2Language(Glib::ustring rfcName)
|
|
||||||
{
|
|
||||||
if (rfcName.length() < 2) {
|
|
||||||
return Glib::ustring("default");
|
|
||||||
}
|
|
||||||
|
|
||||||
Glib::ustring major = rfcName.substr(0, 2).lowercase();
|
|
||||||
Glib::ustring minor;
|
|
||||||
|
|
||||||
if (rfcName.length() >= 5) {
|
|
||||||
minor = rfcName.substr(3, 2).uppercase();
|
|
||||||
}
|
|
||||||
|
|
||||||
//printf("Lang: %s - %s\n",major.c_str(),minor.c_str());
|
|
||||||
|
|
||||||
if (major == "ca") {
|
|
||||||
return "Catala";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "zh") {
|
|
||||||
return (minor == "CN" || minor == "SG") ? "Chinese (Simplified)" : "Chinese (Traditional)";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "cs") {
|
|
||||||
return "Czech";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "da") {
|
|
||||||
return "Dansk";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "de") {
|
|
||||||
return "Deutsch";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "es") {
|
|
||||||
return "Espanol";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "eu") {
|
|
||||||
return "Euskara";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "fr") {
|
|
||||||
return "Francais";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "el") {
|
|
||||||
return "Greek";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "he") {
|
|
||||||
return "Hebrew";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "it") {
|
|
||||||
return "Italiano";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "ja") {
|
|
||||||
return "Japanese";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "lv") {
|
|
||||||
return "Latvian";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "hu") {
|
|
||||||
return "Magyar";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "nl") {
|
|
||||||
return "Nederlands";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "nn" || major == "nb") {
|
|
||||||
return "Norsk BM";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "pl") {
|
|
||||||
return "Polish";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "pt") {
|
|
||||||
return "Portugues (Brasil)";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "ru") {
|
|
||||||
return "Russian";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "sr") {
|
|
||||||
return "Serbian (Cyrilic Characters)";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "sk") {
|
|
||||||
return "Slovak";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "fi") {
|
|
||||||
return "Suomi";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "sv") {
|
|
||||||
return "Swedish";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (major == "tr") {
|
|
||||||
return "Turkish";
|
|
||||||
}
|
|
||||||
|
|
||||||
// Don't split en-US, en-GB, etc. since only default english is constantly updated
|
|
||||||
return "default";
|
|
||||||
}
|
|
||||||
|
|
||||||
Glib::ustring MultiLangMgr::getStr (std::string key)
|
|
||||||
{
|
|
||||||
|
|
||||||
std::map<std::string, Glib::ustring>::iterator r = transTable.find (key);
|
|
||||||
|
|
||||||
if (r != transTable.end()) {
|
|
||||||
return r->second;
|
|
||||||
} else if (fallBack) {
|
|
||||||
return fallBack->getStr (key);
|
|
||||||
} else {
|
|
||||||
return key;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -20,38 +20,38 @@
|
|||||||
#define _MULTILANGMGR_
|
#define _MULTILANGMGR_
|
||||||
|
|
||||||
#include <map>
|
#include <map>
|
||||||
|
#include <memory>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <glibmm.h>
|
|
||||||
|
#include <glibmm/ustring.h>
|
||||||
|
|
||||||
class MultiLangMgr
|
class MultiLangMgr
|
||||||
{
|
{
|
||||||
|
public:
|
||||||
std::map<std::string, Glib::ustring> transTable;
|
MultiLangMgr ();
|
||||||
MultiLangMgr* fallBack;
|
MultiLangMgr (const Glib::ustring& fname, MultiLangMgr* fallbackMgr = nullptr);
|
||||||
|
|
||||||
Glib::ustring TranslateRFC2Language(Glib::ustring rfcName);
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
MultiLangMgr () : fallBack (NULL) {}
|
bool load (const Glib::ustring& fname, MultiLangMgr* fallbackMgr = nullptr);
|
||||||
MultiLangMgr (Glib::ustring fname) : fallBack (NULL)
|
|
||||||
{
|
|
||||||
load (fname);
|
|
||||||
}
|
|
||||||
MultiLangMgr (Glib::ustring fname, MultiLangMgr* fb) : fallBack (NULL)
|
|
||||||
{
|
|
||||||
load (fname, fb);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool load (Glib::ustring fname, MultiLangMgr* fb = NULL);
|
public:
|
||||||
bool save (Glib::ustring fname);
|
Glib::ustring getStr (const std::string& key) const;
|
||||||
|
|
||||||
|
public:
|
||||||
|
static bool isOSLanguageDetectSupported ();
|
||||||
|
static Glib::ustring getOSUserLanguage ();
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::map<std::string, Glib::ustring> translations;
|
||||||
|
std::unique_ptr<MultiLangMgr> fallbackMgr;
|
||||||
|
|
||||||
bool isOSLanguageDetectSupported();
|
|
||||||
Glib::ustring getOSUserLanguage();
|
|
||||||
Glib::ustring getStr (std::string key);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
extern MultiLangMgr langMgr;
|
extern MultiLangMgr langMgr;
|
||||||
|
|
||||||
Glib::ustring M (std::string key);
|
inline Glib::ustring M (const std::string& key)
|
||||||
|
{
|
||||||
|
return langMgr.getStr (key);
|
||||||
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
Loading…
x
Reference in New Issue
Block a user