mirror of
				https://github.com/cookiengineer/audacity
				synced 2025-10-26 15:23:48 +01:00 
			
		
		
		
	Finish copying msgid to msgstr for `en.po`. Rename POT file. Rename MO files generated from PO files. Signed-off-by: Emily Mabrey <emilymabrey93@gmail.com>
		
			
				
	
	
		
			410 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			410 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /**********************************************************************
 | |
| 
 | |
|   Tenacity: A Digital Audio Editor
 | |
| 
 | |
|   Languages.cpp
 | |
| 
 | |
|   Dominic Mazzoni
 | |
| 
 | |
| 
 | |
| *******************************************************************//*!
 | |
| 
 | |
| \file Languages.cpp
 | |
| \brief Determine installed languages.
 | |
| 
 | |
|   Figure out what translations are installed and return a list
 | |
|   of language codes (like "es", "fr", or "pt-br") and corresponding
 | |
|   language names (like "Español", "Français", and "Português").
 | |
|   We use our own list of translations of language names (i.e.
 | |
|   "Français" instead of "French") but we fallback on the language
 | |
|   name in wxWidgets if we don't have it listed.
 | |
| 
 | |
|   This code is designed to work well with all of the current
 | |
|   languages, but adapt to any language that wxWidgets supports.
 | |
|   Other languages will only be supported if they're added to
 | |
|   the database using wxLocale::AddLanguage.
 | |
| 
 | |
|   But for the most part, this means that somebody could add a NEW
 | |
|   translation and have it work immediately.
 | |
| 
 | |
| *//*******************************************************************/
 | |
| 
 | |
| 
 | |
| 
 | |
| #include "Languages.h"
 | |
| #include <memory>
 | |
| #include "wxArrayStringEx.h"
 | |
| 
 | |
| #include "Internat.h"
 | |
| #include "wxArrayStringEx.h"
 | |
| 
 | |
| #include <wx/defs.h>
 | |
| #include <wx/dir.h>
 | |
| #include <wx/filename.h>
 | |
| #include <wx/intl.h>
 | |
| #include <wx/textfile.h>
 | |
| #include <wx/utils.h> // for wxSetEnv
 | |
| 
 | |
| #include <clocale>
 | |
| #include <unordered_map>
 | |
| 
 | |
| using LangHash = std::unordered_map<wxString, TranslatableString>;
 | |
| using ReverseLangHash = std::unordered_map<TranslatableString, wxString>;
 | |
| 
 | |
| static void FindFilesInPathList(const wxString & pattern,
 | |
|    const FilePaths & pathList, FilePaths & results)
 | |
| {
 | |
|    wxFileName ff;
 | |
|    for (const auto &path : pathList) {
 | |
|       ff = path + wxFILE_SEP_PATH + pattern;
 | |
|       wxDir::GetAllFiles(ff.GetPath(), &results, ff.GetFullName(), wxDIR_FILES);
 | |
|    }
 | |
| }
 | |
| 
 | |
| static bool TranslationExists(const FilePaths &pathList, wxString code)
 | |
| {
 | |
|    FilePaths results;
 | |
|    FindFilesInPathList(code + L"/tenacity.mo", pathList, results);
 | |
| #if defined(__WXMAC__)
 | |
|    FindFilesInPathList(code + L".lproj/tenacity.mo", pathList, results);
 | |
| #endif
 | |
|    FindFilesInPathList(code + L"/LC_MESSAGES/tenacity.mo", pathList, results);
 | |
|    return (results.size() > 0);
 | |
| }
 | |
| 
 | |
| #ifdef __WXMAC__
 | |
| #include <CoreFoundation/CFLocale.h>
 | |
| #include <wx/osx/core/cfstring.h>
 | |
| #endif
 | |
| 
 | |
| namespace Languages {
 | |
| 
 | |
| wxString GetSystemLanguageCode(const FilePaths &pathList)
 | |
| {
 | |
|    wxArrayString langCodes;
 | |
|    TranslatableStrings langNames;
 | |
| 
 | |
|    GetLanguages(pathList, langCodes, langNames);
 | |
| 
 | |
|    int sysLang = wxLocale::GetSystemLanguage();
 | |
| 
 | |
|    const wxLanguageInfo *info;
 | |
| 
 | |
| #ifdef __WXMAC__
 | |
|    // PRL: Bug 1227, system language on Mac may not be right because wxW3 is
 | |
|    // dependent on country code too in wxLocale::GetSystemLanguage().
 | |
| 
 | |
|    if (sysLang == wxLANGUAGE_UNKNOWN)
 | |
|    {
 | |
|       // wxW3 did a too-specific lookup of language and country, when
 | |
|       // there is nothing for that combination; try it by language alone.
 | |
| 
 | |
|       // The following lines are cribbed from that function.
 | |
|       wxCFRef<CFLocaleRef> userLocaleRef(CFLocaleCopyCurrent());
 | |
|       wxCFStringRef str(wxCFRetain((CFStringRef)CFLocaleGetValue(userLocaleRef, kCFLocaleLanguageCode)));
 | |
|       auto lang = str.AsString();
 | |
| 
 | |
|       // Now avoid wxLocale::GetLanguageInfo(), instead calling:
 | |
|       info = wxLocale::FindLanguageInfo(lang);
 | |
|    }
 | |
|    else
 | |
| #endif
 | |
|    {
 | |
|       info = wxLocale::GetLanguageInfo(sysLang);
 | |
|    }
 | |
| 
 | |
|    if (info) {
 | |
|       wxString fullCode = info->CanonicalName;
 | |
|       if (fullCode.length() < 2)
 | |
|          return wxT("en");
 | |
| 
 | |
|       wxString code = fullCode.Left(2);
 | |
|       unsigned int i;
 | |
| 
 | |
|       for(i=0; i<langCodes.size(); i++) {
 | |
|          if (langCodes[i] == fullCode)
 | |
|             return fullCode;
 | |
| 
 | |
|          if (langCodes[i] == code)
 | |
|             return code;
 | |
|       }
 | |
|    }
 | |
| 
 | |
|    return wxT("en");
 | |
| }
 | |
| 
 | |
| void GetLanguages( FilePaths pathList,
 | |
|    wxArrayString &langCodes, TranslatableStrings &langNames)
 | |
| {
 | |
|    static const char *const utf8Names[] = {
 | |
| "af Afrikaans",
 | |
| "ar \330\247\331\204\330\271\330\261\330\250\331\212\330\251",
 | |
| "be \320\221\320\265\320\273\320\260\321\200\321\203\321\201\320\272\320\260\321\217",
 | |
| "bg \320\221\321\212\320\273\320\263\320\260\321\200\321\201\320\272\320\270",
 | |
| "bn \340\246\254\340\246\276\340\246\202\340\246\262\340\246\276",
 | |
| "bs Bosanski",
 | |
| "ca Catal\303\240",
 | |
| "ca_ES@valencia Valenci\303\240",
 | |
| "co Corsu",
 | |
| "cs \304\214e\305\241tina",
 | |
| "cy Cymraeg",
 | |
| "da Dansk",
 | |
| "de Deutsch",
 | |
| "el \316\225\316\273\316\273\316\267\316\275\316\271\316\272\316\254",
 | |
| "en English",
 | |
| "es Espa\303\261ol",
 | |
| "eu Euskara",
 | |
| "eu_ES Euskara (Espainiako)",
 | |
| "fa \331\201\330\247\330\261\330\263\333\214",
 | |
| "fi Suomi",
 | |
| "fr Fran\303\247ais",
 | |
| "ga Gaeilge",
 | |
| "gl Galego",
 | |
| "he \327\242\327\221\327\250\327\231\327\252",
 | |
| "hi \340\244\271\340\244\277\340\244\250\340\245\215\340\244\246\340\245\200",
 | |
| "hr Hrvatski",
 | |
| "hu Magyar",
 | |
| "hy \325\200\325\241\325\265\325\245\326\200\325\245\325\266",
 | |
| "id Bahasa Indonesia",
 | |
| "it Italiano",
 | |
| "ja \346\227\245\346\234\254\350\252\236",
 | |
| "ka \341\203\245\341\203\220\341\203\240\341\203\227\341\203\243\341\203\232\341\203\230",
 | |
| "km \341\236\201\341\237\201\341\236\230\341\236\232\341\236\227\341\236\266\341\236\237\341\236\266",
 | |
| "ko \355\225\234\352\265\255\354\226\264",
 | |
| "lt Lietuvi\305\263",
 | |
| "mk \320\234\320\260\320\272\320\265\320\264\320\276\320\275\321\201\320\272\320\270",
 | |
| "mr \340\244\256\340\244\260\340\244\276\340\244\240\340\245\200",
 | |
| "my \341\200\231\341\200\274\341\200\224\341\200\272\341\200\231\341\200\254\341\200\205\341\200\254",
 | |
| "nb Norsk",
 | |
| "nl Nederlands",
 | |
| "oc Occitan",
 | |
| "pl Polski",
 | |
| "pt Portugu\303\252s",
 | |
| "pt_BR Portugu\303\252s (Brasil)",
 | |
| "ro Rom\303\242n\304\203",
 | |
| "ru \320\240\321\203\321\201\321\201\320\272\320\270\320\271",
 | |
| "sk Sloven\304\215ina",
 | |
| "sl Sloven\305\241\304\215ina",
 | |
| "sr_RS \320\241\321\200\320\277\321\201\320\272\320\270",
 | |
| "sr_RS@latin Srpski",
 | |
| "sv Svenska",
 | |
| "ta \340\256\244\340\256\256\340\256\277\340\256\264\340\257\215",
 | |
| "tg \320\242\320\276\322\267\320\270\320\272\323\243",
 | |
| "tr T\303\274rk\303\247e",
 | |
| "uk \320\243\320\272\321\200\320\260\321\227\320\275\321\201\321\214\320\272\320\260",
 | |
| "vi Ti\341\272\277ng Vi\341\273\207t",
 | |
| "zh_CN \344\270\255\346\226\207\357\274\210\347\256\200\344\275\223\357\274\211",
 | |
| "zh_TW \344\270\255\346\226\207\357\274\210\347\271\201\351\253\224\357\274\211",
 | |
|    };
 | |
| 
 | |
|    TranslatableStrings tempNames;
 | |
|    wxArrayString tempCodes;
 | |
|    ReverseLangHash reverseHash;
 | |
|    LangHash tempHash;
 | |
| 
 | |
|    const LangHash localLanguageName = []{
 | |
|       LangHash localLanguageName;
 | |
|       for ( auto utf8Name : utf8Names )
 | |
|       {
 | |
|          auto str = wxString::FromUTF8(utf8Name);
 | |
|          auto code = str.BeforeFirst(' ');
 | |
|          auto name = str.AfterFirst(' ');
 | |
|          localLanguageName[code] = Verbatim( name );
 | |
|       }
 | |
|       return localLanguageName;
 | |
|    }();
 | |
| 
 | |
| #if defined(__WXGTK__)
 | |
|    {
 | |
|       wxFileName pathNorm{ wxString{INSTALL_PREFIX} + L"/share/locale" };
 | |
|       pathNorm.Normalize();
 | |
|       const wxString newPath{ pathNorm.GetFullPath() };
 | |
|       if (pathList.end() ==
 | |
|           std::find(pathList.begin(), pathList.end(), newPath))
 | |
|          pathList.push_back(newPath);
 | |
|    }
 | |
| #endif
 | |
| 
 | |
|    // For each language in our list we look for a corresponding entry in
 | |
|    // wxLocale.  
 | |
|    for ( auto end = localLanguageName.end(), i = localLanguageName.begin();
 | |
|       i != end; ++i )
 | |
|    {
 | |
|       const wxLanguageInfo *info = wxLocale::FindLanguageInfo(i->first);
 | |
| 
 | |
|       if (!info) {
 | |
|          wxASSERT(info != NULL);
 | |
|          continue;
 | |
|       }
 | |
| 
 | |
|       wxString fullCode = info->CanonicalName;
 | |
|       wxString code = fullCode.Left(2);
 | |
|       auto name = Verbatim( info->Description );
 | |
| 
 | |
|       // Logic: Languages codes are sometimes hierarchical, with a
 | |
|       // general language code and then a subheading.  For example,
 | |
|       // zh_TW for Traditional Chinese and zh_CN for Simplified
 | |
|       // Chinese - but just zh for Chinese in general.  First we look
 | |
|       // for the full code, like zh_TW.  If that doesn't exist, we
 | |
|       // look for a code corresponding to the first two letters.
 | |
|       // Note that if the language for a fullCode exists but we only
 | |
|       // have a name for the short code, we will use the short code's
 | |
|       // name but associate it with the full code.  This allows someone
 | |
|       // to drop in a NEW language and still get reasonable behavior.
 | |
| 
 | |
|       if (fullCode.length() < 2)
 | |
|          continue;
 | |
| 
 | |
|       auto found = localLanguageName.find( code );
 | |
|       if ( found != end ) {
 | |
|          name = found->second;
 | |
|       }
 | |
|       found = localLanguageName.find( fullCode );
 | |
|       if ( found != end ) {
 | |
|          name = found->second;
 | |
|       }
 | |
| 
 | |
|       if (TranslationExists(pathList, fullCode)) {
 | |
|          code = fullCode;
 | |
|       }
 | |
| 
 | |
|       if (!tempHash[code].empty())
 | |
|          continue;
 | |
| 
 | |
|       if (TranslationExists(pathList, code) || code==wxT("en")) {
 | |
|          tempCodes.push_back(code);
 | |
|          tempNames.push_back(name);
 | |
|          tempHash[code] = name;
 | |
| 
 | |
| /*         wxLogDebug(wxT("code=%s name=%s fullCode=%s name=%s -> %s"),
 | |
|                       code, localLanguageName[code],
 | |
|                       fullCode, localLanguageName[fullCode],
 | |
|                       name);*/
 | |
|       }
 | |
|    }
 | |
| 
 | |
|    // JKC: Adding language for simplified tenacity.
 | |
|    {
 | |
|       wxString code;
 | |
|       code = wxT("en-simple");
 | |
|       auto name = XO("Simplified");
 | |
|       if (TranslationExists(pathList, code) ) {
 | |
|          tempCodes.push_back(code);
 | |
|          tempNames.push_back(name);
 | |
|          tempHash[code] = name;
 | |
|       }
 | |
|    }
 | |
| 
 | |
| 
 | |
|    // Sort
 | |
|    unsigned int j;
 | |
|    for(j=0; j<tempNames.size(); j++){
 | |
|       reverseHash[tempNames[j]] = tempCodes[j];
 | |
|    }
 | |
| 
 | |
|    std::sort( tempNames.begin(), tempNames.end(),
 | |
|       []( const TranslatableString &a, const TranslatableString &b ){
 | |
|          return a.Translation() < b.Translation();
 | |
|       } );
 | |
| 
 | |
|    // Add system language
 | |
|    langNames.push_back(XO("System"));
 | |
|    langCodes.push_back(wxT("System"));
 | |
| 
 | |
|    for(j=0; j<tempNames.size(); j++) {
 | |
|       langNames.push_back(tempNames[j]);
 | |
|       langCodes.push_back(reverseHash[tempNames[j]]);
 | |
|    }
 | |
| }
 | |
| 
 | |
| static std::unique_ptr<wxLocale> sLocale;
 | |
| static wxString sLocaleName;
 | |
| 
 | |
| wxString SetLang( const FilePaths &pathList, const wxString & lang )
 | |
| {
 | |
|    wxString result = lang;
 | |
| 
 | |
|    sLocale.reset();
 | |
| 
 | |
| #if defined(__WXMAC__)
 | |
|    // This should be reviewed again during the wx3 conversion.
 | |
| 
 | |
|    // On OSX, if the LANG environment variable isn't set when
 | |
|    // using a language like Japanese, an assertion will trigger
 | |
|    // because conversion to Japanese from "?" doesn't return a
 | |
|    // valid length, so make OSX happy by defining/overriding
 | |
|    // the LANG environment variable with U.S. English for now.
 | |
|    wxSetEnv(wxT("LANG"), wxT("en_US.UTF-8"));
 | |
| #endif
 | |
| 
 | |
|    const wxLanguageInfo *info = NULL;
 | |
|    if (!lang.empty() && lang != wxT("System")) {
 | |
|       // Try to find the given language
 | |
|       info = wxLocale::FindLanguageInfo(lang);
 | |
|    }
 | |
|    if (!info)
 | |
|    {
 | |
|       // Not given a language or can't find it; substitute the system language
 | |
|       result = Languages::GetSystemLanguageCode(pathList);
 | |
|       info = wxLocale::FindLanguageInfo(result);
 | |
|       if (!info)
 | |
|          // Return the substituted system language, but we can't complete setup
 | |
|          // Should we try to do something better?
 | |
|          return result;
 | |
|    }
 | |
|    sLocale = std::make_unique<wxLocale>(info->Language);
 | |
| 
 | |
|    for( const auto &path : pathList )
 | |
|       sLocale->AddCatalogLookupPathPrefix( path );
 | |
| 
 | |
|    // LL:  Must add the wxWidgets catalog manually since the search
 | |
|    //      paths were not set up when mLocale was created.  The
 | |
|    //      catalogs are search in LIFO order, so add wxstd first.
 | |
|    sLocale->AddCatalog(wxT("wxstd"));
 | |
| 
 | |
|    // Must match TranslationExists() in Languages.cpp
 | |
|    sLocale->AddCatalog("tenacity");
 | |
| 
 | |
|    // Initialize internationalisation (number formats etc.)
 | |
|    //
 | |
|    // This must go _after_ creating the wxLocale instance because
 | |
|    // creating the wxLocale instance sets the application-wide locale.
 | |
| 
 | |
|    Internat::Init();
 | |
| 
 | |
|    using future1 = decltype(
 | |
|       // The file of unused strings is part of the source tree scanned by
 | |
|       // xgettext when compiling the catalog template tenacity.pot.
 | |
|       // Including it here doesn't change that but does make the C++ compiler
 | |
|       // check for correct syntax, but also generate no object code for them.
 | |
| #include "FutureStrings.h"
 | |
|       0
 | |
|    );
 | |
| 
 | |
|    sLocaleName = wxSetlocale(LC_ALL, NULL);
 | |
| 
 | |
|    return result;
 | |
| }
 | |
| 
 | |
| wxString GetLocaleName()
 | |
| {
 | |
|    return sLocaleName;
 | |
| }
 | |
| 
 | |
| wxString GetLang()
 | |
| {
 | |
|    if (sLocale)
 | |
|       return sLocale->GetSysName();
 | |
|    else
 | |
|       return {};
 | |
| }
 | |
| 
 | |
| wxString GetLangShort()
 | |
| {
 | |
|    if (sLocale)
 | |
|       return sLocale->GetName();
 | |
|    else
 | |
|       return {};
 | |
| }
 | |
| }
 |