1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/unix/mimetype.cpp 
   3 // Purpose:     classes and functions to manage MIME types 
   4 // Author:      Vadim Zeitlin 
   8 // Copyright:   (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr> 
   9 // Licence:     wxWindows licence (part of wxExtra library) 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // for compilers that support precompilation, includes "wx.h". 
  13 #include "wx/wxprec.h" 
  19 #if wxUSE_MIMETYPE && wxUSE_FILE 
  21 #include "wx/unix/mimetype.h" 
  24     #include "wx/dynarray.h" 
  25     #include "wx/string.h" 
  32 #include "wx/confbase.h" 
  36 #include "wx/tokenzr.h" 
  37 #include "wx/iconloc.h" 
  38 #include "wx/filename.h" 
  40 #include "wx/apptrait.h" 
  42 // other standard headers 
  52     wxMimeTextFile(const wxString
& fname
) 
  59        wxFFile 
file( m_fname 
); 
  63        size_t size 
= file
.Length(); 
  64        wxCharBuffer 
buffer( size 
); 
  65        file
.Read( (void*) (const char*) buffer
, size 
); 
  67        // Check for valid UTF-8 here? 
  68        wxString all 
= wxString::FromUTF8( buffer
, size 
); 
  70        wxStringTokenizer 
tok( all
, "\n" ); 
  71        while (tok
.HasMoreTokens()) 
  73           wxString t 
= tok
.GetNextToken(); 
  75           if ((!!t
) && (t
.Find( "comment" ) != 0) && (t
.Find( "#" ) != 0) && (t
.Find( "generic" ) != 0)) 
  81     unsigned int GetLineCount() const { return m_text
.GetCount(); } 
  82     wxString 
&GetLine( unsigned int line 
) { return m_text
[line
]; } 
  84     int pIndexOf(const wxString
& sSearch
, 
  85                  bool bIncludeComments 
= false, 
  88         wxString sTest 
= sSearch
; 
  90         for(size_t i 
= iStart
; i 
< GetLineCount(); i
++) 
  92             wxString sLine 
= GetLine(i
); 
  93             if(bIncludeComments 
|| ! sLine
.StartsWith(wxT("#"))) 
  95                 if(sLine
.StartsWith(sTest
)) 
 102     wxString 
GetVerb(size_t i
) 
 104         if (i 
> GetLineCount() ) 
 105             return wxEmptyString
; 
 107         wxString sTmp 
= GetLine(i
).BeforeFirst(wxT('=')); 
 111     wxString 
GetCmd(size_t i
) 
 113         if (i 
> GetLineCount() ) 
 114             return wxEmptyString
; 
 116         wxString sTmp 
= GetLine(i
).AfterFirst(wxT('=')); 
 121     wxArrayString m_text
; 
 125 // ---------------------------------------------------------------------------- 
 127 // ---------------------------------------------------------------------------- 
 129 // MIME code tracing mask 
 130 #define TRACE_MIME wxT("mime") 
 133 // Read a XDG *.desktop file of type 'Application' 
 134 void wxMimeTypesManagerImpl::LoadXDGApp(const wxString
& filename
) 
 136     wxLogTrace(TRACE_MIME
, wxT("loading XDG file %s"), filename
.c_str()); 
 138     wxMimeTextFile 
file(filename
); 
 142     // Here, only type 'Application' should be considered. 
 143     int nIndex 
= file
.pIndexOf( "Type=" ); 
 144     if (nIndex 
!= wxNOT_FOUND 
&& file
.GetCmd(nIndex
) != "application") 
 147     // The hidden entry specifies a file to be ignored. 
 148     nIndex 
= file
.pIndexOf( "Hidden=" ); 
 149     if (nIndex 
!= wxNOT_FOUND 
&& file
.GetCmd(nIndex
) == "true") 
 152     // Semicolon separated list of mime types handled by the application. 
 153     nIndex 
= file
.pIndexOf( wxT("MimeType=") ); 
 154     if (nIndex 
== wxNOT_FOUND
) 
 156     wxString mimetypes 
= file
.GetCmd (nIndex
); 
 158     // Name of the application 
 160     nIndex 
= wxNOT_FOUND
; 
 161 #if wxUSE_INTL // try "Name[locale name]" first 
 162     wxLocale 
*locale 
= wxGetLocale(); 
 164         nIndex 
= file
.pIndexOf(_T("Name[")+locale
->GetName()+_T("]=")); 
 166     if(nIndex 
== wxNOT_FOUND
) 
 167         nIndex 
= file
.pIndexOf( wxT("Name=") ); 
 168     if(nIndex 
!= wxNOT_FOUND
) 
 169         nameapp 
= file
.GetCmd(nIndex
); 
 171     // Icon of the application. 
 172     wxString nameicon
, namemini
; 
 173     nIndex 
= wxNOT_FOUND
; 
 174 #if wxUSE_INTL // try "Icon[locale name]" first 
 176         nIndex 
= file
.pIndexOf(_T("Icon[")+locale
->GetName()+_T("]=")); 
 178     if(nIndex 
== wxNOT_FOUND
) 
 179         nIndex 
= file
.pIndexOf( wxT("Icon=") ); 
 180     if(nIndex 
!= wxNOT_FOUND
) { 
 181         nameicon 
= wxString(wxT("--icon ")) + file
.GetCmd(nIndex
); 
 182         namemini 
= wxString(wxT("--miniicon ")) + file
.GetCmd(nIndex
); 
 185     // Replace some of the field code in the 'Exec' entry. 
 186     // TODO: deal with %d, %D, %n, %N, %k and %v (but last one is deprecated) 
 187     nIndex 
= file
.pIndexOf( wxT("Exec=") ); 
 188     if (nIndex 
== wxNOT_FOUND
) 
 190     wxString sCmd 
= file
.GetCmd(nIndex
); 
 191     // we expect %f; others including  %F and %U and %u are possible 
 192     sCmd
.Replace(wxT("%F"), wxT("%f")); 
 193     sCmd
.Replace(wxT("%U"), wxT("%f")); 
 194     sCmd
.Replace(wxT("%u"), wxT("%f")); 
 195     if (0 == sCmd
.Replace ( wxT("%f"), wxT("%s") )) 
 196         sCmd 
= sCmd 
+ wxT(" %s"); 
 197     sCmd
.Replace(wxT("%c"), nameapp
); 
 198     sCmd
.Replace(wxT("%i"), nameicon
); 
 199     sCmd
.Replace(wxT("%m"), namemini
); 
 201     wxStringTokenizer 
tokenizer(mimetypes
, _T(";")); 
 202     while(tokenizer
.HasMoreTokens()) { 
 203         wxString mimetype 
= tokenizer
.GetNextToken().Lower(); 
 204         nIndex 
= m_aTypes
.Index(mimetype
); 
 205         if(nIndex 
!= wxNOT_FOUND
) { // is this a known MIME type? 
 206             wxMimeTypeCommands
* entry 
= m_aEntries
[nIndex
]; 
 207             entry
->AddOrReplaceVerb(wxT("open"), sCmd
); 
 212 void wxMimeTypesManagerImpl::LoadXDGAppsFilesFromDir(const wxString
& dirname
) 
 214     // Don't complain if we don't have permissions to read - it confuses users 
 217     if(! wxDir::Exists(dirname
)) 
 220     if ( !dir
.IsOpened() ) 
 224     // Look into .desktop files 
 225     bool cont 
= dir
.GetFirst(&filename
, _T("*.desktop"), wxDIR_FILES
); 
 228         wxFileName 
p(dirname
, filename
); 
 229         LoadXDGApp( p
.GetFullPath() ); 
 230         cont 
= dir
.GetNext(&filename
); 
 234     // RR: I'm not sure this makes any sense. On my system we'll just 
 235     //     scan the YAST2 and other useless directories 
 237     // Look recursively into subdirs 
 238     cont 
= dir
.GetFirst(&filename
, wxEmptyString
, wxDIR_DIRS
); 
 241         wxFileName 
p(dirname
, wxEmptyString
); 
 242         p
.AppendDir(filename
); 
 243         LoadXDGAppsFilesFromDir( p
.GetPath() ); 
 244         cont 
= dir
.GetNext(&filename
); 
 250 void wxMimeTypesManagerImpl::LoadXDGGlobs(const wxString
& filename
) 
 252     if ( !wxFileName::FileExists(filename
) ) 
 255     wxLogTrace(TRACE_MIME
, wxT("loading XDG globs file from %s"), filename
.c_str()); 
 257     wxMimeTextFile 
file(filename
); 
 262     for (i 
= 0; i 
< file
.GetLineCount(); i
++) 
 264        wxStringTokenizer 
tok( file
.GetLine(i
), ":" ); 
 265        wxString mime 
= tok
.GetNextToken(); 
 266        wxString ext 
= tok
.GetNextToken(); 
 271        AddToMimeData(mime
, wxEmptyString
, NULL
, exts
, wxEmptyString
, true ); 
 275 // ---------------------------------------------------------------------------- 
 276 // wxFileTypeImpl (Unix) 
 277 // ---------------------------------------------------------------------------- 
 279 wxString 
wxFileTypeImpl::GetExpandedCommand(const wxString 
& verb
, const wxFileType::MessageParameters
& params
) const 
 283     while ( (i 
< m_index
.GetCount() ) && sTmp
.empty() ) 
 285         sTmp 
= m_manager
->GetCommand( verb
, m_index
[i
] ); 
 289     return wxFileType::ExpandCommand(sTmp
, params
); 
 292 bool wxFileTypeImpl::GetIcon(wxIconLocation 
*iconLoc
) const 
 296     while ( (i 
< m_index
.GetCount() ) && sTmp
.empty() ) 
 298         sTmp 
= m_manager
->m_aIcons
[m_index
[i
]]; 
 307         iconLoc
->SetFileName(sTmp
); 
 313 bool wxFileTypeImpl::GetMimeTypes(wxArrayString
& mimeTypes
) const 
 316     size_t nCount 
= m_index
.GetCount(); 
 317     for (size_t i 
= 0; i 
< nCount
; i
++) 
 318         mimeTypes
.Add(m_manager
->m_aTypes
[m_index
[i
]]); 
 323 size_t wxFileTypeImpl::GetAllCommands(wxArrayString 
*verbs
, 
 324                                   wxArrayString 
*commands
, 
 325                                   const wxFileType::MessageParameters
& params
) const 
 327     wxString vrb
, cmd
, sTmp
; 
 329     wxMimeTypeCommands 
* sPairs
; 
 331     // verbs and commands have been cleared already in mimecmn.cpp... 
 332     // if we find no entries in the exact match, try the inexact match 
 333     for (size_t n 
= 0; ((count 
== 0) && (n 
< m_index
.GetCount())); n
++) 
 335         // list of verb = command pairs for this mimetype 
 336         sPairs 
= m_manager
->m_aEntries 
[m_index
[n
]]; 
 338         for ( i 
= 0; i 
< sPairs
->GetCount(); i
++ ) 
 340             vrb 
= sPairs
->GetVerb(i
); 
 341             // some gnome entries have "." inside 
 342             vrb 
= vrb
.AfterLast(wxT('.')); 
 343             cmd 
= sPairs
->GetCmd(i
); 
 346                  cmd 
= wxFileType::ExpandCommand(cmd
, params
); 
 348                  if ( vrb
.IsSameAs(wxT("open"))) 
 351                         verbs
->Insert(vrb
, 0u); 
 353                         commands 
->Insert(cmd
, 0u); 
 369 bool wxFileTypeImpl::GetExtensions(wxArrayString
& extensions
) 
 371     const wxString strExtensions 
= m_manager
->GetExtension(m_index
[0]); 
 374     // one extension in the space or comma-delimited list 
 376     wxString::const_iterator end 
= strExtensions
.end(); 
 377     for ( wxString::const_iterator p 
= strExtensions
.begin(); /* nothing */; ++p 
) 
 379         if ( p 
== end 
|| *p 
== wxT(' ') || *p 
== wxT(',') ) 
 381             if ( !strExt
.empty() ) 
 383                 extensions
.Add(strExt
); 
 386             //else: repeated spaces 
 387             // (shouldn't happen, but it's not that important if it does happen) 
 392         else if ( *p 
== wxT('.') ) 
 394             // remove the dot from extension (but only if it's the first char) 
 395             if ( !strExt
.empty() ) 
 399             //else: no, don't append it 
 410 // set an arbitrary command: 
 411 // could adjust the code to ask confirmation if it already exists and 
 412 // overwriteprompt is true, but this is currently ignored as *Associate* has 
 413 // no overwrite prompt 
 415 wxFileTypeImpl::SetCommand(const wxString
& cmd
, 
 416                            const wxString
& verb
, 
 417                            bool WXUNUSED(overwriteprompt
)) 
 419     wxArrayString strExtensions
; 
 420     wxString strDesc
, strIcon
; 
 422     wxArrayString strTypes
; 
 423     GetMimeTypes(strTypes
); 
 424     if ( strTypes
.IsEmpty() ) 
 427     wxMimeTypeCommands 
*entry 
= new wxMimeTypeCommands(); 
 428     entry
->Add(verb 
+ wxT("=")  + cmd 
+ wxT(" %s ")); 
 431     size_t nCount 
= strTypes
.GetCount(); 
 432     for ( size_t i 
= 0; i 
< nCount
; i
++ ) 
 434         if ( m_manager
->DoAssociation
 
 443             // DoAssociation() took ownership of entry, don't delete it below 
 454 // ignore index on the grounds that we only have one icon in a Unix file 
 455 bool wxFileTypeImpl::SetDefaultIcon(const wxString
& strIcon
, int WXUNUSED(index
)) 
 460     wxArrayString strExtensions
; 
 463     wxArrayString strTypes
; 
 464     GetMimeTypes(strTypes
); 
 465     if ( strTypes
.IsEmpty() ) 
 468     wxMimeTypeCommands 
*entry 
= new wxMimeTypeCommands(); 
 470     size_t nCount 
= strTypes
.GetCount(); 
 471     for ( size_t i 
= 0; i 
< nCount
; i
++ ) 
 473         if ( m_manager
->DoAssociation
 
 482             // we don't need to free entry now, DoAssociation() took ownership 
 494 // ---------------------------------------------------------------------------- 
 495 // wxMimeTypesManagerImpl (Unix) 
 496 // ---------------------------------------------------------------------------- 
 498 wxMimeTypesManagerImpl::wxMimeTypesManagerImpl() 
 500     m_initialized 
= false; 
 503 void wxMimeTypesManagerImpl::InitIfNeeded() 
 505     if ( !m_initialized 
) 
 507         // set the flag first to prevent recursion 
 508         m_initialized 
= true; 
 510         wxString wm 
= wxTheApp
->GetTraits()->GetDesktopEnvironment(); 
 512         if (wm 
== wxT("KDE")) 
 513             Initialize( wxMAILCAP_KDE  
); 
 514         else if (wm 
== wxT("GNOME")) 
 515             Initialize( wxMAILCAP_GNOME 
); 
 523 // read system and user mailcaps and other files 
 524 void wxMimeTypesManagerImpl::Initialize(int mailcapStyles
, 
 525                                         const wxString
& sExtraDir
) 
 528     // XDG tables are never installed on OpenVMS 
 532     // Read MIME type - extension associations 
 533     LoadXDGGlobs( "/usr/share/mime/globs" ); 
 534     LoadXDGGlobs( "/usr/local/share/mime/globs" ); 
 536     // Load desktop files for XDG, and then override them with the defaults. 
 537     // We will override them one desktop file at a time, rather 
 538     // than one mime type at a time, but it should be a reasonable 
 541         wxString xdgDataHome 
= wxGetenv("XDG_DATA_HOME"); 
 542         if ( xdgDataHome
.empty() ) 
 543             xdgDataHome 
= wxGetHomeDir() + "/.local/share"; 
 544         wxString xdgDataDirs 
= wxGetenv("XDG_DATA_DIRS"); 
 545         if ( xdgDataDirs
.empty() ) 
 547             xdgDataDirs 
= "/usr/local/share:/usr/share"; 
 548             if (mailcapStyles 
& wxMAILCAP_GNOME
) 
 549                 xdgDataDirs 
+= ":/usr/share/gnome:/opt/gnome/share"; 
 550             if (mailcapStyles 
& wxMAILCAP_KDE
) 
 551                 xdgDataDirs 
+= ":/usr/share/kde3:/opt/kde3/share"; 
 553         if ( !sExtraDir
.empty() ) 
 556            xdgDataDirs 
+= sExtraDir
; 
 560         wxStringTokenizer 
tokenizer(xdgDataDirs
, ":"); 
 561         while ( tokenizer
.HasMoreTokens() ) 
 563             wxString p 
= tokenizer
.GetNextToken(); 
 566         dirs
.insert(dirs
.begin(), xdgDataHome
); 
 568         wxString defaultsList
; 
 570         for (i 
= 0; i 
< dirs
.GetCount(); i
++) 
 572             wxString f 
= dirs
[i
]; 
 573             if (f
.Last() != '/') f 
+= '/'; 
 574             f 
+= "applications/defaults.list"; 
 582         // Load application files and associate them to corresponding mime types. 
 583         size_t nDirs 
= dirs
.GetCount(); 
 584         for (size_t nDir 
= 0; nDir 
< nDirs
; nDir
++) 
 586             wxString dirStr 
= dirs
[nDir
]; 
 587             if (dirStr
.Last() != '/') dirStr 
+= '/'; 
 588             dirStr 
+= "applications"; 
 589             LoadXDGAppsFilesFromDir(dirStr
); 
 592         if (!defaultsList
.IsEmpty()) 
 594             wxArrayString deskTopFilesSeen
; 
 596             wxMimeTextFile 
textfile(defaultsList
); 
 597             if ( textfile
.Open() ) 
 599                 int nIndex 
= textfile
.pIndexOf( wxT("[Default Applications]") ); 
 600                 if (nIndex 
!= wxNOT_FOUND
) 
 602                     for (i 
= nIndex
+1; i 
< textfile
.GetLineCount(); i
++) 
 604                         if (textfile
.GetLine(i
).Find(wxT("=")) != wxNOT_FOUND
) 
 606                             wxString desktopFile 
= textfile
.GetCmd(i
); 
 608                             if (deskTopFilesSeen
.Index(desktopFile
) == wxNOT_FOUND
) 
 610                                 deskTopFilesSeen
.Add(desktopFile
); 
 612                                 for (j 
= 0; j 
< dirs
.GetCount(); j
++) 
 614                                     wxString desktopPath 
= dirs
[j
]; 
 615                                     if (desktopPath
.Last() != '/') desktopPath 
+= '/'; 
 616                                     desktopPath 
+= "applications/"; 
 617                                     desktopPath 
+= desktopFile
; 
 619                                     if (wxFileExists(desktopPath
)) 
 620                                         LoadXDGApp(desktopPath
); 
 632 // clear data so you can read another group of WM files 
 633 void wxMimeTypesManagerImpl::ClearData() 
 637     m_aExtensions
.Clear(); 
 638     m_aDescriptions
.Clear(); 
 640     WX_CLEAR_ARRAY(m_aEntries
); 
 644 wxMimeTypesManagerImpl::~wxMimeTypesManagerImpl() 
 649 wxFileType 
* wxMimeTypesManagerImpl::Associate(const wxFileTypeInfo
& ftInfo
) 
 653     wxString strType 
= ftInfo
.GetMimeType(); 
 654     wxString strDesc 
= ftInfo
.GetDescription(); 
 655     wxString strIcon 
= ftInfo
.GetIconFile(); 
 657     wxMimeTypeCommands 
*entry 
= new wxMimeTypeCommands(); 
 659     if ( ! ftInfo
.GetOpenCommand().empty()) 
 660         entry
->Add(wxT("open=")  + ftInfo
.GetOpenCommand() + wxT(" %s ")); 
 661     if ( ! ftInfo
.GetPrintCommand().empty()) 
 662         entry
->Add(wxT("print=") + ftInfo
.GetPrintCommand() + wxT(" %s ")); 
 664     // now find where these extensions are in the data store and remove them 
 665     wxArrayString sA_Exts 
= ftInfo
.GetExtensions(); 
 666     wxString sExt
, sExtStore
; 
 668     size_t nExtCount 
= sA_Exts
.GetCount(); 
 669     for (i
=0; i 
< nExtCount
; i
++) 
 671         sExt 
= sA_Exts
.Item(i
); 
 673         // clean up to just a space before and after 
 674         sExt
.Trim().Trim(false); 
 675         sExt 
= wxT(' ') + sExt 
+ wxT(' '); 
 676         size_t nCount 
= m_aExtensions
.GetCount(); 
 677         for (nIndex 
= 0; nIndex 
< nCount
; nIndex
++) 
 679             sExtStore 
= m_aExtensions
.Item(nIndex
); 
 680             if (sExtStore
.Replace(sExt
, wxT(" ") ) > 0) 
 681                 m_aExtensions
.Item(nIndex
) = sExtStore
; 
 685     if ( !DoAssociation(strType
, strIcon
, entry
, sA_Exts
, strDesc
) ) 
 688     return GetFileTypeFromMimeType(strType
); 
 691 bool wxMimeTypesManagerImpl::DoAssociation(const wxString
& strType
, 
 692                                            const wxString
& strIcon
, 
 693                                            wxMimeTypeCommands 
*entry
, 
 694                                            const wxArrayString
& strExtensions
, 
 695                                            const wxString
& strDesc
) 
 697     int nIndex 
= AddToMimeData(strType
, strIcon
, entry
, strExtensions
, strDesc
, true); 
 699     if ( nIndex 
== wxNOT_FOUND 
) 
 705 int wxMimeTypesManagerImpl::AddToMimeData(const wxString
& strType
, 
 706                                           const wxString
& strIcon
, 
 707                                           wxMimeTypeCommands 
*entry
, 
 708                                           const wxArrayString
& strExtensions
, 
 709                                           const wxString
& strDesc
, 
 710                                           bool replaceExisting
) 
 714     // ensure mimetype is always lower case 
 715     wxString mimeType 
= strType
.Lower(); 
 717     // is this a known MIME type? 
 718     int nIndex 
= m_aTypes
.Index(mimeType
); 
 719     if ( nIndex 
== wxNOT_FOUND 
) 
 721         // We put MIME types containing  "application" at the end, so that 
 722         // if the MIME type for the extention "htm" is searched for, it will 
 723         // rather find "text/html" than "application/x-mozilla-bookmarks". 
 724         if (mimeType
.Find( "application" ) == 0) 
 727            m_aTypes
.Add(mimeType
); 
 728            m_aIcons
.Add(strIcon
); 
 729            m_aEntries
.Add(entry 
? entry 
: new wxMimeTypeCommands
); 
 731            // change nIndex so we can use it below to add the extensions 
 732            m_aExtensions
.Add(wxEmptyString
); 
 733            nIndex 
= m_aExtensions
.size() - 1; 
 735            m_aDescriptions
.Add(strDesc
); 
 740            m_aTypes
.Insert(mimeType
,0); 
 741            m_aIcons
.Insert(strIcon
,0); 
 742            m_aEntries
.Insert(entry 
? entry 
: new wxMimeTypeCommands
,0); 
 744            // change nIndex so we can use it below to add the extensions 
 745            m_aExtensions
.Insert(wxEmptyString
,0); 
 748            m_aDescriptions
.Insert(strDesc
,0); 
 751     else // yes, we already have it 
 753         if ( replaceExisting 
) 
 755             // if new description change it 
 756             if ( !strDesc
.empty()) 
 757                 m_aDescriptions
[nIndex
] = strDesc
; 
 759             // if new icon change it 
 760             if ( !strIcon
.empty()) 
 761                 m_aIcons
[nIndex
] = strIcon
; 
 765                 delete m_aEntries
[nIndex
]; 
 766                 m_aEntries
[nIndex
] = entry
; 
 769         else // add data we don't already have ... 
 771             // if new description add only if none 
 772             if ( m_aDescriptions
[nIndex
].empty() ) 
 773                 m_aDescriptions
[nIndex
] = strDesc
; 
 775             // if new icon and no existing icon 
 776             if ( m_aIcons
[nIndex
].empty() ) 
 777                 m_aIcons
[nIndex
] = strIcon
; 
 779             // add any new entries... 
 782                 wxMimeTypeCommands 
*entryOld 
= m_aEntries
[nIndex
]; 
 784                 size_t count 
= entry
->GetCount(); 
 785                 for ( size_t i 
= 0; i 
< count
; i
++ ) 
 787                     const wxString
& verb 
= entry
->GetVerb(i
); 
 788                     if ( !entryOld
->HasVerb(verb
) ) 
 790                         entryOld
->AddOrReplaceVerb(verb
, entry
->GetCmd(i
)); 
 794                 // as we don't store it anywhere, it won't be deleted later as 
 795                 // usual -- do it immediately instead 
 801     // always add the extensions to this mimetype 
 802     wxString
& exts 
= m_aExtensions
[nIndex
]; 
 804     // add all extensions we don't have yet 
 806     size_t count 
= strExtensions
.GetCount(); 
 807     for ( size_t i 
= 0; i 
< count
; i
++ ) 
 809         ext 
= strExtensions
[i
]; 
 812         if ( exts
.Find(ext
) == wxNOT_FOUND 
) 
 818     // check data integrity 
 819     wxASSERT( m_aTypes
.GetCount() == m_aEntries
.GetCount() && 
 820               m_aTypes
.GetCount() == m_aExtensions
.GetCount() && 
 821               m_aTypes
.GetCount() == m_aIcons
.GetCount() && 
 822               m_aTypes
.GetCount() == m_aDescriptions
.GetCount() ); 
 827 wxFileType 
* wxMimeTypesManagerImpl::GetFileTypeFromExtension(const wxString
& ext
) 
 834     size_t count 
= m_aExtensions
.GetCount(); 
 835     for ( size_t n 
= 0; n 
< count
; n
++ ) 
 837         wxStringTokenizer 
tk(m_aExtensions
[n
], wxT(' ')); 
 839         while ( tk
.HasMoreTokens() ) 
 841             // consider extensions as not being case-sensitive 
 842             if ( tk
.GetNextToken().IsSameAs(ext
, false /* no case */) ) 
 845                 wxFileType 
*fileType 
= new wxFileType
; 
 846                 fileType
->m_impl
->Init(this, n
); 
 856 wxFileType 
* wxMimeTypesManagerImpl::GetFileTypeFromMimeType(const wxString
& mimeType
) 
 860     wxFileType 
* fileType 
= NULL
; 
 861     // mime types are not case-sensitive 
 862     wxString 
mimetype(mimeType
); 
 863     mimetype
.MakeLower(); 
 865     // first look for an exact match 
 866     int index 
= m_aTypes
.Index(mimetype
); 
 868     if ( index 
!= wxNOT_FOUND 
) 
 870         fileType 
= new wxFileType
; 
 871         fileType
->m_impl
->Init(this, index
); 
 874     // then try to find "text/*" as match for "text/plain" (for example) 
 875     // NB: if mimeType doesn't contain '/' at all, BeforeFirst() will return 
 876     //     the whole string - ok. 
 879     wxString strCategory 
= mimetype
.BeforeFirst(wxT('/')); 
 881     size_t nCount 
= m_aTypes
.GetCount(); 
 882     for ( size_t n 
= 0; n 
< nCount
; n
++ ) 
 884         if ( (m_aTypes
[n
].BeforeFirst(wxT('/')) == strCategory 
) && 
 885                 m_aTypes
[n
].AfterFirst(wxT('/')) == wxT("*") ) 
 892     if ( index 
!= wxNOT_FOUND 
) 
 894        // don't throw away fileType that was already found 
 896             fileType 
= new wxFileType
; 
 897         fileType
->m_impl
->Init(this, index
); 
 903 wxString 
wxMimeTypesManagerImpl::GetCommand(const wxString 
& verb
, size_t nIndex
) const 
 905     wxString command
, testcmd
, sV
, sTmp
; 
 906     sV 
= verb 
+ wxT("="); 
 908     // list of verb = command pairs for this mimetype 
 909     wxMimeTypeCommands 
* sPairs 
= m_aEntries 
[nIndex
]; 
 912     size_t nCount 
= sPairs
->GetCount(); 
 913     for ( i 
= 0; i 
< nCount
; i
++ ) 
 915         sTmp 
= sPairs
->GetVerbCmd (i
); 
 916         if ( sTmp
.Contains(sV
) ) 
 917             command 
= sTmp
.AfterFirst(wxT('=')); 
 923 void wxMimeTypesManagerImpl::AddFallback(const wxFileTypeInfo
& filetype
) 
 928     const wxArrayString
& exts 
= filetype
.GetExtensions(); 
 929     size_t nExts 
= exts
.GetCount(); 
 930     for ( size_t nExt 
= 0; nExt 
< nExts
; nExt
++ ) 
 933             extensions 
+= wxT(' '); 
 935         extensions 
+= exts
[nExt
]; 
 938     AddMimeTypeInfo(filetype
.GetMimeType(), 
 940                     filetype
.GetDescription()); 
 943 void wxMimeTypesManagerImpl::AddMimeTypeInfo(const wxString
& strMimeType
, 
 944                                              const wxString
& strExtensions
, 
 945                                              const wxString
& strDesc
) 
 947     // reading mailcap may find image/* , while 
 948     // reading mime.types finds image/gif and no match is made 
 949     // this means all the get functions don't work  fix this 
 951     wxString sTmp 
= strExtensions
; 
 954     sTmp
.Trim().Trim(false); 
 956     while (!sTmp
.empty()) 
 958         sExts
.Add(sTmp
.AfterLast(wxT(' '))); 
 959         sTmp 
= sTmp
.BeforeLast(wxT(' ')); 
 962     AddToMimeData(strMimeType
, strIcon
, NULL
, sExts
, strDesc
, true); 
 965 size_t wxMimeTypesManagerImpl::EnumAllFileTypes(wxArrayString
& mimetypes
) 
 971     size_t count 
= m_aTypes
.GetCount(); 
 972     for ( size_t n 
= 0; n 
< count
; n
++ ) 
 974         // don't return template types from here (i.e. anything containg '*') 
 975         const wxString 
&type 
= m_aTypes
[n
]; 
 976         if ( type
.Find(wxT('*')) == wxNOT_FOUND 
) 
 982     return mimetypes
.GetCount(); 
 985 // ---------------------------------------------------------------------------- 
 986 // writing to MIME type files 
 987 // ---------------------------------------------------------------------------- 
 989 bool wxMimeTypesManagerImpl::Unassociate(wxFileType 
*ft
) 
 993     wxArrayString sMimeTypes
; 
 994     ft
->GetMimeTypes(sMimeTypes
); 
 997     size_t nCount 
= sMimeTypes
.GetCount(); 
 998     for (i 
= 0; i 
< nCount
; i 
++) 
1000         const wxString 
&sMime 
= sMimeTypes
.Item(i
); 
1001         int nIndex 
= m_aTypes
.Index(sMime
); 
1002         if ( nIndex 
== wxNOT_FOUND
) 
1004             // error if we get here ?? 
1009             m_aTypes
.RemoveAt(nIndex
); 
1010             m_aEntries
.RemoveAt(nIndex
); 
1011             m_aExtensions
.RemoveAt(nIndex
); 
1012             m_aDescriptions
.RemoveAt(nIndex
); 
1013             m_aIcons
.RemoveAt(nIndex
); 
1016     // check data integrity 
1017     wxASSERT( m_aTypes
.GetCount() == m_aEntries
.GetCount() && 
1018             m_aTypes
.GetCount() == m_aExtensions
.GetCount() && 
1019             m_aTypes
.GetCount() == m_aIcons
.GetCount() && 
1020             m_aTypes
.GetCount() == m_aDescriptions
.GetCount() ); 
1026   // wxUSE_MIMETYPE && wxUSE_FILE