1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        wx/translation.h 
   3 // Purpose:     Internationalization and localisation for wxWidgets 
   4 // Author:      Vadim Zeitlin, Vaclav Slavik, 
   5 //              Michael N. Filippov <michael@idisys.iae.nsk.su> 
   8 // Copyright:   (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr> 
   9 //              (c) 2010 Vaclav Slavik <vslavik@fastmail.fm> 
  10 // Licence:     wxWindows licence 
  11 ///////////////////////////////////////////////////////////////////////////// 
  13 #ifndef _WX_TRANSLATION_H_ 
  14 #define _WX_TRANSLATION_H_ 
  17 #include "wx/string.h" 
  21 #include "wx/buffer.h" 
  22 #include "wx/language.h" 
  23 #include "wx/hashmap.h" 
  24 #include "wx/strconv.h" 
  25 #include "wx/scopedptr.h" 
  27 // ============================================================================ 
  29 // ============================================================================ 
  31 // ---------------------------------------------------------------------------- 
  33 // ---------------------------------------------------------------------------- 
  35 // gettext() style macros (notice that xgettext should be invoked with 
  36 // --keyword="_" --keyword="wxPLURAL:1,2" options 
  37 // to extract the strings from the sources) 
  38 #ifndef WXINTL_NO_GETTEXT_MACRO 
  39     #define _(s)                     wxGetTranslation((s)) 
  40     #define wxPLURAL(sing, plur, n)  wxGetTranslation((sing), (plur), n) 
  43 // another one which just marks the strings for extraction, but doesn't 
  44 // perform the translation (use -kwxTRANSLATE with xgettext!) 
  45 #define wxTRANSLATE(str) str 
  47 // ---------------------------------------------------------------------------- 
  49 // ---------------------------------------------------------------------------- 
  51 class WXDLLIMPEXP_FWD_BASE wxArrayString
; 
  52 class WXDLLIMPEXP_FWD_BASE wxTranslationsLoader
; 
  53 class WXDLLIMPEXP_FWD_BASE wxLocale
; 
  55 class wxPluralFormsCalculator
; 
  56 wxDECLARE_SCOPED_PTR(wxPluralFormsCalculator
, wxPluralFormsCalculatorPtr
) 
  58 // ---------------------------------------------------------------------------- 
  59 // wxMsgCatalog corresponds to one loaded message catalog. 
  60 // ---------------------------------------------------------------------------- 
  62 class WXDLLIMPEXP_BASE wxMsgCatalog
 
  65     // Ctor is protected, because CreateFromXXX functions must be used, 
  66     // but destruction should be unrestricted 
  71     // load the catalog from disk or from data; caller is responsible for 
  72     // deleting them if not NULL 
  73     static wxMsgCatalog 
*CreateFromFile(const wxString
& filename
, 
  74                                         const wxString
& domain
); 
  76     static wxMsgCatalog 
*CreateFromData(const wxScopedCharBuffer
& data
, 
  77                                         const wxString
& domain
); 
  79     // get name of the catalog 
  80     wxString 
GetDomain() const { return m_domain
; } 
  82     // get the translated string: returns NULL if not found 
  83     const wxString 
*GetString(const wxString
& sz
, unsigned n 
= UINT_MAX
) const; 
  86     wxMsgCatalog(const wxString
& domain
) 
  87         : m_pNext(NULL
), m_domain(domain
) 
  94     // variable pointing to the next element in a linked list (or NULL) 
  95     wxMsgCatalog 
*m_pNext
; 
  96     friend class wxTranslations
; 
  98     wxStringToStringHashMap m_messages
; // all messages in the catalog 
  99     wxString                m_domain
;   // name of the domain 
 102     // the conversion corresponding to this catalog charset if we installed it 
 107     wxPluralFormsCalculatorPtr m_pluralFormsCalculator
; 
 110 // ---------------------------------------------------------------------------- 
 111 // wxTranslations: message catalogs 
 112 // ---------------------------------------------------------------------------- 
 114 // this class allows to get translations for strings 
 115 class WXDLLIMPEXP_BASE wxTranslations
 
 121     // returns current translations object, may return NULL 
 122     static wxTranslations 
*Get(); 
 123     // sets current translations object (takes ownership; may be NULL) 
 124     static void Set(wxTranslations 
*t
); 
 126     // changes loader to non-default one; takes ownership of 'loader' 
 127     void SetLoader(wxTranslationsLoader 
*loader
); 
 129     void SetLanguage(wxLanguage lang
); 
 130     void SetLanguage(const wxString
& lang
); 
 132     // get languages available for this app 
 133     wxArrayString 
GetAvailableTranslations(const wxString
& domain
) const; 
 135     // add standard wxWidgets catalog ("wxstd") 
 136     bool AddStdCatalog(); 
 138     // add catalog with given domain name and language, looking it up via 
 139     // wxTranslationsLoader 
 140     bool AddCatalog(const wxString
& domain
); 
 141     bool AddCatalog(const wxString
& domain
, wxLanguage msgIdLanguage
); 
 143     bool AddCatalog(const wxString
& domain
, 
 144                     wxLanguage msgIdLanguage
, 
 145                     const wxString
& msgIdCharset
); 
 148     // check if the given catalog is loaded 
 149     bool IsLoaded(const wxString
& domain
) const; 
 151     // access to translations 
 152     const wxString
& GetString(const wxString
& origString
, 
 153                               const wxString
& domain 
= wxEmptyString
) const; 
 154     const wxString
& GetString(const wxString
& origString
, 
 155                               const wxString
& origString2
, 
 157                               const wxString
& domain 
= wxEmptyString
) const; 
 159     wxString 
GetHeaderValue(const wxString
& header
, 
 160                             const wxString
& domain 
= wxEmptyString
) const; 
 162     // this is hack to work around a problem with wxGetTranslation() which 
 163     // returns const wxString& and not wxString, so when it returns untranslated 
 164     // string, it needs to have a copy of it somewhere 
 165     static const wxString
& GetUntranslatedString(const wxString
& str
); 
 168     // perform loading of the catalog via m_loader 
 169     bool LoadCatalog(const wxString
& domain
, const wxString
& lang
); 
 171     // find best translation for given domain 
 172     wxString 
ChooseLanguageForDomain(const wxString
& domain
, 
 173                                      const wxString
& msgIdLang
); 
 175     // find catalog by name in a linked list, return NULL if !found 
 176     wxMsgCatalog 
*FindCatalog(const wxString
& domain
) const; 
 178     // same as Set(), without taking ownership; only for wxLocale 
 179     static void SetNonOwned(wxTranslations 
*t
); 
 180     friend class wxLocale
; 
 184     wxTranslationsLoader 
*m_loader
; 
 186     wxMsgCatalog 
*m_pMsgCat
; // pointer to linked list of catalogs 
 190 // abstraction of translations discovery and loading 
 191 class WXDLLIMPEXP_BASE wxTranslationsLoader
 
 194     wxTranslationsLoader() {} 
 195     virtual ~wxTranslationsLoader() {} 
 197     virtual wxMsgCatalog 
*LoadCatalog(const wxString
& domain
, 
 198                                       const wxString
& lang
) = 0; 
 200     virtual wxArrayString 
GetAvailableTranslations(const wxString
& domain
) const = 0; 
 204 // standard wxTranslationsLoader implementation, using filesystem 
 205 class WXDLLIMPEXP_BASE wxFileTranslationsLoader
 
 206     : public wxTranslationsLoader
 
 209     static void AddCatalogLookupPathPrefix(const wxString
& prefix
); 
 211     virtual wxMsgCatalog 
*LoadCatalog(const wxString
& domain
, 
 212                                       const wxString
& lang
); 
 214     virtual wxArrayString 
GetAvailableTranslations(const wxString
& domain
) const; 
 219 // loads translations from win32 resources 
 220 class WXDLLIMPEXP_BASE wxResourceTranslationsLoader
 
 221     : public wxTranslationsLoader
 
 224     virtual wxMsgCatalog 
*LoadCatalog(const wxString
& domain
, 
 225                                       const wxString
& lang
); 
 227     virtual wxArrayString 
GetAvailableTranslations(const wxString
& domain
) const; 
 230     // returns resource type to use for translations 
 231     virtual wxString 
GetResourceType() const { return "MOFILE"; } 
 233     // returns module to load resources from 
 234     virtual WXHINSTANCE 
GetModule() const { return 0; } 
 236 #endif // __WINDOWS__ 
 239 // ---------------------------------------------------------------------------- 
 241 // ---------------------------------------------------------------------------- 
 243 // get the translation of the string in the current locale 
 244 inline const wxString
& wxGetTranslation(const wxString
& str
, 
 245                                         const wxString
& domain 
= wxEmptyString
) 
 247     wxTranslations 
*trans 
= wxTranslations::Get(); 
 249         return trans
->GetString(str
, domain
); 
 251         // NB: this function returns reference to a string, so we have to keep 
 252         //     a copy of it somewhere 
 253         return wxTranslations::GetUntranslatedString(str
); 
 256 inline const wxString
& wxGetTranslation(const wxString
& str1
, 
 257                                         const wxString
& str2
, 
 259                                         const wxString
& domain 
= wxEmptyString
) 
 261     wxTranslations 
*trans 
= wxTranslations::Get(); 
 263         return trans
->GetString(str1
, str2
, n
, domain
); 
 265         // NB: this function returns reference to a string, so we have to keep 
 266         //     a copy of it somewhere 
 268                ? wxTranslations::GetUntranslatedString(str1
) 
 269                : wxTranslations::GetUntranslatedString(str2
); 
 274 // the macros should still be defined - otherwise compilation would fail 
 276 #if !defined(WXINTL_NO_GETTEXT_MACRO) 
 280     #define wxPLURAL(sing, plur, n)  ((n) == 1 ? (sing) : (plur)) 
 283 #define wxTRANSLATE(str) str 
 285 // NB: we use a template here in order to avoid using 
 286 //     wxLocale::GetUntranslatedString() above, which would be required if 
 287 //     we returned const wxString&; this way, the compiler should be able to 
 288 //     optimize wxGetTranslation() away 
 290 template<typename TString
> 
 291 inline TString 
wxGetTranslation(TString str
) 
 294 template<typename TString
, typename TDomain
> 
 295 inline TString 
wxGetTranslation(TString str
, TDomain 
WXUNUSED(domain
)) 
 298 template<typename TString
, typename TDomain
> 
 299 inline TString 
wxGetTranslation(TString str1
, TString str2
, size_t n
) 
 300     { return n 
== 1 ? str1 
: str2
; } 
 302 template<typename TString
, typename TDomain
> 
 303 inline TString 
wxGetTranslation(TString str1
, TString str2
, size_t n
, 
 304                                 TDomain 
WXUNUSED(domain
)) 
 305     { return n 
== 1 ? str1 
: str2
; } 
 307 #endif // wxUSE_INTL/!wxUSE_INTL 
 309 // define this one just in case it occurs somewhere (instead of preferred 
 311 #if !defined(WXINTL_NO_GETTEXT_MACRO) 
 312     #if !defined(gettext_noop) 
 313         #define gettext_noop(str) (str) 
 320 #endif // _WX_TRANSLATION_H_