/////////////////////////////////////////////////////////////////////////////// // Name: common/fmapbase.cpp // Purpose: wxFontMapperBase class implementation // Author: Vadim Zeitlin // Modified by: // Created: 21.06.2003 (extracted from common/fontmap.cpp) // RCS-ID: $Id$ // Copyright: (c) 1999-2003 Vadim Zeitlin // License: wxWindows license /////////////////////////////////////////////////////////////////////////////// // ============================================================================ // declarations // ============================================================================ // ---------------------------------------------------------------------------- // headers // ---------------------------------------------------------------------------- // for compilers that support precompilation, includes "wx.h". #include "wx/wxprec.h" #ifdef __BORLANDC__ #pragma hdrstop #endif #if wxUSE_FONTMAP #ifndef WX_PRECOMP #include "wx/app.h" #include "wx/log.h" #include "wx/intl.h" #endif //WX_PRECOMP #if defined(__WXMSW__) #include "wx/msw/private.h" // includes windows.h for LOGFONT #include "wx/msw/winundef.h" #endif #include "wx/fontmap.h" #include "wx/fmappriv.h" #include "wx/apptrait.h" #include "wx/module.h" // wxMemoryConfig uses wxFileConfig #if wxUSE_CONFIG && wxUSE_FILECONFIG #include "wx/config.h" #include "wx/memconf.h" #endif // ---------------------------------------------------------------------------- // constants // ---------------------------------------------------------------------------- // encodings supported by GetEncodingDescription static wxFontEncoding gs_encodings[] = { wxFONTENCODING_ISO8859_1, wxFONTENCODING_ISO8859_2, wxFONTENCODING_ISO8859_3, wxFONTENCODING_ISO8859_4, wxFONTENCODING_ISO8859_5, wxFONTENCODING_ISO8859_6, wxFONTENCODING_ISO8859_7, wxFONTENCODING_ISO8859_8, wxFONTENCODING_ISO8859_9, wxFONTENCODING_ISO8859_10, wxFONTENCODING_ISO8859_11, wxFONTENCODING_ISO8859_12, wxFONTENCODING_ISO8859_13, wxFONTENCODING_ISO8859_14, wxFONTENCODING_ISO8859_15, wxFONTENCODING_KOI8, wxFONTENCODING_KOI8_U, wxFONTENCODING_CP932, wxFONTENCODING_CP936, wxFONTENCODING_CP949, wxFONTENCODING_CP950, wxFONTENCODING_CP1250, wxFONTENCODING_CP1251, wxFONTENCODING_CP1252, wxFONTENCODING_CP1253, wxFONTENCODING_CP1254, wxFONTENCODING_CP1255, wxFONTENCODING_CP1256, wxFONTENCODING_CP1257, wxFONTENCODING_CP437, wxFONTENCODING_UTF7, wxFONTENCODING_UTF8, wxFONTENCODING_UTF16, wxFONTENCODING_UTF16BE, wxFONTENCODING_UTF16LE, wxFONTENCODING_UTF32, wxFONTENCODING_UTF32BE, wxFONTENCODING_UTF32LE, wxFONTENCODING_EUC_JP, }; // the descriptions for them static const wxChar* gs_encodingDescs[] = { wxTRANSLATE( "Western European (ISO-8859-1)" ), wxTRANSLATE( "Central European (ISO-8859-2)" ), wxTRANSLATE( "Esperanto (ISO-8859-3)" ), wxTRANSLATE( "Baltic (old) (ISO-8859-4)" ), wxTRANSLATE( "Cyrillic (ISO-8859-5)" ), wxTRANSLATE( "Arabic (ISO-8859-6)" ), wxTRANSLATE( "Greek (ISO-8859-7)" ), wxTRANSLATE( "Hebrew (ISO-8859-8)" ), wxTRANSLATE( "Turkish (ISO-8859-9)" ), wxTRANSLATE( "Nordic (ISO-8859-10)" ), wxTRANSLATE( "Thai (ISO-8859-11)" ), wxTRANSLATE( "Indian (ISO-8859-12)" ), wxTRANSLATE( "Baltic (ISO-8859-13)" ), wxTRANSLATE( "Celtic (ISO-8859-14)" ), wxTRANSLATE( "Western European with Euro (ISO-8859-15)" ), wxTRANSLATE( "KOI8-R" ), wxTRANSLATE( "KOI8-U" ), wxTRANSLATE( "Windows Japanese (CP 932)" ), wxTRANSLATE( "Windows Chinese Simplified (CP 936)" ), wxTRANSLATE( "Windows Korean (CP 949)" ), wxTRANSLATE( "Windows Chinese Traditional (CP 950)" ), wxTRANSLATE( "Windows Central European (CP 1250)" ), wxTRANSLATE( "Windows Cyrillic (CP 1251)" ), wxTRANSLATE( "Windows Western European (CP 1252)" ), wxTRANSLATE( "Windows Greek (CP 1253)" ), wxTRANSLATE( "Windows Turkish (CP 1254)" ), wxTRANSLATE( "Windows Hebrew (CP 1255)" ), wxTRANSLATE( "Windows Arabic (CP 1256)" ), wxTRANSLATE( "Windows Baltic (CP 1257)" ), wxTRANSLATE( "Windows/DOS OEM (CP 437)" ), wxTRANSLATE( "Unicode 7 bit (UTF-7)" ), wxTRANSLATE( "Unicode 8 bit (UTF-8)" ), wxTRANSLATE( "Unicode 16 bit (UTF-16)" ), wxTRANSLATE( "Unicode 16 bit Big Endian (UTF-16BE)" ), wxTRANSLATE( "Unicode 16 bit Little Endian (UTF-16LE)" ), wxTRANSLATE( "Unicode 32 bit (UTF-32)" ), wxTRANSLATE( "Unicode 32 bit Big Endian (UTF-32BE)" ), wxTRANSLATE( "Unicode 32 bit Little Endian (UTF-32LE)" ), wxTRANSLATE( "Extended Unix Codepage for Japanese (EUC-JP)" ), }; // and the internal names (these are not translated on purpose!) static const wxChar* gs_encodingNames[] = { wxT( "iso-8859-1" ), wxT( "iso-8859-2" ), wxT( "iso-8859-3" ), wxT( "iso-8859-4" ), wxT( "iso-8859-5" ), wxT( "iso-8859-6" ), wxT( "iso-8859-7" ), wxT( "iso-8859-8" ), wxT( "iso-8859-9" ), wxT( "iso-8859-10" ), wxT( "iso-8859-11" ), wxT( "iso-8859-12" ), wxT( "iso-8859-13" ), wxT( "iso-8859-14" ), wxT( "iso-8859-15" ), wxT( "koi8-r" ), wxT( "koi8-u" ), wxT( "windows-932" ), wxT( "windows-936" ), wxT( "windows-949" ), wxT( "windows-950" ), wxT( "windows-1250" ), wxT( "windows-1251" ), wxT( "windows-1252" ), wxT( "windows-1253" ), wxT( "windows-1254" ), wxT( "windows-1255" ), wxT( "windows-1256" ), wxT( "windows-1257" ), wxT( "windows-437" ), wxT( "utf-7" ), wxT( "utf-8" ), wxT( "utf-16" ), wxT( "utf-16be" ), wxT( "utf-16le" ), wxT( "utf-32" ), wxT( "utf-32be" ), wxT( "utf-32le" ), wxT( "euc-jp" ), }; wxCOMPILE_TIME_ASSERT( WXSIZEOF(gs_encodingDescs) == WXSIZEOF(gs_encodings) && WXSIZEOF(gs_encodingNames) == WXSIZEOF(gs_encodings), EncodingsArraysNotInSync ); // ---------------------------------------------------------------------------- // private classes // ---------------------------------------------------------------------------- // clean up the font mapper object class wxFontMapperModule : public wxModule { public: wxFontMapperModule() : wxModule() { } virtual bool OnInit() { return TRUE; } virtual void OnExit() { delete wxFontMapper::Set(NULL); } DECLARE_DYNAMIC_CLASS(wxFontMapperModule) }; IMPLEMENT_DYNAMIC_CLASS(wxFontMapperModule, wxModule) // ============================================================================ // wxFontMapperBase implementation // ============================================================================ wxFontMapper *wxFontMapperBase::sm_instance = NULL; // ---------------------------------------------------------------------------- // ctor and dtor // ---------------------------------------------------------------------------- wxFontMapperBase::wxFontMapperBase() { #if wxUSE_CONFIG && wxUSE_FILECONFIG m_config = NULL; m_configIsDummy = FALSE; #endif // wxUSE_CONFIG } wxFontMapperBase::~wxFontMapperBase() { #if wxUSE_CONFIG && wxUSE_FILECONFIG if ( m_configIsDummy ) delete m_config; #endif // wxUSE_CONFIG } /* static */ wxFontMapper *wxFontMapperBase::Get() { if ( !sm_instance ) { wxAppTraits *traits = wxTheApp ? wxTheApp->GetTraits() : NULL; if ( traits ) { sm_instance = traits->CreateFontMapper(); wxASSERT_MSG( sm_instance, _T("wxAppTraits::CreateFontMapper() failed") ); } if ( !sm_instance ) { // last resort: we must create something because the existing code // relies on always having a valid font mapper object sm_instance = (wxFontMapper *)new wxFontMapperBase; } } return sm_instance; } /* static */ wxFontMapper *wxFontMapperBase::Set(wxFontMapper *mapper) { wxFontMapper *old = sm_instance; sm_instance = mapper; return old; } #if wxUSE_CONFIG && wxUSE_FILECONFIG // ---------------------------------------------------------------------------- // config usage customisation // ---------------------------------------------------------------------------- /* static */ const wxChar *wxFontMapperBase::GetDefaultConfigPath() { return FONTMAPPER_ROOT_PATH; } void wxFontMapperBase::SetConfigPath(const wxString& prefix) { wxCHECK_RET( !prefix.IsEmpty() && prefix[0] == wxCONFIG_PATH_SEPARATOR, wxT("an absolute path should be given to wxFontMapper::SetConfigPath()") ); m_configRootPath = prefix; } // ---------------------------------------------------------------------------- // get config object and path for it // ---------------------------------------------------------------------------- wxConfigBase *wxFontMapperBase::GetConfig() { if ( !m_config ) { // try the default m_config = wxConfig::Get(FALSE /*don't create on demand*/ ); if ( !m_config ) { // we still want to have a config object because otherwise we would // keep asking the user the same questions in the interactive mode, // so create a dummy config which won't write to any files/registry // but will allow us to remember the results of the questions at // least during this run m_config = new wxMemoryConfig; m_configIsDummy = TRUE; // VS: we can't call wxConfig::Set(m_config) here because that would // disable automatic wxConfig instance creation if this code was // called before wxApp::OnInit (this happens in wxGTK -- it sets // default wxFont encoding in wxApp::Initialize()) } } if ( m_configIsDummy && wxConfig::Get(FALSE) != NULL ) { // VS: in case we created dummy m_config (see above), we want to switch back // to the real one as soon as one becomes available. delete m_config; m_config = wxConfig::Get(FALSE); m_configIsDummy = FALSE; // FIXME: ideally, we should add keys from dummy config to the real one now, // but it is a low-priority task because typical wxWin application // either doesn't use wxConfig at all or creates wxConfig object in // wxApp::OnInit(), before any real interaction with the user takes // place... } return m_config; } const wxString& wxFontMapperBase::GetConfigPath() { if ( !m_configRootPath ) { // use the default m_configRootPath = GetDefaultConfigPath(); } return m_configRootPath; } // ---------------------------------------------------------------------------- // config helpers // ---------------------------------------------------------------------------- bool wxFontMapperBase::ChangePath(const wxString& pathNew, wxString *pathOld) { wxConfigBase *config = GetConfig(); if ( !config ) return FALSE; *pathOld = config->GetPath(); wxString path = GetConfigPath(); if ( path.IsEmpty() || path.Last() != wxCONFIG_PATH_SEPARATOR ) { path += wxCONFIG_PATH_SEPARATOR; } wxASSERT_MSG( !pathNew || (pathNew[0] != wxCONFIG_PATH_SEPARATOR), wxT("should be a relative path") ); path += pathNew; config->SetPath(path); return TRUE; } void wxFontMapperBase::RestorePath(const wxString& pathOld) { GetConfig()->SetPath(pathOld); } #endif // ---------------------------------------------------------------------------- // charset/encoding correspondence // ---------------------------------------------------------------------------- wxFontEncoding wxFontMapperBase::CharsetToEncoding(const wxString& charset, bool WXUNUSED(interactive)) { int enc = NonInteractiveCharsetToEncoding(charset); if ( enc == wxFONTENCODING_UNKNOWN ) { // we should return wxFONTENCODING_SYSTEM from here for unknown // encodings enc = wxFONTENCODING_SYSTEM; } return (wxFontEncoding)enc; } int wxFontMapperBase::NonInteractiveCharsetToEncoding(const wxString& charset) { wxFontEncoding encoding = wxFONTENCODING_SYSTEM; // we're going to modify it, make a copy wxString cs = charset; #if wxUSE_CONFIG && wxUSE_FILECONFIG // first try the user-defined settings wxFontMapperPathChanger path(this, FONTMAPPER_CHARSET_PATH); if ( path.IsOk() ) { wxConfigBase *config = GetConfig(); // do we have an encoding for this charset? long value = config->Read(charset, -1l); if ( value != -1 ) { if ( value == wxFONTENCODING_UNKNOWN ) { // don't try to find it, in particular don't ask the user return value; } if ( value >= 0 && value <= wxFONTENCODING_MAX ) { encoding = (wxFontEncoding)value; } else { wxLogDebug(wxT("corrupted config data: invalid encoding %ld for charset '%s' ignored"), value, charset.c_str()); } } if ( encoding == wxFONTENCODING_SYSTEM ) { // may be we have an alias? config->SetPath(FONTMAPPER_CHARSET_ALIAS_PATH); wxString alias = config->Read(charset); if ( !!alias ) { // yes, we do - use it instead cs = alias; } } } #endif // wxUSE_CONFIG // if didn't find it there, try to recognize it ourselves if ( encoding == wxFONTENCODING_SYSTEM ) { // trim any spaces cs.Trim(true); cs.Trim(false); // discard the optional quotes if ( !cs.empty() ) { if ( cs[0u] == _T('"') && cs.Last() == _T('"') ) { cs = wxString(cs.c_str(), cs.length() - 1); } } cs.MakeUpper(); if ( cs.empty() || cs == _T("US-ASCII") ) { encoding = wxFONTENCODING_DEFAULT; } else if ( cs == wxT("UTF-7") ) { encoding = wxFONTENCODING_UTF7; } else if ( cs == wxT("UTF-8") ) { encoding = wxFONTENCODING_UTF8; } else if ( cs == wxT("UTF-16") ) { encoding = wxFONTENCODING_UTF16; } else if ( cs == wxT("UTF-16BE") ) { encoding = wxFONTENCODING_UTF16BE; } else if ( cs == wxT("UTF-16LE") ) { encoding = wxFONTENCODING_UTF16LE; } else if ( cs == wxT("UTF-32") || cs == wxT("UCS-4") ) { encoding = wxFONTENCODING_UTF32; } else if ( cs == wxT("UTF-32BE") || cs == wxT("UCS-4BE") ) { encoding = wxFONTENCODING_UTF32BE; } else if ( cs == wxT("UTF-32LE") || cs == wxT("UCS-4LE") ) { encoding = wxFONTENCODING_UTF32LE; } else if ( cs == wxT("GB2312") ) { encoding = wxFONTENCODING_GB2312; } else if ( cs == wxT("BIG5") ) { encoding = wxFONTENCODING_BIG5; } else if ( cs == wxT("SJIS") || cs == wxT("SHIFT_JIS") || cs == wxT("SHIFT-JIS") ) { encoding = wxFONTENCODING_SHIFT_JIS; } else if ( cs == wxT("EUC-JP") || cs == wxT("EUC_JP") ) { encoding = wxFONTENCODING_EUC_JP; } else if ( cs == wxT("KOI8-R") || cs == wxT("KOI8-RU") ) { // although koi8-ru is not strictly speaking the same as koi8-r, // they are similar enough to make mapping it to koi8 better than // not recognizing it at all encoding = wxFONTENCODING_KOI8; } else if ( cs == wxT("KOI8-U") ) { encoding = wxFONTENCODING_KOI8_U; } else if ( cs.Left(3) == wxT("ISO") ) { // the dash is optional (or, to be exact, it is not, but // several brokenmails "forget" it) const wxChar *p = cs.c_str() + 3; if ( *p == wxT('-') ) p++; // printf( "iso %s\n", (const char*) cs.ToAscii() ); unsigned int value; if ( wxSscanf(p, wxT("8859-%u"), &value) == 1 ) { // printf( "value %d\n", (int)value ); // make it 0 based and check that it is strictly positive in // the process (no such thing as iso8859-0 encoding) if ( (value-- > 0) && (value < wxFONTENCODING_ISO8859_MAX - wxFONTENCODING_ISO8859_1) ) { // it's a valid ISO8859 encoding value += wxFONTENCODING_ISO8859_1; encoding = (wxFontEncoding)value; } } } else if ( cs.Left(4) == wxT("8859") ) { const wxChar *p = cs.c_str(); unsigned int value; if ( wxSscanf(p, wxT("8859-%u"), &value) == 1 ) { // printf( "value %d\n", (int)value ); // make it 0 based and check that it is strictly positive in // the process (no such thing as iso8859-0 encoding) if ( (value-- > 0) && (value < wxFONTENCODING_ISO8859_MAX - wxFONTENCODING_ISO8859_1) ) { // it's a valid ISO8859 encoding value += wxFONTENCODING_ISO8859_1; encoding = (wxFontEncoding)value; } } } else // check for Windows charsets { size_t len; if ( cs.Left(7) == wxT("WINDOWS") ) { len = 7; } else if ( cs.Left(2) == wxT("CP") ) { len = 2; } else // not a Windows encoding { len = 0; } if ( len ) { const wxChar *p = cs.c_str() + len; if ( *p == wxT('-') ) p++; int value; if ( wxSscanf(p, wxT("%u"), &value) == 1 ) { if ( value >= 1250 ) { value -= 1250; if ( value < wxFONTENCODING_CP12_MAX - wxFONTENCODING_CP1250 ) { // a valid Windows code page value += wxFONTENCODING_CP1250; encoding = (wxFontEncoding)value; } } switch ( value ) { case 932: encoding = wxFONTENCODING_CP932; break; case 936: encoding = wxFONTENCODING_CP936; break; case 949: encoding = wxFONTENCODING_CP949; break; case 950: encoding = wxFONTENCODING_CP950; break; } } } } //else: unknown } return encoding; } /* static */ size_t wxFontMapperBase::GetSupportedEncodingsCount() { return WXSIZEOF(gs_encodings); } /* static */ wxFontEncoding wxFontMapperBase::GetEncoding(size_t n) { wxCHECK_MSG( n < WXSIZEOF(gs_encodings), wxFONTENCODING_SYSTEM, _T("wxFontMapper::GetEncoding(): invalid index") ); return gs_encodings[n]; } /* static */ wxString wxFontMapperBase::GetEncodingDescription(wxFontEncoding encoding) { if ( encoding == wxFONTENCODING_DEFAULT ) { return _("Default encoding"); } const size_t count = WXSIZEOF(gs_encodingDescs); for ( size_t i = 0; i < count; i++ ) { if ( gs_encodings[i] == encoding ) { return wxGetTranslation(gs_encodingDescs[i]); } } wxString str; str.Printf(_("Unknown encoding (%d)"), encoding); return str; } /* static */ wxString wxFontMapperBase::GetEncodingName(wxFontEncoding encoding) { if ( encoding == wxFONTENCODING_DEFAULT ) { return _("default"); } const size_t count = WXSIZEOF(gs_encodingNames); for ( size_t i = 0; i < count; i++ ) { if ( gs_encodings[i] == encoding ) { return gs_encodingNames[i]; } } wxString str; str.Printf(_("unknown-%d"), encoding); return str; } /* static */ wxFontEncoding wxFontMapperBase::GetEncodingFromName(const wxString& name) { const size_t count = WXSIZEOF(gs_encodingNames); for ( size_t i = 0; i < count; i++ ) { if ( gs_encodingNames[i] == name ) { return gs_encodings[i]; } } if ( name == _("default") ) { return wxFONTENCODING_DEFAULT; } return wxFONTENCODING_MAX; } #endif // wxUSE_FONTMAP