2007-04-26 04:41:58 +00:00
|
|
|
|
/* \file Messages.cpp
|
2003-04-23 18:47:21 +00:00
|
|
|
|
* This file is part of LyX, the document processor.
|
|
|
|
|
* Licence details can be found in the file COPYING.
|
|
|
|
|
*
|
|
|
|
|
* \author Lars Gullik Bj<EFBFBD>nnes
|
|
|
|
|
*
|
2003-08-23 00:17:00 +00:00
|
|
|
|
* Full author contact details are available in file CREDITS.
|
2003-04-23 18:47:21 +00:00
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
2007-04-26 04:41:58 +00:00
|
|
|
|
#include "Messages.h"
|
2007-01-14 09:55:08 +00:00
|
|
|
|
|
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
|
|
#include "support/docstring.h"
|
2005-07-17 11:58:54 +00:00
|
|
|
|
#include "support/environment.h"
|
2007-01-14 09:55:08 +00:00
|
|
|
|
#include "support/filetools.h"
|
2005-01-10 19:17:43 +00:00
|
|
|
|
#include "support/package.h"
|
2007-03-04 13:04:18 +00:00
|
|
|
|
#include "support/unicode.h"
|
2003-04-23 18:47:21 +00:00
|
|
|
|
|
2005-09-26 09:20:15 +00:00
|
|
|
|
#include <boost/current_function.hpp>
|
2004-10-19 09:11:02 +00:00
|
|
|
|
#include <boost/regex.hpp>
|
|
|
|
|
|
2005-10-18 13:34:51 +00:00
|
|
|
|
#include <cerrno>
|
2007-01-14 09:55:08 +00:00
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
boost::regex const reg("^([^\\[]*)\\[\\[[^\\]]*\\]\\]$");
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
#ifdef ENABLE_NLS
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_LOCALE_H
|
|
|
|
|
# include <locale.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
# if HAVE_GETTEXT
|
|
|
|
|
# include <libintl.h> // use the header already in the system *EK*
|
|
|
|
|
# else
|
|
|
|
|
# include "../intl/libintl.h"
|
|
|
|
|
# endif
|
2005-10-18 13:34:51 +00:00
|
|
|
|
|
2007-01-13 09:31:47 +00:00
|
|
|
|
using std::endl;
|
2007-01-14 09:55:08 +00:00
|
|
|
|
using std::make_pair;
|
|
|
|
|
using std::map;
|
2007-01-13 09:31:47 +00:00
|
|
|
|
using std::string;
|
2003-06-30 23:56:22 +00:00
|
|
|
|
|
2006-10-21 00:16:43 +00:00
|
|
|
|
namespace lyx {
|
|
|
|
|
|
|
|
|
|
using support::package;
|
|
|
|
|
using support::getEnv;
|
|
|
|
|
using support::setEnv;
|
2003-10-06 15:43:21 +00:00
|
|
|
|
|
2006-11-12 11:06:02 +00:00
|
|
|
|
|
2007-01-14 09:55:08 +00:00
|
|
|
|
// This version use the traditional gettext.
|
|
|
|
|
Messages::Messages(string const & l)
|
2007-02-24 22:54:19 +00:00
|
|
|
|
: lang_(l), warned_(false)
|
2007-01-14 09:55:08 +00:00
|
|
|
|
{
|
|
|
|
|
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);
|
2007-04-01 10:09:49 +00:00
|
|
|
|
LYXERR(Debug::DEBUG) << BOOST_CURRENT_FUNCTION
|
2007-01-14 09:55:08 +00:00
|
|
|
|
<< ": language(" << lang_ << ")" << endl;
|
|
|
|
|
}
|
2006-11-12 11:06:02 +00:00
|
|
|
|
|
2003-04-23 18:47:21 +00:00
|
|
|
|
|
2007-01-14 09:55:08 +00:00
|
|
|
|
docstring const Messages::get(string const & m) const
|
|
|
|
|
{
|
|
|
|
|
if (m.empty())
|
|
|
|
|
return docstring();
|
|
|
|
|
|
|
|
|
|
// Look for the translated string in the cache.
|
|
|
|
|
TranslationCache::iterator it = cache_.find(m);
|
|
|
|
|
if (it != cache_.end())
|
|
|
|
|
return it->second;
|
|
|
|
|
// The string was not found, use gettext to generate it:
|
|
|
|
|
|
|
|
|
|
// 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__)
|
2007-02-24 22:54:19 +00:00
|
|
|
|
if (!warned_ && !lc_msgs) {
|
|
|
|
|
warned_ = true;
|
2007-01-14 09:55:08 +00:00
|
|
|
|
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;
|
2007-04-06 13:09:54 +00:00
|
|
|
|
string const locale_dir = package().locale_dir().toFilesystemEncoding();
|
|
|
|
|
char const * c = bindtextdomain(PACKAGE, locale_dir.c_str());
|
2007-01-14 09:55:08 +00:00
|
|
|
|
int e = errno;
|
|
|
|
|
if (e) {
|
2007-04-01 10:09:49 +00:00
|
|
|
|
LYXERR(Debug::DEBUG)
|
2007-01-14 09:55:08 +00:00
|
|
|
|
<< BOOST_CURRENT_FUNCTION << '\n'
|
|
|
|
|
<< "Error code: " << errno << '\n'
|
|
|
|
|
<< "Lang, mess: " << lang_ << " " << m << '\n'
|
2007-04-06 13:09:54 +00:00
|
|
|
|
<< "Directory : " << package().locale_dir().absFilename() << '\n'
|
2007-01-14 09:55:08 +00:00
|
|
|
|
<< "Rtn value : " << c << endl;
|
|
|
|
|
}
|
|
|
|
|
|
2007-03-04 13:04:18 +00:00
|
|
|
|
if (!bind_textdomain_codeset(PACKAGE, ucs4_codeset)) {
|
2007-04-01 10:09:49 +00:00
|
|
|
|
LYXERR(Debug::DEBUG)
|
2007-01-14 09:55:08 +00:00
|
|
|
|
<< BOOST_CURRENT_FUNCTION << '\n'
|
|
|
|
|
<< "Error code: " << errno << '\n'
|
2007-03-04 13:04:18 +00:00
|
|
|
|
<< "Codeset : " << ucs4_codeset << '\n'
|
2007-01-14 09:55:08 +00:00
|
|
|
|
<< 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 {
|
2007-04-01 10:09:49 +00:00
|
|
|
|
LYXERR(Debug::DEBUG) << "We got a translation" << endl;
|
2007-01-14 09:55:08 +00:00
|
|
|
|
char_type const * ucs4 = reinterpret_cast<char_type const *>(msg);
|
|
|
|
|
translated = ucs4;
|
|
|
|
|
}
|
|
|
|
|
#ifdef HAVE_LC_MESSAGES
|
|
|
|
|
setlocale(LC_MESSAGES, lang.c_str());
|
|
|
|
|
#endif
|
|
|
|
|
setlocale(LC_CTYPE, oldCTYPE.c_str());
|
|
|
|
|
|
|
|
|
|
std::pair<TranslationCache::iterator, bool> result =
|
|
|
|
|
cache_.insert(std::make_pair(m, translated));
|
|
|
|
|
|
|
|
|
|
BOOST_ASSERT(result.second);
|
|
|
|
|
|
|
|
|
|
return result.first->second;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace lyx
|
|
|
|
|
|
|
|
|
|
#else // ENABLE_NLS
|
|
|
|
|
// This is the dummy variant.
|
|
|
|
|
|
|
|
|
|
using std::endl;
|
|
|
|
|
using std::make_pair;
|
|
|
|
|
using std::map;
|
|
|
|
|
using std::string;
|
|
|
|
|
|
|
|
|
|
namespace lyx {
|
|
|
|
|
|
|
|
|
|
Messages::Messages(string const & l) {}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
docstring const Messages::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);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace lyx
|
|
|
|
|
|
|
|
|
|
#endif
|
2003-04-27 22:11:45 +00:00
|
|
|
|
|
2003-04-24 22:09:46 +00:00
|
|
|
|
#if 0
|
2003-04-23 18:47:21 +00:00
|
|
|
|
|
2003-09-05 16:31:30 +00:00
|
|
|
|
-#include <locale>
|
2003-04-24 22:09:46 +00:00
|
|
|
|
|
2007-01-14 09:55:08 +00:00
|
|
|
|
namespace lyx {
|
|
|
|
|
|
2003-04-24 22:09:46 +00:00
|
|
|
|
// This version of the Pimpl utilizes the message capability of
|
2003-04-28 01:52:18 +00:00
|
|
|
|
// libstdc++ that is distributed with GNU G++.
|
2003-04-24 22:09:46 +00:00
|
|
|
|
class Messages::Pimpl {
|
|
|
|
|
public:
|
|
|
|
|
typedef std::messages<char>::catalog catalog;
|
|
|
|
|
|
2003-04-27 22:11:45 +00:00
|
|
|
|
Pimpl(string const & l)
|
|
|
|
|
: lang_(l),
|
2003-04-24 22:09:46 +00:00
|
|
|
|
loc_gl(lang_.c_str()),
|
|
|
|
|
mssg_gl(std::use_facet<std::messages<char> >(loc_gl))
|
|
|
|
|
{
|
|
|
|
|
//lyxerr << "Messages: language(" << l
|
2006-09-09 11:16:28 +00:00
|
|
|
|
// << ") in dir(" << dir << ")" << endl;
|
2003-04-24 22:09:46 +00:00
|
|
|
|
|
2007-04-06 13:09:54 +00:00
|
|
|
|
string const locale_dir = package().locale_dir().toFilesystemEncoding();
|
|
|
|
|
cat_gl = mssg_gl.open(PACKAGE, loc_gl, locale_dir.c_str());
|
2003-04-24 22:09:46 +00:00
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
~Pimpl()
|
|
|
|
|
{
|
|
|
|
|
mssg_gl.close(cat_gl);
|
|
|
|
|
}
|
|
|
|
|
|
2006-09-09 11:16:28 +00:00
|
|
|
|
docstring const get(string const & msg) const
|
2003-04-24 22:09:46 +00:00
|
|
|
|
{
|
|
|
|
|
return mssg_gl.get(cat_gl, 0, 0, msg);
|
|
|
|
|
}
|
|
|
|
|
private:
|
|
|
|
|
///
|
|
|
|
|
string lang_;
|
|
|
|
|
///
|
|
|
|
|
std::locale loc_gl;
|
|
|
|
|
///
|
|
|
|
|
std::messages<char> const & mssg_gl;
|
|
|
|
|
///
|
|
|
|
|
catalog cat_gl;
|
|
|
|
|
};
|
|
|
|
|
|
2007-01-14 09:55:08 +00:00
|
|
|
|
} // namespace lyx
|
2003-04-24 22:09:46 +00:00
|
|
|
|
|
|
|
|
|
#endif
|