]> git.saurik.com Git - wxWidgets.git/blame - src/common/intl.cpp
check that wxTheApp != NULL in wxLog::GetActiveTarget
[wxWidgets.git] / src / common / intl.cpp
CommitLineData
c801d85f
KB
1/////////////////////////////////////////////////////////////////////////////
2// Name: intl.cpp
3// Purpose: Internationalization and localisation for wxWindows
4// Author: Vadim Zeitlin
5// Modified by:
6// Created: 29/01/98
7// RCS-ID: $Id$
8// Copyright: (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9// Licence: wxWindows license
10/////////////////////////////////////////////////////////////////////////////
11
12// ============================================================================
1678ad78 13// declaration
c801d85f
KB
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
20#ifdef __GNUG__
21#pragma implementation "intl.h"
22#endif
23
24// For compilers that support precompilation, includes "wx.h".
25#include "wx/wxprec.h"
26
27#ifdef __BORLANDC__
28#pragma hdrstop
29#endif
30
1678ad78
GL
31// standard headers
32#include <locale.h>
7502ba29 33
1678ad78
GL
34// wxWindows
35#include "wx/defs.h"
36#include "wx/string.h"
c801d85f
KB
37#include "wx/intl.h"
38#include "wx/file.h"
39#include "wx/log.h"
40#include "wx/utils.h"
41
42#include <stdlib.h>
43
44// ----------------------------------------------------------------------------
45// constants
46// ----------------------------------------------------------------------------
47
48// magic number identifying the .mo format file
49const uint32 MSGCATALOG_MAGIC = 0x950412de;
50const uint32 MSGCATALOG_MAGIC_SW = 0xde120495;
51
52// extension of ".mo" files
53#define MSGCATALOG_EXTENSION ".mo"
54
55// ----------------------------------------------------------------------------
1678ad78 56// global functions
c801d85f
KB
57// ----------------------------------------------------------------------------
58
59// suppress further error messages about missing translations
60// (if you don't have one catalog file, you wouldn't like to see the
61// error message for each string in it, so normally it's given only
62// once)
1678ad78 63void wxSuppressTransErrors();
c801d85f
KB
64
65// restore the logging
1678ad78 66void wxRestoreTransErrors();
c801d85f
KB
67
68// get the current state
1678ad78 69bool wxIsLoggingTransErrors();
c801d85f 70
1678ad78
GL
71// get the current locale object (## may be NULL!)
72extern wxLocale *wxSetLocale(wxLocale *pLocale);
c801d85f
KB
73
74// ----------------------------------------------------------------------------
75// wxMsgCatalog corresponds to one disk-file message catalog.
76//
77// This is a "low-level" class and is used only by wxLocale (that's why
78// it's designed to be stored in a linked list)
79// ----------------------------------------------------------------------------
80
81class wxMsgCatalog
82{
83public:
84 // ctor & dtor
85 wxMsgCatalog();
86 ~wxMsgCatalog();
87
88 // load the catalog from disk (szDirPrefix corresponds to language)
89 bool Load(const char *szDirPrefix, const char *szName);
90 bool IsLoaded() const { return m_pData != NULL; }
91
92 // get name of the catalog
93 const char *GetName() const { return m_pszName; }
94
95 // get the translated string: returns NULL if not found
96 const char *GetString(const char *sz) const;
97
98 // public variable pointing to the next element in a linked list (or NULL)
99 wxMsgCatalog *m_pNext;
100
101private:
102 // this implementation is binary compatible with GNU gettext() version 0.10
103
104 // an entry in the string table
105 struct wxMsgTableEntry
106 {
107 uint32 nLen; // length of the string
108 uint32 ofsString; // pointer to the string
109 };
110
111 // header of a .mo file
112 struct wxMsgCatalogHeader
113 {
114 uint32 magic, // offset +00: magic id
115 revision, // +04: revision
116 numStrings; // +08: number of strings in the file
117 uint32 ofsOrigTable, // +0C: start of original string table
118 ofsTransTable; // +10: start of translated string table
119 uint32 nHashSize, // +14: hash table size
120 ofsHashTable; // +18: offset of hash table start
121 };
122
123 // all data is stored here, NULL if no data loaded
124 uint8 *m_pData;
125
126 // data description
127 uint32 m_numStrings, // number of strings in this domain
128 m_nHashSize; // number of entries in hash table
129 uint32 *m_pHashTable; // pointer to hash table
130 wxMsgTableEntry *m_pOrigTable, // pointer to original strings
131 *m_pTransTable; // translated
132
133 const char *StringAtOfs(wxMsgTableEntry *pTable, uint32 index) const
134 { return (const char *)(m_pData + Swap(pTable[index].ofsString)); }
135
136 // utility functions
137 // calculate the hash value of given string
138 static inline uint32 GetHash(const char *sz);
139 // big<->little endian
140 inline uint32 Swap(uint32 ui) const;
141
142 // internal state
143 bool HasHashTable() const // true if hash table is present
144 { return m_nHashSize > 2 && m_pHashTable != NULL; }
145
146 bool m_bSwapped; // wrong endianness?
147
148 char *m_pszName; // name of the domain
149};
150
151// ============================================================================
152// implementation
153// ============================================================================
154
155// ----------------------------------------------------------------------------
156// wxMsgCatalog class
157// ----------------------------------------------------------------------------
158
159// calculate hash value using the so called hashpjw function by P.J. Weinberger
160// [see Aho/Sethi/Ullman, COMPILERS: Principles, Techniques and Tools]
161uint32 wxMsgCatalog::GetHash(const char *sz)
162{
163 #define HASHWORDBITS 32 // the length of uint32
164
165 uint32 hval = 0;
166 uint32 g;
167 while ( *sz != '\0' ) {
168 hval <<= 4;
169 hval += (uint32)*sz++;
170 g = hval & ((uint32)0xf << (HASHWORDBITS - 4));
171 if ( g != 0 ) {
172 hval ^= g >> (HASHWORDBITS - 8);
173 hval ^= g;
174 }
175 }
176
177 return hval;
178}
179
180// swap the 2 halves of 32 bit integer if needed
181uint32 wxMsgCatalog::Swap(uint32 ui) const
182{
183 return m_bSwapped ? (ui << 24) | ((ui & 0xff00) << 8) |
184 ((ui >> 8) & 0xff00) | (ui >> 24)
185 : ui;
186}
187
188wxMsgCatalog::wxMsgCatalog()
189{
190 m_pData = NULL;
191 m_pszName = NULL;
192}
193
194wxMsgCatalog::~wxMsgCatalog()
195{
196 DELETEA(m_pData);
197 DELETEA(m_pszName);
198}
199
200class NoTransErr
201{
202 public:
203 NoTransErr() { wxSuppressTransErrors(); }
204 ~NoTransErr() { wxRestoreTransErrors(); }
205};
206
207// open disk file and read in it's contents
208bool wxMsgCatalog::Load(const char *szDirPrefix, const char *szName)
209{
1678ad78
GL
210 // search order (assume language 'foo') is
211 // 1) $LC_PATH/foo/LC_MESSAGES (if LC_PATH set)
212 // 2) ./foo/LC_MESSAGES
213 // 3) ./foo
c801d85f
KB
214 // 4) . (Added by JACS)
215 //
216 // under UNIX we search also in:
1678ad78
GL
217 // 5) /usr/share/locale/foo/LC_MESSAGES (Linux)
218 // 6) /usr/lib/locale/foo/LC_MESSAGES (Solaris)
c801d85f
KB
219 #define MSG_PATH FILE_SEP_PATH + "LC_MESSAGES" PATH_SEP
220
221 wxString strPath("");
222 const char *pszLcPath = getenv("LC_PATH");
223 if ( pszLcPath != NULL )
224 strPath += pszLcPath + wxString(szDirPrefix) + MSG_PATH; // (1)
225
226 // NB: '<<' is unneeded between too literal strings:
227 // they are concatenated at compile time
1678ad78
GL
228 strPath += "./" + wxString(szDirPrefix) + MSG_PATH // (2)
229 + "./" + szDirPrefix + FILE_SEP_PATH + PATH_SEP // (3)
230 + "." + PATH_SEP
c801d85f 231 #ifdef __UNIX__
1678ad78
GL
232 "/usr/share/locale/" + szDirPrefix + MSG_PATH // (5)
233 "/usr/lib/locale/" + szDirPrefix + MSG_PATH // (6)
c801d85f
KB
234 #endif //UNIX
235 ;
236
237 wxString strFile = szName;
238 strFile += MSGCATALOG_EXTENSION;
239
240 // don't give translation errors here because the wxstd catalog might
241 // not yet be loaded (and it's normal)
242 //
243 // (we're using an object because we have several return paths)
244 NoTransErr noTransErr;
245
1678ad78
GL
246 wxLogVerbose("looking for catalog '%s' in path '%s'.",
247 szName, strPath.c_str());
c801d85f
KB
248
249 wxString strFullName;
250 if ( !wxFindFileInPath(&strFullName, strPath, strFile) ) {
1678ad78 251 wxLogWarning("catalog file for domain '%s' not found.", szName);
c801d85f
KB
252 return FALSE;
253 }
254
255 // open file
1678ad78
GL
256 wxLogVerbose("using catalog '%s' from '%s'.",
257 szName, strFullName.c_str());
c801d85f
KB
258
259 wxFile fileMsg(strFullName);
260 if ( !fileMsg.IsOpened() )
1678ad78 261 return FALSE;
c801d85f
KB
262
263 // get the file size
1678ad78
GL
264 off_t nSize = fileMsg.Length();
265 if ( nSize == wxInvalidOffset )
266 return FALSE;
c801d85f
KB
267
268 // read the whole file in memory
269 m_pData = new uint8[nSize];
270 if ( fileMsg.Read(m_pData, nSize) != nSize ) {
271 DELETEA(m_pData);
272 m_pData = NULL;
1678ad78 273 return FALSE;
c801d85f
KB
274 }
275
276 // examine header
1678ad78 277 bool bValid = (size_t)nSize > sizeof(wxMsgCatalogHeader);
c801d85f
KB
278
279 wxMsgCatalogHeader *pHeader;
280 if ( bValid ) {
281 pHeader = (wxMsgCatalogHeader *)m_pData;
282
283 // we'll have to swap all the integers if it's true
284 m_bSwapped = pHeader->magic == MSGCATALOG_MAGIC_SW;
285
286 // check the magic number
287 bValid = m_bSwapped || pHeader->magic == MSGCATALOG_MAGIC;
288 }
289
290 if ( !bValid ) {
291 // it's either too short or has incorrect magic number
1678ad78 292 wxLogWarning("'%s' is not a valid message catalog.", strFullName.c_str());
c801d85f
KB
293
294 DELETEA(m_pData);
295 m_pData = NULL;
296 return FALSE;
297 }
298
299 // initialize
300 m_numStrings = Swap(pHeader->numStrings);
1678ad78
GL
301 m_pOrigTable = (wxMsgTableEntry *)(m_pData +
302 Swap(pHeader->ofsOrigTable));
303 m_pTransTable = (wxMsgTableEntry *)(m_pData +
304 Swap(pHeader->ofsTransTable));
c801d85f
KB
305
306 m_nHashSize = Swap(pHeader->nHashSize);
307 m_pHashTable = (uint32 *)(m_pData + Swap(pHeader->ofsHashTable));
308
309 m_pszName = new char[strlen(szName) + 1];
310 strcpy(m_pszName, szName);
311
312 // everything is fine
313 return TRUE;
314}
315
316// search for a string
317const char *wxMsgCatalog::GetString(const char *szOrig) const
318{
319 if ( szOrig == NULL )
320 return NULL;
321
322 if ( HasHashTable() ) { // use hash table for lookup if possible
323 uint32 nHashVal = GetHash(szOrig);
324 uint32 nIndex = nHashVal % m_nHashSize;
325
326 uint32 nIncr = 1 + (nHashVal % (m_nHashSize - 2));
327
328 while ( TRUE ) {
329 uint32 nStr = Swap(m_pHashTable[nIndex]);
330 if ( nStr == 0 )
331 return NULL;
332
333 if ( strcmp(szOrig, StringAtOfs(m_pOrigTable, nStr - 1)) == 0 )
334 return StringAtOfs(m_pTransTable, nStr - 1);
335
336 if ( nIndex >= m_nHashSize - nIncr)
337 nIndex -= m_nHashSize - nIncr;
338 else
339 nIndex += nIncr;
340 }
341 }
342 else { // no hash table: use default binary search
343 uint32 bottom = 0,
344 top = m_numStrings,
345 current;
346 while ( bottom < top ) {
347 current = (bottom + top) / 2;
348 int res = strcmp(szOrig, StringAtOfs(m_pOrigTable, current));
349 if ( res < 0 )
350 top = current;
351 else if ( res > 0 )
352 bottom = current + 1;
353 else // found!
354 return StringAtOfs(m_pTransTable, current);
355 }
356 }
357
358 // not found
359 return NULL;
360}
361
362// ----------------------------------------------------------------------------
363// wxLocale
364// ----------------------------------------------------------------------------
365
23fcecf7 366wxLocale::wxLocale()
c801d85f 367{
23fcecf7
VZ
368 m_pszOldLocale = NULL;
369 m_pMsgCat = NULL;
370}
371
372// NB: this function has (desired) side effect of changing current locale
373bool wxLocale::Init(const char *szName,
374 const char *szShort,
375 const char *szLocale,
376 bool bLoadDefault)
377{
378 m_strLocale = szName;
379 m_strShort = szShort;
380
c801d85f
KB
381 // change current locale (default: same as long name)
382 if ( szLocale == NULL )
383 szLocale = szName;
384 m_pszOldLocale = setlocale(LC_ALL, szLocale);
385 if ( m_pszOldLocale == NULL )
1678ad78 386 wxLogError("locale '%s' can not be set.", szLocale);
c801d85f
KB
387
388 // the short name will be used to look for catalog files as well,
389 // so we need something here
390 if ( m_strShort.IsEmpty() ) {
1678ad78 391 // #### I don't know how these 2 letter abbreviations are formed,
c801d85f
KB
392 // this wild guess is almost surely wrong
393 m_strShort = wxToLower(szLocale[0]) + wxToLower(szLocale[1]);
394 }
395
396 // save the old locale to be able to restore it later
397 m_pOldLocale = wxSetLocale(this);
398
399 // load the default catalog with wxWindows standard messages
400 m_pMsgCat = NULL;
23fcecf7 401 bool bOk = TRUE;
c801d85f 402 if ( bLoadDefault )
23fcecf7
VZ
403 bOk = AddCatalog("wxstd");
404
405 return bOk;
c801d85f
KB
406}
407
408// clean up
409wxLocale::~wxLocale()
410{
411 // free memory
412 wxMsgCatalog *pTmpCat;
413 while ( m_pMsgCat != NULL ) {
414 pTmpCat = m_pMsgCat;
415 m_pMsgCat = m_pMsgCat->m_pNext;
416 delete pTmpCat;
417 }
418
419 // restore old locale
420 wxSetLocale(m_pOldLocale);
421 setlocale(LC_ALL, m_pszOldLocale);
422}
423
424// get the translation of given string in current locale
425const char *wxLocale::GetString(const char *szOrigString,
426 const char *szDomain) const
427{
428 wxASSERT( szOrigString != NULL ); // would be pretty silly
429
430 const char *pszTrans = NULL;
431
432 wxMsgCatalog *pMsgCat;
433 if ( szDomain != NULL ) {
434 pMsgCat = FindCatalog(szDomain);
435
436 // does the catalog exist?
437 if ( pMsgCat != NULL )
438 pszTrans = pMsgCat->GetString(szOrigString);
439 }
440 else {
441 // search in all domains
442 for ( pMsgCat = m_pMsgCat; pMsgCat != NULL; pMsgCat = pMsgCat->m_pNext ) {
443 pszTrans = pMsgCat->GetString(szOrigString);
444 if ( pszTrans != NULL ) // take the first found
445 break;
446 }
447 }
448
449 if ( pszTrans == NULL ) {
450 if ( wxIsLoggingTransErrors() ) {
451 // suppress further error messages
452 // (do it before LogWarning to prevent infinite recursion!)
453 wxSuppressTransErrors();
454
455 if ( szDomain != NULL )
1678ad78
GL
456 wxLogWarning("string '%s' not found in domain '%s' for locale '%s'.",
457 szOrigString, szDomain, m_strLocale.c_str());
c801d85f 458 else
1678ad78
GL
459 wxLogWarning("string '%s' not found in locale '%s'.",
460 szOrigString, m_strLocale.c_str());
c801d85f
KB
461 }
462
463 return szOrigString;
464 }
465 else
466 return pszTrans;
467}
468
469// find catalog by name in a linked list, return NULL if !found
470wxMsgCatalog *wxLocale::FindCatalog(const char *szDomain) const
471{
472// linear search in the linked list
473 wxMsgCatalog *pMsgCat;
474 for ( pMsgCat = m_pMsgCat; pMsgCat != NULL; pMsgCat = pMsgCat->m_pNext ) {
475 if ( Stricmp(pMsgCat->GetName(), szDomain) == 0 )
476 return pMsgCat;
477 }
478
479 return NULL;
480}
481
482// check if the given catalog is loaded
483bool wxLocale::IsLoaded(const char *szDomain) const
484{
485 return FindCatalog(szDomain) != NULL;
486}
487
488// add a catalog to our linked list
489bool wxLocale::AddCatalog(const char *szDomain)
490{
491 wxMsgCatalog *pMsgCat = new wxMsgCatalog;
492
493 if ( pMsgCat->Load(m_strShort, szDomain) ) {
494 // add it to the head of the list so that in GetString it will
495 // be searched before the catalogs added earlier
496 pMsgCat->m_pNext = m_pMsgCat;
497 m_pMsgCat = pMsgCat;
498
499 return TRUE;
500 }
501 else {
502 // don't add it because it couldn't be loaded anyway
503 delete pMsgCat;
504
505 return FALSE;
506 }
507}
508
509// ----------------------------------------------------------------------------
510// global functions and variables
511// ----------------------------------------------------------------------------
512
513// translation errors logging
514// --------------------------
515
516static bool gs_bGiveTransErrors = TRUE;
517
518void wxSuppressTransErrors()
519{
520 gs_bGiveTransErrors = FALSE;
521}
522
523void wxRestoreTransErrors()
524{
525 gs_bGiveTransErrors = TRUE;
526}
527
528bool wxIsLoggingTransErrors()
529{
530 return gs_bGiveTransErrors;
531}
532
533// retrieve/change current locale
534// ------------------------------
535
536// the current locale object
537wxLocale *g_pLocale = NULL;
538
1678ad78
GL
539wxLocale *wxGetLocale()
540{
541 return g_pLocale;
542}
543
c801d85f
KB
544wxLocale *wxSetLocale(wxLocale *pLocale)
545{
546 wxLocale *pOld = g_pLocale;
547 g_pLocale = pLocale;
548 return pOld;
549}