X-Git-Url: https://git.lyx.org/gitweb/?a=blobdiff_plain;f=src%2Fmessages.C;h=90a345470cbaaac57c0cb2aa09bbb14cdfa914c7;hb=35204f8f33d7400a5fefeffea533fb4cb4097211;hp=9966178e5a967eb7f33d23c29e918e83f8fe47ec;hpb=795e9a7586527296ed037e2a0f1d3a321ba9e7ed;p=lyx.git diff --git a/src/messages.C b/src/messages.C index 9966178e5a..90a345470c 100644 --- a/src/messages.C +++ b/src/messages.C @@ -4,35 +4,256 @@ * * \author Lars Gullik Bjønnes * - * Full author contact details are available in file CREDITS + * Full author contact details are available in file CREDITS. */ #include -#include "messages.h" #include "debug.h" +#include "messages.h" +#include "support/filetools.h" +#include "support/environment.h" +#include "support/package.h" +#include "support/docstring.h" +#include "support/types.h" + +#include +#include + +#include + +namespace lyx { + +using support::package; +using support::getEnv; +using support::setEnv; + +using std::string; using std::endl; -Messages::Messages(string const & l, string const & dir) - : lang_(l), localedir_(dir), - loc_gl(lang_.c_str()), - mssg_gl(std::use_facet >(loc_gl)) -{ - lyxerr << "Messages: language(" << l << ") in dir(" << dir << ")" << endl; +static boost::regex const reg("^([^\\[]*)\\[\\[[^\\]]*\\]\\]$"); - cat_gl = mssg_gl.open("lyx", loc_gl, localedir_.c_str()); -} +#ifdef ENABLE_NLS + + +#if 0 + +-#include + +// This version of the Pimpl utilizes the message capability of +// libstdc++ that is distributed with GNU G++. +class Messages::Pimpl { +public: + typedef std::messages::catalog catalog; + + Pimpl(string const & l) + : lang_(l), + loc_gl(lang_.c_str()), + mssg_gl(std::use_facet >(loc_gl)) + { + //lyxerr << "Messages: language(" << l + // << ") in dir(" << dir << ")" << endl; + + cat_gl = mssg_gl.open(PACKAGE, loc_gl, package().locale_dir().c_str()); + + } + + ~Pimpl() + { + mssg_gl.close(cat_gl); + } + + docstring const get(string const & msg) const + { + return mssg_gl.get(cat_gl, 0, 0, msg); + } +private: + /// + string lang_; + /// + std::locale loc_gl; + /// + std::messages const & mssg_gl; + /// + catalog cat_gl; +}; +#else + +#ifdef HAVE_LOCALE_H +# include +#endif +# if HAVE_GETTEXT +# include // use the header already in the system *EK* +# else +# include "../intl/libintl.h" +# endif + +// This is a more traditional variant. +class Messages::Pimpl { +public: + Pimpl(string const & l) + : lang_(l) + { + if ( lang_.empty() ) { + char const * lc_msgs = 0; +#ifdef HAVE_LC_MESSAGES + lc_msgs = setlocale(LC_MESSAGES, NULL); +#endif + lang_ = lc_msgs ? lc_msgs : ""; + } + // strip off any encoding suffix, i.e., assume 8-bit po files + string::size_type i = lang_.find("."); + lang_ = lang_.substr(0, i); + lyxerr[Debug::DEBUG] << BOOST_CURRENT_FUNCTION + << ": language(" << lang_ << ")" << endl; + } + + ~Pimpl() {} + + docstring const get(string const & m) const + { + if (m.empty()) + return from_ascii(m); + + // In this order, see support/filetools.C: + string lang = getEnv("LC_ALL"); + if (lang.empty()) { + lang = getEnv("LC_MESSAGES"); + if (lang.empty()) { + lang = getEnv("LANG"); + if (lang.empty()) + lang = "C"; + } + } +#ifdef HAVE_LC_MESSAGES + char const * lc_msgs = setlocale(LC_MESSAGES, lang_.c_str()); +#endif + // setlocale fails (returns NULL) if the corresponding locale + // is not installed. + // On windows (mingw and cygwin) it always returns NULL. + // Since this method gets called for every translatable + // buffer string like e.g. "Figure:" we warn only once. +#if !defined(_WIN32) && !defined(__CYGWIN__) + static bool warned = false; + if (!warned && !lc_msgs) { + warned = true; + lyxerr << "Locale " << lang_ << " could not be set" << endl; + } +#endif + // CTYPE controls what getmessage thinks what encoding the po file uses + char const * lc_ctype = setlocale(LC_CTYPE, NULL); + string oldCTYPE = lc_ctype ? lc_ctype : ""; + + setlocale(LC_CTYPE, lang_.c_str()); + errno = 0; + char const * c = bindtextdomain(PACKAGE, package().locale_dir().c_str()); + int e = errno; + if (e) { + lyxerr[Debug::DEBUG] + << BOOST_CURRENT_FUNCTION << '\n' + << "Error code: " << errno << '\n' + << "Lang, mess: " << lang_ << " " << m << '\n' + << "Directory : " << package().locale_dir() << '\n' + << "Rtn value : " << c << endl; + } +#ifdef WORDS_BIGENDIAN + static const char * codeset = "UCS-4BE"; +#else + static const char * codeset = "UCS-4LE"; +#endif + if (!bind_textdomain_codeset(PACKAGE, codeset)) { + lyxerr[Debug::DEBUG] + << BOOST_CURRENT_FUNCTION << '\n' + << "Error code: " << errno << '\n' + << "Codeset : " << codeset << '\n' + << endl; + } + + textdomain(PACKAGE); + char const * tmp = m.c_str(); + char const * msg = gettext(tmp); + docstring translated; + if (!msg || msg == tmp) { + if (!msg) + lyxerr << "Undefined result from gettext" << endl; + //else + // lyxerr << "Same as entered returned" << endl; + // Some english words have different translations, + // depending on context. In these cases the original + // string is augmented by context information (e.g. + // "To:[[as in 'From page x to page y']]" and + // "To:[[as in 'From format x to format y']]". + // This means that we need to filter out everything + // in double square brackets at the end of the + // string, otherwise the user sees bogus messages. + // If we are unable to honour the request we just + // return what we got in. + boost::smatch sub; + if (regex_match(m, sub, reg)) + translated = from_ascii(sub.str(1)); + else + translated = from_ascii(tmp); + } else { + lyxerr[Debug::DEBUG] << "We got a translation" << endl; + char_type const * ucs4 = reinterpret_cast(msg); + translated = ucs4; + } +#ifdef HAVE_LC_MESSAGES + setlocale(LC_MESSAGES, lang.c_str()); +#endif + setlocale(LC_CTYPE, oldCTYPE.c_str()); + return translated; + } +private: + /// + string lang_; +}; +#endif + +#else // ENABLE_NLS +// This is the dummy variant. +class Messages::Pimpl { +public: + Pimpl(string const &) {} + + ~Pimpl() {} + + docstring const get(string const & m) const + { + // See comment above + boost::smatch sub; + if (regex_match(m, sub, reg)) + return from_ascii(sub.str(1)); + else + return from_ascii(m); + } +}; +#endif + + +Messages::Messages() + : pimpl_(new Pimpl("")) +{} + + +Messages::Messages(string const & l) + : pimpl_(new Pimpl(l)) +{} + + +// We need this for the sake of scoped_ptr Messages::~Messages() -{ - mssg_gl.close(cat_gl); -} +{} -string const Messages::get(string const & msg) const +docstring const Messages::get(string const & msg) const { - return mssg_gl.get(cat_gl, 0, 0, msg); + return pimpl_->get(msg); } + + +} // namespace lyx