X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/076c0a8ee8c9238af2037b66ef97bda7cede4e4a..c0c05e79b20d6ca372ece6ac5995b0e3db18a29b:/src/common/translation.cpp diff --git a/src/common/translation.cpp b/src/common/translation.cpp index 6ea7c334ac..1e89106f9a 100644 --- a/src/common/translation.cpp +++ b/src/common/translation.cpp @@ -45,7 +45,6 @@ #include "wx/filename.h" #include "wx/tokenzr.h" #include "wx/fontmap.h" -#include "wx/scopedptr.h" #include "wx/stdpaths.h" #include "wx/hashset.h" @@ -65,29 +64,22 @@ const size_t32 MSGCATALOG_MAGIC_SW = 0xde120495; #define TRACE_I18N wxS("i18n") -// the constants describing the format of ll_CC locale string -static const size_t LEN_LANG = 2; - -// ---------------------------------------------------------------------------- -// global functions -// ---------------------------------------------------------------------------- +// ============================================================================ +// implementation +// ============================================================================ namespace { -// get just the language part -inline wxString ExtractLang(const wxString& langFull) -{ - return langFull.Left(LEN_LANG); -} +#if !wxUSE_UNICODE +// We need to keep track of (char*) msgids in non-Unicode legacy builds. Instead +// of making the public wxMsgCatalog and wxTranslationsLoader APIs ugly, we +// store them in this global map. +wxStringToStringHashMap gs_msgIdCharset; +#endif } // anonymous namespace - -// ============================================================================ -// implementation -// ============================================================================ - // ---------------------------------------------------------------------------- // Plural forms parser // ---------------------------------------------------------------------------- @@ -320,10 +312,10 @@ class wxPluralFormsNode public: wxPluralFormsNode(const wxPluralFormsToken& token) : m_token(token) {} const wxPluralFormsToken& token() const { return m_token; } - const wxPluralFormsNode* node(size_t i) const + const wxPluralFormsNode* node(unsigned i) const { return m_nodes[i].get(); } - void setNode(size_t i, wxPluralFormsNode* n); - wxPluralFormsNode* releaseNode(size_t i); + void setNode(unsigned i, wxPluralFormsNode* n); + wxPluralFormsNode* releaseNode(unsigned i); wxPluralFormsToken::Number evaluate(wxPluralFormsToken::Number n) const; private: @@ -351,12 +343,12 @@ void wxPluralFormsNodePtr::reset(wxPluralFormsNode *p) } -void wxPluralFormsNode::setNode(size_t i, wxPluralFormsNode* n) +void wxPluralFormsNode::setNode(unsigned i, wxPluralFormsNode* n) { m_nodes[i].reset(n); } -wxPluralFormsNode* wxPluralFormsNode::releaseNode(size_t i) +wxPluralFormsNode* wxPluralFormsNode::releaseNode(unsigned i) { return m_nodes[i].release(); } @@ -433,7 +425,7 @@ private: wxPluralFormsNodePtr m_plural; }; -wxDEFINE_SCOPED_PTR_TYPE(wxPluralFormsCalculator) +wxDEFINE_SCOPED_PTR(wxPluralFormsCalculator, wxPluralFormsCalculatorPtr) void wxPluralFormsCalculator::init(wxPluralFormsToken::Number nplurals, wxPluralFormsNode* plural) @@ -804,21 +796,23 @@ wxPluralFormsCalculator* wxPluralFormsCalculator::make(const char* s) // http://www.gnu.org/software/autoconf/manual/gettext/MO-Files.html // ---------------------------------------------------------------------------- -WX_DECLARE_EXPORTED_STRING_HASH_MAP(wxString, wxMessagesHash); - class wxMsgCatalogFile { public: + typedef wxScopedCharBuffer DataBuffer; + // ctor & dtor wxMsgCatalogFile(); ~wxMsgCatalogFile(); // load the catalog from disk - bool Load(const wxString& filename, - wxPluralFormsCalculatorPtr& rPluralFormsCalculator); + bool LoadFile(const wxString& filename, + wxPluralFormsCalculatorPtr& rPluralFormsCalculator); + bool LoadData(const DataBuffer& data, + wxPluralFormsCalculatorPtr& rPluralFormsCalculator); // fills the hash with string-translation pairs - bool FillHash(wxMessagesHash& hash, const wxString& msgIdCharset) const; + bool FillHash(wxStringToStringHashMap& hash, const wxString& domain) const; // return the charset of the strings in this catalog or empty string if // none/unknown @@ -847,7 +841,7 @@ private: }; // all data is stored here - wxMemoryBuffer m_data; + DataBuffer m_data; // data description size_t32 m_numStrings; // number of strings in this domain @@ -865,25 +859,18 @@ private: : ui; } - // just return the pointer to the start of the data as "char *" to - // facilitate doing pointer arithmetic with it - char *StringData() const - { - return static_cast(m_data.GetData()); - } - const char *StringAtOfs(wxMsgTableEntry *pTable, size_t32 n) const { const wxMsgTableEntry * const ent = pTable + n; // this check could fail for a corrupt message catalog size_t32 ofsString = Swap(ent->ofsString); - if ( ofsString + Swap(ent->nLen) > m_data.GetDataLen()) + if ( ofsString + Swap(ent->nLen) > m_data.length()) { return NULL; } - return StringData() + ofsString; + return m_data.data() + ofsString; } bool m_bSwapped; // wrong endianness? @@ -891,49 +878,6 @@ private: wxDECLARE_NO_COPY_CLASS(wxMsgCatalogFile); }; - -// ---------------------------------------------------------------------------- -// wxMsgCatalog corresponds to one loaded message catalog. -// -// This is a "low-level" class and is used only by wxLocale (that's why -// it's designed to be stored in a linked list) -// ---------------------------------------------------------------------------- - -class wxMsgCatalog -{ -public: -#if !wxUSE_UNICODE - wxMsgCatalog() { m_conv = NULL; } - ~wxMsgCatalog(); -#endif - - // load the catalog from disk - bool Load(const wxString& filename, - const wxString& domain, - const wxString& msgIdCharset); - - // get name of the catalog - wxString GetDomain() const { return m_domain; } - - // get the translated string: returns NULL if not found - const wxString *GetString(const wxString& sz, size_t n = size_t(-1)) const; - - // public variable pointing to the next element in a linked list (or NULL) - wxMsgCatalog *m_pNext; - -private: - wxMessagesHash m_messages; // all messages in the catalog - wxString m_domain; // name of the domain - -#if !wxUSE_UNICODE - // the conversion corresponding to this catalog charset if we installed it - // as the global one - wxCSConv *m_conv; -#endif - - wxPluralFormsCalculatorPtr m_pluralFormsCalculator; -}; - // ---------------------------------------------------------------------------- // wxMsgCatalogFile clas // ---------------------------------------------------------------------------- @@ -947,8 +891,8 @@ wxMsgCatalogFile::~wxMsgCatalogFile() } // open disk file and read in it's contents -bool wxMsgCatalogFile::Load(const wxString& filename, - wxPluralFormsCalculatorPtr& rPluralFormsCalculator) +bool wxMsgCatalogFile::LoadFile(const wxString& filename, + wxPluralFormsCalculatorPtr& rPluralFormsCalculator) { wxFile fileMsg(filename); if ( !fileMsg.IsOpened() ) @@ -962,17 +906,36 @@ bool wxMsgCatalogFile::Load(const wxString& filename, size_t nSize = wx_truncate_cast(size_t, lenFile); wxASSERT_MSG( nSize == lenFile + size_t(0), wxS("message catalog bigger than 4GB?") ); + wxMemoryBuffer filedata; + // read the whole file in memory - if ( fileMsg.Read(m_data.GetWriteBuf(nSize), nSize) != lenFile ) + if ( fileMsg.Read(filedata.GetWriteBuf(nSize), nSize) != lenFile ) + return false; + + filedata.UngetWriteBuf(nSize); + + bool ok = LoadData + ( + DataBuffer::CreateOwned((char*)filedata.release(), nSize), + rPluralFormsCalculator + ); + if ( !ok ) + { + wxLogWarning(_("'%s' is not a valid message catalog."), filename.c_str()); return false; + } - m_data.UngetWriteBuf(nSize); + return true; +} +bool wxMsgCatalogFile::LoadData(const DataBuffer& data, + wxPluralFormsCalculatorPtr& rPluralFormsCalculator) +{ // examine header - bool bValid = m_data.GetDataLen() > sizeof(wxMsgCatalogHeader); + bool bValid = data.length() > sizeof(wxMsgCatalogHeader); - const wxMsgCatalogHeader *pHeader = (wxMsgCatalogHeader *)m_data.GetData(); + const wxMsgCatalogHeader *pHeader = (wxMsgCatalogHeader *)data.data(); if ( bValid ) { // we'll have to swap all the integers if it's true m_bSwapped = pHeader->magic == MSGCATALOG_MAGIC_SW; @@ -983,16 +946,17 @@ bool wxMsgCatalogFile::Load(const wxString& filename, if ( !bValid ) { // it's either too short or has incorrect magic number - wxLogWarning(_("'%s' is not a valid message catalog."), filename.c_str()); - + wxLogWarning(_("Invalid message catalog.")); return false; } + m_data = data; + // initialize m_numStrings = Swap(pHeader->numStrings); - m_pOrigTable = (wxMsgTableEntry *)(StringData() + + m_pOrigTable = (wxMsgTableEntry *)(data.data() + Swap(pHeader->ofsOrigTable)); - m_pTransTable = (wxMsgTableEntry *)(StringData() + + m_pTransTable = (wxMsgTableEntry *)(data.data() + Swap(pHeader->ofsTransTable)); // now parse catalog's header and try to extract catalog charset and @@ -1055,10 +1019,10 @@ bool wxMsgCatalogFile::Load(const wxString& filename, return true; } -bool wxMsgCatalogFile::FillHash(wxMessagesHash& hash, - const wxString& msgIdCharset) const +bool wxMsgCatalogFile::FillHash(wxStringToStringHashMap& hash, + const wxString& domain) const { - wxUnusedVar(msgIdCharset); // silence warning in Unicode build + wxUnusedVar(domain); // silence warning in Unicode build // conversion to use to convert catalog strings to the GUI encoding wxMBConv *inputConv = NULL; @@ -1086,6 +1050,8 @@ bool wxMsgCatalogFile::FillHash(wxMessagesHash& hash, } #if !wxUSE_UNICODE + wxString msgIdCharset = gs_msgIdCharset[domain]; + // conversion to apply to msgid strings before looking them up: we only // need it if the msgids are neither in 7 bit ASCII nor in the same // encoding as the catalog @@ -1177,31 +1143,48 @@ wxMsgCatalog::~wxMsgCatalog() } #endif // !wxUSE_UNICODE -bool wxMsgCatalog::Load(const wxString& filename, - const wxString& domain, - const wxString& msgIdCharset) +/* static */ +wxMsgCatalog *wxMsgCatalog::CreateFromFile(const wxString& filename, + const wxString& domain) { + wxScopedPtr cat(new wxMsgCatalog(domain)); + wxMsgCatalogFile file; - m_domain = domain; + if ( !file.LoadFile(filename, cat->m_pluralFormsCalculator) ) + return NULL; - if ( !file.Load(filename, m_pluralFormsCalculator) ) - return false; + if ( !file.FillHash(cat->m_messages, domain) ) + return NULL; - if ( !file.FillHash(m_messages, msgIdCharset) ) - return false; + return cat.release(); +} - return true; +/* static */ +wxMsgCatalog *wxMsgCatalog::CreateFromData(const wxScopedCharBuffer& data, + const wxString& domain) +{ + wxScopedPtr cat(new wxMsgCatalog(domain)); + + wxMsgCatalogFile file; + + if ( !file.LoadData(data, cat->m_pluralFormsCalculator) ) + return NULL; + + if ( !file.FillHash(cat->m_messages, domain) ) + return NULL; + + return cat.release(); } -const wxString *wxMsgCatalog::GetString(const wxString& str, size_t n) const +const wxString *wxMsgCatalog::GetString(const wxString& str, unsigned n) const { int index = 0; - if (n != size_t(-1)) + if (n != UINT_MAX) { index = m_pluralFormsCalculator->evaluate(n); } - wxMessagesHash::const_iterator i; + wxStringToStringHashMap::const_iterator i; if (index != 0) { i = m_messages.find(wxString(str) + wxChar(index)); // plural @@ -1330,7 +1313,7 @@ bool wxTranslations::AddCatalog(const wxString& domain, wxLanguage msgIdLanguage, const wxString& msgIdCharset) { - m_msgIdCharset[domain] = msgIdCharset; + gs_msgIdCharset[domain] = msgIdCharset; return AddCatalog(domain, msgIdLanguage); } #endif // !wxUSE_UNICODE @@ -1367,6 +1350,8 @@ bool wxTranslations::LoadCatalog(const wxString& domain, const wxString& lang) { wxCHECK_MSG( m_loader, false, "loader can't be NULL" ); + wxMsgCatalog *cat = NULL; + #if wxUSE_FONTMAP // first look for the catalog for this language and the current locale: // notice that we don't use the system name for the locale as this would @@ -1378,31 +1363,44 @@ bool wxTranslations::LoadCatalog(const wxString& domain, const wxString& lang) wxString fullname(lang); fullname << wxS('.') << wxFontMapperBase::GetEncodingName(encSys); - if ( m_loader->LoadCatalog(this, domain, fullname) ) - return true; + cat = m_loader->LoadCatalog(domain, fullname); } #endif // wxUSE_FONTMAP - // Next try: use the provided name language name: - if ( m_loader->LoadCatalog(this, domain, lang) ) - return true; + if ( !cat ) + { + // Next try: use the provided name language name: + cat = m_loader->LoadCatalog(domain, lang); + } - // Also try just base locale name: for things like "fr_BE" (Belgium - // French) we should use fall back on plain "fr" if no Belgium-specific - // message catalogs exist - if ( lang.length() > LEN_LANG && lang[LEN_LANG] == wxS('_') ) + if ( !cat ) { - if ( m_loader->LoadCatalog(this, domain, ExtractLang(lang)) ) - return true; + // Also try just base locale name: for things like "fr_BE" (Belgium + // French) we should use fall back on plain "fr" if no Belgium-specific + // message catalogs exist + wxString baselang = lang.BeforeFirst('_'); + if ( lang != baselang ) + cat = m_loader->LoadCatalog(domain, baselang); } - // Nothing worked, the catalog just isn't there - wxLogTrace(TRACE_I18N, - "Catalog \"%s.mo\" not found for language \"%s\".", - domain, lang); - return false; -} + if ( cat ) + { + // add it to the head of the list so that in GetString it will + // be searched before the catalogs added earlier + cat->m_pNext = m_pMsgCat; + m_pMsgCat = cat; + return true; + } + else + { + // Nothing worked, the catalog just isn't there + wxLogTrace(TRACE_I18N, + "Catalog \"%s.mo\" not found for language \"%s\".", + domain, lang); + return false; + } +} // check if the given catalog is loaded bool wxTranslations::IsLoaded(const wxString& domain) const @@ -1411,34 +1409,6 @@ bool wxTranslations::IsLoaded(const wxString& domain) const } -bool wxTranslations::LoadCatalogFile(const wxString& filename, - const wxString& domain) -{ - wxMsgCatalog *pMsgCat = new wxMsgCatalog; - -#if wxUSE_UNICODE - const bool ok = pMsgCat->Load(filename, domain, wxEmptyString/*unused*/); -#else - const bool ok = pMsgCat->Load(filename, domain, - m_msgIdCharset[domain]); -#endif - - if ( !ok ) - { - // don't add it because it couldn't be loaded anyway - delete pMsgCat; - return false; - } - - // add it to the head of the list so that in GetString it will - // be searched before the catalogs added earlier - pMsgCat->m_pNext = m_pMsgCat; - m_pMsgCat = pMsgCat; - - return true; -} - - wxString wxTranslations::ChooseLanguageForDomain(const wxString& WXUNUSED(domain), const wxString& WXUNUSED(msgIdLang)) { @@ -1475,12 +1445,12 @@ const wxString& wxTranslations::GetUntranslatedString(const wxString& str) const wxString& wxTranslations::GetString(const wxString& origString, const wxString& domain) const { - return GetString(origString, origString, size_t(-1), domain); + return GetString(origString, origString, UINT_MAX, domain); } const wxString& wxTranslations::GetString(const wxString& origString, const wxString& origString2, - size_t n, + unsigned n, const wxString& domain) const { if ( origString.empty() ) @@ -1515,12 +1485,12 @@ const wxString& wxTranslations::GetString(const wxString& origString, TRACE_I18N, "string \"%s\"%s not found in %slocale '%s'.", origString, - ((long)n) != -1 ? wxString::Format("[%ld]", (long)n) : wxString(), + n != UINT_MAX ? wxString::Format("[%ld]", (long)n) : wxString(), !domain.empty() ? wxString::Format("domain '%s' ", domain) : wxString(), m_lang ); - if (n == size_t(-1)) + if (n == UINT_MAX) return GetUntranslatedString(origString); else return GetUntranslatedString(n == 1 ? origString : origString2); @@ -1547,14 +1517,14 @@ wxString wxTranslations::GetHeaderValue(const wxString& header, if ( pMsgCat == NULL ) return wxEmptyString; - trans = pMsgCat->GetString(wxEmptyString, (size_t)-1); + trans = pMsgCat->GetString(wxEmptyString, UINT_MAX); } else { // search in all domains for ( pMsgCat = m_pMsgCat; pMsgCat != NULL; pMsgCat = pMsgCat->m_pNext ) { - trans = pMsgCat->GetString(wxEmptyString, (size_t)-1); + trans = pMsgCat->GetString(wxEmptyString, UINT_MAX); if ( trans != NULL ) // take the first found break; } @@ -1702,13 +1672,9 @@ void wxFileTranslationsLoader::AddCatalogLookupPathPrefix(const wxString& prefix } -bool wxFileTranslationsLoader::LoadCatalog(wxTranslations *translations, - const wxString& domain, - const wxString& lang) +wxMsgCatalog *wxFileTranslationsLoader::LoadCatalog(const wxString& domain, + const wxString& lang) { - wxCHECK_MSG( lang.length() >= LEN_LANG, false, - "invalid language specification" ); - wxString searchPath = GetFullSearchPath(lang); wxLogTrace(TRACE_I18N, wxS("Looking for \"%s.mo\" in search path \"%s\""), @@ -1719,15 +1685,51 @@ bool wxFileTranslationsLoader::LoadCatalog(wxTranslations *translations, wxString strFullName; if ( !wxFindFileInPath(&strFullName, searchPath, fn.GetFullPath()) ) - return false; + return NULL; // open file and read its data wxLogVerbose(_("using catalog '%s' from '%s'."), domain, strFullName.c_str()); wxLogTrace(TRACE_I18N, wxS("Using catalog \"%s\"."), strFullName.c_str()); - return translations->LoadCatalogFile(strFullName, domain); + return wxMsgCatalog::CreateFromFile(strFullName, domain); } + +// ---------------------------------------------------------------------------- +// wxResourceTranslationsLoader +// ---------------------------------------------------------------------------- + +#ifdef __WINDOWS__ +wxMsgCatalog *wxResourceTranslationsLoader::LoadCatalog(const wxString& domain, + const wxString& lang) +{ + + const void *mo_data = NULL; + size_t mo_size = 0; + + const wxString resname = wxString::Format("%s_%s", domain, lang); + + if ( !wxLoadUserResource(&mo_data, &mo_size, + resname, + GetResourceType(), + GetModule()) ) + return false; + + wxLogTrace(TRACE_I18N, + "Using catalog from Windows resource \"%s\".", resname); + + wxMsgCatalog *cat = wxMsgCatalog::CreateFromData( + wxCharBuffer::CreateNonOwned(static_cast(mo_data), mo_size), + domain); + + if ( !cat ) + wxLogWarning(_("Resource '%s' is not a valid message catalog."), resname); + + return cat; +} +#endif // __WINDOWS__ + + // ---------------------------------------------------------------------------- // wxTranslationsModule module (for destruction of gs_translations) // ----------------------------------------------------------------------------