547 lines
20 KiB
C++
547 lines
20 KiB
C++
/////////////////////////////////////////////////////////////////////////////
|
|
// Name: wx/translation.h
|
|
// Purpose: Internationalization and localisation for wxWidgets
|
|
// Author: Vadim Zeitlin, Vaclav Slavik,
|
|
// Michael N. Filippov <michael@idisys.iae.nsk.su>
|
|
// Created: 2010-04-23
|
|
// Copyright: (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
|
|
// (c) 2010 Vaclav Slavik <vslavik@fastmail.fm>
|
|
// Licence: wxWindows licence
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
#ifndef _WX_TRANSLATION_H_
|
|
#define _WX_TRANSLATION_H_
|
|
|
|
#include "wx/defs.h"
|
|
#include "wx/string.h"
|
|
|
|
#if wxUSE_INTL
|
|
|
|
#include "wx/buffer.h"
|
|
#include "wx/language.h"
|
|
#include "wx/strconv.h"
|
|
|
|
// This is a hack, but this header used to include wx/hashmap.h which, in turn,
|
|
// included wx/wxcrt.h and it turns out quite some existing code relied on it
|
|
// by using the CRT wrapper functions declared there without explicitly
|
|
// including that header, so keep including it from here to let it continue to
|
|
// compile.
|
|
#include "wx/wxcrt.h"
|
|
|
|
#include <memory>
|
|
#include <unordered_map>
|
|
|
|
using wxTranslationsHashMap = std::unordered_map<wxString, wxString>;
|
|
|
|
// ============================================================================
|
|
// global decls
|
|
// ============================================================================
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// macros
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// gettext() style macros (notice that xgettext should be invoked with
|
|
// --keyword="_" --keyword="wxPLURAL:1,2" options
|
|
// to extract the strings from the sources)
|
|
#ifndef WXINTL_NO_GETTEXT_MACRO
|
|
#define _(s) wxUnderscoreWrapper((s))
|
|
#endif
|
|
|
|
#define wxPLURAL(sing, plur, n) wxPluralWrapper((sing), (plur), n)
|
|
|
|
// wx-specific macro for translating strings in the given context: if you use
|
|
// them, you need to also add
|
|
// --keyword="wxGETTEXT_IN_CONTEXT:1c,2" --keyword="wxGETTEXT_IN_CONTEXT_PLURAL:1c,2,3"
|
|
// options to xgettext invocation.
|
|
#define wxGETTEXT_IN_CONTEXT(c, s) \
|
|
wxGettextInContextWrapper((c), (s))
|
|
#define wxGETTEXT_IN_CONTEXT_PLURAL(c, sing, plur, n) \
|
|
wxGettextInContextPluralWrapper((c), (sing), (plur), (n))
|
|
|
|
// another one which just marks the strings for extraction, but doesn't
|
|
// perform the translation (use -kwxTRANSLATE with xgettext!)
|
|
#define wxTRANSLATE(str) str
|
|
|
|
// another one which just marks the strings, with a context, for extraction,
|
|
// but doesn't perform the translation (use -kwxTRANSLATE_IN_CONTEXT:1c,2 with
|
|
// xgettext!)
|
|
#define wxTRANSLATE_IN_CONTEXT(c, str) str
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// forward decls
|
|
// ----------------------------------------------------------------------------
|
|
|
|
class WXDLLIMPEXP_FWD_BASE wxArrayString;
|
|
class WXDLLIMPEXP_FWD_BASE wxTranslationsLoader;
|
|
class WXDLLIMPEXP_FWD_BASE wxLocale;
|
|
|
|
class wxPluralFormsCalculator;
|
|
using wxPluralFormsCalculatorPtr = std::unique_ptr<wxPluralFormsCalculator>;
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// wxMsgCatalog corresponds to one loaded message catalog.
|
|
// ----------------------------------------------------------------------------
|
|
|
|
class WXDLLIMPEXP_BASE wxMsgCatalog
|
|
{
|
|
public:
|
|
// Ctor is protected, because CreateFromXXX functions must be used,
|
|
// but destruction should be unrestricted
|
|
~wxMsgCatalog();
|
|
|
|
// load the catalog from disk or from data; caller is responsible for
|
|
// deleting them if not null
|
|
static wxMsgCatalog *CreateFromFile(const wxString& filename,
|
|
const wxString& domain);
|
|
|
|
static wxMsgCatalog *CreateFromData(const wxScopedCharBuffer& data,
|
|
const wxString& domain);
|
|
|
|
// get name of the catalog
|
|
wxString GetDomain() const { return m_domain; }
|
|
|
|
// get the translated string: returns nullptr if not found
|
|
const wxString *GetString(const wxString& sz, unsigned n = UINT_MAX, const wxString& ct = wxEmptyString) const;
|
|
|
|
protected:
|
|
wxMsgCatalog(const wxString& domain);
|
|
|
|
private:
|
|
// variable pointing to the next element in a linked list (or nullptr)
|
|
wxMsgCatalog *m_pNext;
|
|
friend class wxTranslations;
|
|
|
|
wxTranslationsHashMap m_messages; // all messages in the catalog
|
|
wxString m_domain; // name of the domain
|
|
|
|
wxPluralFormsCalculatorPtr m_pluralFormsCalculator;
|
|
};
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// wxTranslations: message catalogs
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// this class allows to get translations for strings
|
|
class WXDLLIMPEXP_BASE wxTranslations
|
|
{
|
|
public:
|
|
wxTranslations();
|
|
~wxTranslations();
|
|
|
|
// returns current translations object, may return nullptr
|
|
static wxTranslations *Get();
|
|
// sets current translations object (takes ownership; may be null)
|
|
static void Set(wxTranslations *t);
|
|
|
|
// changes loader to non-default one; takes ownership of 'loader'
|
|
void SetLoader(wxTranslationsLoader *loader);
|
|
|
|
void SetLanguage(wxLanguage lang);
|
|
void SetLanguage(const wxString& lang);
|
|
|
|
// get languages available for this app
|
|
wxArrayString GetAvailableTranslations(const wxString& domain) const;
|
|
|
|
// find best available translation language for given domain
|
|
wxString GetBestAvailableTranslation(const wxString& domain);
|
|
|
|
wxString GetBestTranslation(const wxString& domain, wxLanguage msgIdLanguage);
|
|
wxString GetBestTranslation(const wxString& domain,
|
|
const wxString& msgIdLanguage = wxASCII_STR("en"));
|
|
|
|
// add catalog for the given domain returning true if it could be found by
|
|
// wxTranslationsLoader
|
|
bool AddAvailableCatalog(const wxString& domain, wxLanguage msgIdLanguage = wxLANGUAGE_ENGLISH_US);
|
|
|
|
// add standard wxWidgets catalog ("wxstd")
|
|
bool AddStdCatalog();
|
|
|
|
// add catalog with given domain name and language, looking it up via
|
|
// wxTranslationsLoader -- unlike AddAvailableCatalog(), this function also
|
|
// returns true if this catalog is not needed at all because msgIdLanguage
|
|
// is an acceptable language to use directly
|
|
bool AddCatalog(const wxString& domain,
|
|
wxLanguage msgIdLanguage = wxLANGUAGE_ENGLISH_US);
|
|
|
|
// check if the given catalog is loaded
|
|
bool IsLoaded(const wxString& domain) const;
|
|
|
|
// access to translations
|
|
const wxString *GetTranslatedString(const wxString& origString,
|
|
const wxString& domain = wxEmptyString,
|
|
const wxString& context = wxEmptyString) const;
|
|
const wxString *GetTranslatedString(const wxString& origString,
|
|
unsigned n,
|
|
const wxString& domain = wxEmptyString,
|
|
const wxString& context = wxEmptyString) const;
|
|
|
|
wxString GetHeaderValue(const wxString& header,
|
|
const wxString& domain = wxEmptyString) const;
|
|
|
|
// this is hack to work around a problem with wxGetTranslation() which
|
|
// returns const wxString& and not wxString, so when it returns untranslated
|
|
// string, it needs to have a copy of it somewhere
|
|
static const wxString& GetUntranslatedString(const wxString& str);
|
|
|
|
private:
|
|
enum class Translations
|
|
{
|
|
NotNeeded = -1,
|
|
NotFound = 0,
|
|
Found = 1
|
|
};
|
|
|
|
Translations DoAddCatalog(const wxString& domain, wxLanguage msgIdLanguage);
|
|
|
|
// perform loading of the catalog via m_loader
|
|
bool LoadCatalog(const wxString& domain, const wxString& lang);
|
|
|
|
// find catalog by name in a linked list, return nullptr if !found
|
|
wxMsgCatalog *FindCatalog(const wxString& domain) const;
|
|
|
|
// same as Set(), without taking ownership; only for wxLocale
|
|
static void SetNonOwned(wxTranslations *t);
|
|
friend class wxLocale;
|
|
|
|
wxString DoGetBestAvailableTranslation(const wxString& domain, const wxString& additionalAvailableLanguage);
|
|
|
|
private:
|
|
wxString m_lang;
|
|
wxTranslationsLoader *m_loader;
|
|
|
|
wxMsgCatalog *m_pMsgCat; // pointer to linked list of catalogs
|
|
|
|
// In addition to keeping all the catalogs in the linked list, we also
|
|
// store them in a hash map indexed by the domain name to allow finding
|
|
// them by name efficiently.
|
|
using wxMsgCatalogMap = std::unordered_map<wxString, wxMsgCatalog*>;
|
|
wxMsgCatalogMap m_catalogMap;
|
|
};
|
|
|
|
|
|
// abstraction of translations discovery and loading
|
|
class WXDLLIMPEXP_BASE wxTranslationsLoader
|
|
{
|
|
public:
|
|
wxTranslationsLoader() = default;
|
|
virtual ~wxTranslationsLoader() = default;
|
|
|
|
virtual wxMsgCatalog *LoadCatalog(const wxString& domain,
|
|
const wxString& lang) = 0;
|
|
|
|
virtual wxArrayString GetAvailableTranslations(const wxString& domain) const = 0;
|
|
};
|
|
|
|
|
|
// standard wxTranslationsLoader implementation, using filesystem
|
|
class WXDLLIMPEXP_BASE wxFileTranslationsLoader
|
|
: public wxTranslationsLoader
|
|
{
|
|
public:
|
|
static void AddCatalogLookupPathPrefix(const wxString& prefix);
|
|
|
|
virtual wxMsgCatalog *LoadCatalog(const wxString& domain,
|
|
const wxString& lang) override;
|
|
|
|
virtual wxArrayString GetAvailableTranslations(const wxString& domain) const override;
|
|
};
|
|
|
|
|
|
#ifdef __WINDOWS__
|
|
// loads translations from win32 resources
|
|
class WXDLLIMPEXP_BASE wxResourceTranslationsLoader
|
|
: public wxTranslationsLoader
|
|
{
|
|
public:
|
|
virtual wxMsgCatalog *LoadCatalog(const wxString& domain,
|
|
const wxString& lang) override;
|
|
|
|
virtual wxArrayString GetAvailableTranslations(const wxString& domain) const override;
|
|
|
|
protected:
|
|
// returns resource type to use for translations
|
|
virtual wxString GetResourceType() const { return wxASCII_STR("MOFILE"); }
|
|
|
|
// returns module to load resources from
|
|
virtual WXHINSTANCE GetModule() const { return nullptr; }
|
|
};
|
|
#endif // __WINDOWS__
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// global functions
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// get the translation of the string in the current locale
|
|
inline const wxString& wxGetTranslation(const wxString& str,
|
|
const wxString& domain = wxString(),
|
|
const wxString& context = wxString())
|
|
{
|
|
wxTranslations *trans = wxTranslations::Get();
|
|
const wxString *transStr = trans ? trans->GetTranslatedString(str, domain, context)
|
|
: nullptr;
|
|
if ( transStr )
|
|
return *transStr;
|
|
else
|
|
// NB: this function returns reference to a string, so we have to keep
|
|
// a copy of it somewhere
|
|
return wxTranslations::GetUntranslatedString(str);
|
|
}
|
|
|
|
inline const wxString& wxGetTranslation(const wxString& str1,
|
|
const wxString& str2,
|
|
unsigned n,
|
|
const wxString& domain = wxString(),
|
|
const wxString& context = wxString())
|
|
{
|
|
wxTranslations *trans = wxTranslations::Get();
|
|
const wxString *transStr = trans ? trans->GetTranslatedString(str1, n, domain, context)
|
|
: nullptr;
|
|
if ( transStr )
|
|
return *transStr;
|
|
else
|
|
// NB: this function returns reference to a string, so we have to keep
|
|
// a copy of it somewhere
|
|
return n == 1
|
|
? wxTranslations::GetUntranslatedString(str1)
|
|
: wxTranslations::GetUntranslatedString(str2);
|
|
}
|
|
|
|
#ifdef wxNO_IMPLICIT_WXSTRING_ENCODING
|
|
|
|
/*
|
|
* It must always be possible to call wxGetTranslation() with const
|
|
* char* arguments.
|
|
*/
|
|
inline const wxString& wxGetTranslation(const char *str,
|
|
const char *domain = "",
|
|
const char *context = "") {
|
|
const wxMBConv &conv = wxConvWhateverWorks;
|
|
return wxGetTranslation(wxString(str, conv), wxString(domain, conv),
|
|
wxString(context, conv));
|
|
}
|
|
|
|
inline const wxString& wxGetTranslation(const char *str1,
|
|
const char *str2,
|
|
unsigned n,
|
|
const char *domain = "",
|
|
const char *context = "") {
|
|
const wxMBConv &conv = wxConvWhateverWorks;
|
|
return wxGetTranslation(wxString(str1, conv), wxString(str2, conv), n,
|
|
wxString(domain, conv),
|
|
wxString(context, conv));
|
|
}
|
|
|
|
// We can't construct wxString implicitly in this case, so use a helper.
|
|
inline wxString wxTRANS_INPUT_STR(const char* s)
|
|
{
|
|
return wxString::FromAscii(s);
|
|
}
|
|
|
|
inline wxString wxTRANS_INPUT_STR(const wchar_t* s)
|
|
{
|
|
return wxString(s);
|
|
}
|
|
#else // !wxNO_IMPLICIT_WXSTRING_ENCODING
|
|
// We can rely on implicit conversion, so don't bother with the helper.
|
|
#define wxTRANS_INPUT_STR(s) s
|
|
#endif // wxNO_IMPLICIT_WXSTRING_ENCODING
|
|
|
|
#ifndef wxNO_REQUIRE_LITERAL_MSGIDS
|
|
|
|
// Wrapper functions that only accept string literals as arguments,
|
|
// not variables, not char* pointers, and define the fall backs only in
|
|
// order to point out to the comment below:
|
|
|
|
/*
|
|
*** LITERALS-MSGID comment marker ***
|
|
|
|
This comment explains the reason for static assert failures below. Please
|
|
see https://wxwidgets.org/help/msgid-literals for a more detailed and
|
|
possibly more up-to-date version of this comment.
|
|
|
|
If you get a compile error when using any of the translation macros, i.e.
|
|
_(), wxPLURAL() etc, it means that you're passing something other than a
|
|
literal string, i.e. just simple "whatever", to wx translation functions.
|
|
This most likely indicates a bug in your program which is now detected when
|
|
it was silently ignored before and should be fixed by changing the code to
|
|
use string literals. For example, existing code doing
|
|
|
|
_(wxString::Format("Hello %s", who))
|
|
|
|
wouldn't work properly even if it were allowed to compile and should be
|
|
changed to
|
|
|
|
wxString::Format(_("Hello %s"), who))
|
|
|
|
However if you can't do this, for some reason, you may choose to predefine
|
|
wxNO_REQUIRE_LITERAL_MSGIDS which disables these checks. Please note that
|
|
this is *NOT* recommended, as the problematic strings still won't be
|
|
translated, because they won't have been extracted by xgettext in the first
|
|
place and the right thing to do remains to fix the code instead.
|
|
*/
|
|
|
|
template<size_t N, typename T>
|
|
const wxString& wxUnderscoreWrapper(const T (&msg)[N])
|
|
{
|
|
return wxGetTranslation(wxTRANS_INPUT_STR(msg));
|
|
}
|
|
|
|
template <typename T>
|
|
wxString wxUnderscoreWrapper(T)
|
|
{
|
|
static_assert(!sizeof(T), "See https://wxwidgets.org/help/msgid-literals or LITERALS-MSGID comment above.");
|
|
return {};
|
|
}
|
|
|
|
template<size_t M, size_t N, typename T>
|
|
const wxString& wxPluralWrapper(const T (&msg)[M],
|
|
const T (&plural)[N],
|
|
int count)
|
|
{
|
|
return wxGetTranslation(wxTRANS_INPUT_STR(msg), wxTRANS_INPUT_STR(plural),
|
|
count);
|
|
}
|
|
|
|
template <typename T, typename U>
|
|
wxString wxPluralWrapper(T, U, int)
|
|
{
|
|
static_assert(!sizeof(T), "See https://wxwidgets.org/help/msgid-literals or LITERALS-MSGID comment above.");
|
|
return {};
|
|
}
|
|
|
|
template<size_t M, size_t N, typename T>
|
|
const wxString& wxGettextInContextWrapper(const T (&ctx)[M],
|
|
const T (&msg)[N])
|
|
{
|
|
return wxGetTranslation(wxTRANS_INPUT_STR(msg), wxString(),
|
|
wxTRANS_INPUT_STR(ctx));
|
|
}
|
|
|
|
template <typename T, typename U>
|
|
wxString wxGettextInContextWrapper(T, U)
|
|
{
|
|
static_assert(!sizeof(T), "See https://wxwidgets.org/help/msgid-literals or LITERALS-MSGID comment above.");
|
|
return {};
|
|
}
|
|
|
|
template<size_t L, size_t M, size_t N, typename T>
|
|
const wxString& wxGettextInContextPluralWrapper(const T (&ctx)[L],
|
|
const T (&msg)[M],
|
|
const T (&plural)[N],
|
|
int count)
|
|
{
|
|
return wxGetTranslation(wxTRANS_INPUT_STR(msg), wxTRANS_INPUT_STR(plural),
|
|
count, wxString(), wxTRANS_INPUT_STR(ctx));
|
|
}
|
|
|
|
template <typename T, typename U, typename V>
|
|
wxString wxGettextInContextPluralWrapper(T, U, V, int)
|
|
{
|
|
static_assert(!sizeof(T), "See https://wxwidgets.org/help/msgid-literals or LITERALS-MSGID comment above.");
|
|
return {};
|
|
}
|
|
|
|
#else // wxNO_REQUIRE_LITERAL_MSGIDS
|
|
|
|
// Wrapper functions that accept both string literals and variables
|
|
// as arguments.
|
|
inline const wxString& wxUnderscoreWrapper(const char *msg)
|
|
{
|
|
return wxGetTranslation(wxTRANS_INPUT_STR(msg));
|
|
}
|
|
|
|
inline const wxString& wxPluralWrapper(const char *msg,
|
|
const char *plural,
|
|
int count)
|
|
{
|
|
return wxGetTranslation(wxTRANS_INPUT_STR(msg), wxTRANS_INPUT_STR(plural),
|
|
count);
|
|
}
|
|
|
|
inline const wxString& wxGettextInContextWrapper(const char *ctx,
|
|
const char *msg)
|
|
{
|
|
return wxGetTranslation(wxTRANS_INPUT_STR(msg), wxString(),
|
|
wxTRANS_INPUT_STR(ctx));
|
|
}
|
|
|
|
inline const wxString& wxGettextInContextPluralWrapper(const char *ctx,
|
|
const char *msg,
|
|
const char *plural,
|
|
int count)
|
|
{
|
|
return wxGetTranslation(wxTRANS_INPUT_STR(msg), wxTRANS_INPUT_STR(plural),
|
|
count, wxString(), wxTRANS_INPUT_STR(ctx));
|
|
}
|
|
|
|
#endif // wxNO_REQUIRE_LITERAL_MSGIDS
|
|
|
|
#else // !wxUSE_INTL
|
|
|
|
// the macros should still be defined - otherwise compilation would fail
|
|
|
|
#if !defined(WXINTL_NO_GETTEXT_MACRO)
|
|
#if !defined(_)
|
|
#ifndef wxNO_IMPLICIT_WXSTRING_ENCODING
|
|
#define _(s) (s)
|
|
#else
|
|
#define _(s) wxASCII_STR(s)
|
|
#endif
|
|
#endif
|
|
#define wxPLURAL(sing, plur, n) ((n) == 1 ? (sing) : (plur))
|
|
#define wxGETTEXT_IN_CONTEXT(c, s) (s)
|
|
#define wxGETTEXT_IN_CONTEXT_PLURAL(c, sing, plur, n) wxPLURAL(sing, plur, n)
|
|
#endif
|
|
|
|
#define wxTRANSLATE(str) str
|
|
#define wxTRANSLATE_IN_CONTEXT(c, str) str
|
|
|
|
// NB: we use a template here in order to avoid using
|
|
// wxLocale::GetUntranslatedString() above, which would be required if
|
|
// we returned const wxString&; this way, the compiler should be able to
|
|
// optimize wxGetTranslation() away
|
|
|
|
template<typename TString>
|
|
inline TString wxGetTranslation(TString str)
|
|
{ return str; }
|
|
|
|
template<typename TString, typename TDomain>
|
|
inline TString wxGetTranslation(TString str, TDomain WXUNUSED(domain))
|
|
{ return str; }
|
|
|
|
template<typename TString, typename TDomain, typename TContext>
|
|
inline TString wxGetTranslation(TString str, TDomain WXUNUSED(domain), TContext WXUNUSED(context))
|
|
{ return str; }
|
|
|
|
template<typename TString, typename TDomain>
|
|
inline TString wxGetTranslation(TString str1, TString str2, size_t n)
|
|
{ return n == 1 ? str1 : str2; }
|
|
|
|
template<typename TString, typename TDomain>
|
|
inline TString wxGetTranslation(TString str1, TString str2, size_t n,
|
|
TDomain WXUNUSED(domain))
|
|
{ return n == 1 ? str1 : str2; }
|
|
|
|
template<typename TString, typename TDomain, typename TContext>
|
|
inline TString wxGetTranslation(TString str1, TString str2, size_t n,
|
|
TDomain WXUNUSED(domain),
|
|
TContext WXUNUSED(context))
|
|
{ return n == 1 ? str1 : str2; }
|
|
|
|
#endif // wxUSE_INTL/!wxUSE_INTL
|
|
|
|
// define this one just in case it occurs somewhere (instead of preferred
|
|
// wxTRANSLATE) too
|
|
#if !defined(WXINTL_NO_GETTEXT_MACRO)
|
|
#if !defined(gettext_noop)
|
|
#define gettext_noop(str) (str)
|
|
#endif
|
|
#if !defined(N_)
|
|
#define N_(s) (s)
|
|
#endif
|
|
#endif
|
|
|
|
#endif // _WX_TRANSLATION_H_
|