]> git.lyx.org Git - lyx.git/blob - src/messages.C
fix crash and localization on MinGW/Windows platform
[lyx.git] / src / messages.C
1 /* \file messages.C
2  * This file is part of LyX, the document processor.
3  * Licence details can be found in the file COPYING.
4  *
5  * \author Lars Gullik Bjønnes
6  *
7  * Full author contact details are available in file CREDITS.
8  */
9
10 #include <config.h>
11
12 #include "debug.h"
13 #include "messages.h"
14 #include "support/filetools.h"
15 #include "support/environment.h"
16 #include "support/package.h"
17
18 #include <boost/current_function.hpp>
19 #include <boost/regex.hpp>
20
21 #include <cerrno>
22
23 using lyx::support::package;
24 using lyx::support::getEnv;
25 using lyx::support::setEnv;
26
27 using std::string;
28
29
30 #ifdef ENABLE_NLS
31
32
33 #if 0
34
35 -#include <locale>
36
37 // This version of the Pimpl utilizes the message capability of
38 // libstdc++ that is distributed with GNU G++.
39 class Messages::Pimpl {
40 public:
41         typedef std::messages<char>::catalog catalog;
42
43         Pimpl(string const & l)
44                 : lang_(l),
45                   loc_gl(lang_.c_str()),
46                   mssg_gl(std::use_facet<std::messages<char> >(loc_gl))
47         {
48                 //lyxerr << "Messages: language(" << l
49                 //       << ") in dir(" << dir << ")" << std::endl;
50
51                 cat_gl = mssg_gl.open(PACKAGE, loc_gl, package().locale_dir().c_str());
52
53         }
54
55         ~Pimpl()
56         {
57                 mssg_gl.close(cat_gl);
58         }
59
60         string const get(string const & msg) const
61         {
62                 return mssg_gl.get(cat_gl, 0, 0, msg);
63         }
64 private:
65         ///
66         string lang_;
67         ///
68         std::locale loc_gl;
69         ///
70         std::messages<char> const & mssg_gl;
71         ///
72         catalog cat_gl;
73 };
74 #else
75
76 #ifdef HAVE_LOCALE_H
77 #  include <locale.h>
78 #endif
79
80 #  if HAVE_GETTEXT
81 #    include <libintl.h>      // use the header already in the system *EK*
82 #  else
83 #    include "../intl/libintl.h"
84 #  endif
85
86 // This is a more traditional variant.
87 class Messages::Pimpl {
88 public:
89         Pimpl(string const & l)
90                 : lang_(l)
91         {
92                 if ( lang_.empty() ) {
93                         char const * lc_msgs = setlocale(LC_MESSAGES, NULL);
94                         lang_ = lc_msgs ? lc_msgs : "";
95                 }
96                 // strip off any encoding suffix, i.e., assume 8-bit po files
97                 string::size_type i = lang_.find(".");
98                 lang_ = lang_.substr(0, i);
99                 lyxerr[Debug::DEBUG] << BOOST_CURRENT_FUNCTION
100                                      << ": language(" << lang_ << ")" << std::endl;
101         }
102
103         ~Pimpl() {}
104
105         string const get(string const & m) const
106         {
107                 if (m.empty())
108                         return m;
109
110                 //string oldMSG = setlocale(LC_MESSAGES, NULL);
111                 // In this order, see support/filetools.C:
112                 string lang = getEnv("LC_ALL");
113                 if (lang.empty()) {
114                         lang = getEnv("LC_MESSAGES");
115                         if (lang.empty()) {
116                                 lang = getEnv("LANG");
117                                 if (lang.empty())
118                                         lang = "C";
119                         }
120                 }
121                 
122                 char const * lc_msgs = setlocale(LC_MESSAGES, lang_.c_str());
123 #ifndef _WIN32
124                 if (!lc_msgs)
125                         lyxerr << "Locale " << lang_ << " could not be set" << std::endl;
126 #endif
127                 // CTYPE controls what getmessage thinks what encoding the po file uses
128                 char const * lc_ctype = setlocale(LC_CTYPE, NULL);
129                 string oldCTYPE = lc_ctype ? lc_ctype : "";
130
131                 setlocale(LC_CTYPE, lang_.c_str());
132                 errno = 0;
133                 char const * c = bindtextdomain(PACKAGE, package().locale_dir().c_str());
134                 int e = errno;
135                 if (e) {
136                         lyxerr[Debug::DEBUG]
137                                 << BOOST_CURRENT_FUNCTION << '\n'
138                                 << "Error code: " << errno << '\n'
139                                 << "Lang, mess: " << lang_ << " " << m << '\n'
140                                 << "Directory : " << package().locale_dir() << '\n'
141                                 << "Rtn value : " << c << std::endl;
142                 }
143                 textdomain(PACKAGE);
144                 const char* msg = gettext(m.c_str());
145                 string translated(msg ? msg : m);
146                 // Some english words have different translations, depending
147                 // on context. In these cases the original string is
148                 // augmented by context information (e.g.
149                 // "To:[[as in 'From page x to page y']]" and
150                 // "To:[[as in 'From format x to format y']]".
151                 // This means that we need to filter out everything in
152                 // double square brackets at the end of the string,
153                 // otherwise the user sees bogus messages.
154                 // If we are unable to honour the request we just
155                 // return what we got in.
156                 static boost::regex const reg("^([^\\[]*)\\[\\[[^\\]]*\\]\\]$");
157                 boost::smatch sub;
158                 if (regex_match(translated, sub, reg))
159                         translated = sub.str(1);
160                 setlocale(LC_MESSAGES, lang.c_str());
161                 setlocale(LC_CTYPE, oldCTYPE.c_str());
162                 return translated;
163         }
164 private:
165         ///
166         string lang_;
167 };
168 #endif
169
170 #else // ENABLE_NLS
171 // This is the dummy variant.
172 class Messages::Pimpl {
173 public:
174         Pimpl(string const &) {}
175
176         ~Pimpl() {}
177
178         string const get(string const & m) const
179         {
180                 return m;
181         }
182 };
183 #endif
184
185
186 Messages::Messages()
187         : pimpl_(new Pimpl(""))
188 {}
189
190
191 Messages::Messages(string const & l)
192         : pimpl_(new Pimpl(l))
193 {}
194
195
196 // We need this for the sake of scoped_ptr
197 Messages::~Messages()
198 {}
199
200
201 string const Messages::get(string const & msg) const
202 {
203         return pimpl_->get(msg);
204 }