]>
git.saurik.com Git - wxWidgets.git/blob - src/msw/mimetype.cpp
   1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        msw/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 ///////////////////////////////////////////////////////////////////////////// 
  13 #pragma implementation "mimetype.h" 
  16 // for compilers that support precompilation, includes "wx.h". 
  17 #include "wx/wxprec.h" 
  25 // Doesn't compile in WIN16 mode 
  29     #include "wx/string.h" 
  32         #include "wx/msgdlg.h" 
  39 #include "wx/dynarray.h" 
  40 #include "wx/confbase.h" 
  43     #include "wx/msw/registry.h" 
  44     #include "wx/msw/private.h" 
  47 #include "wx/msw/mimetype.h" 
  49 // other standard headers 
  52 // in case we're compiling in non-GUI mode 
  53 class WXDLLEXPORT wxIcon
; 
  55 // These classes use Windows registry to retrieve the required information. 
  57 // Keys used (not all of them are documented, so it might actually stop working 
  58 // in future versions of Windows...): 
  59 //  1. "HKCR\MIME\Database\Content Type" contains subkeys for all known MIME 
  60 //     types, each key has a string value "Extension" which gives (dot preceded) 
  61 //     extension for the files of this MIME type. 
  63 //  2. "HKCR\.ext" contains 
  64 //   a) unnamed value containing the "filetype" 
  65 //   b) value "Content Type" containing the MIME type 
  67 // 3. "HKCR\filetype" contains 
  68 //   a) unnamed value containing the description 
  69 //   b) subkey "DefaultIcon" with single unnamed value giving the icon index in 
  71 //   c) shell\open\command and shell\open\print subkeys containing the commands 
  72 //      to open/print the file (the positional parameters are introduced by %1, 
  73 //      %2, ... in these strings, we change them to %s ourselves) 
  75 // although I don't know of any official documentation which mentions this 
  76 // location, uses it, so it isn't likely to change 
  77 static const wxChar 
*MIME_DATABASE_KEY 
= wxT("MIME\\Database\\Content Type\\"); 
  79 void wxFileTypeImpl::Init(const wxString
& strFileType
, const wxString
& ext
) 
  81     // VZ: does it? (FIXME) 
  82     wxCHECK_RET( !ext
.IsEmpty(), _T("needs an extension") ); 
  84     if ( ext
[0u] != wxT('.') ) { 
  89     m_strFileType 
= strFileType
; 
  91         m_strFileType 
= m_ext
.AfterFirst('.') + _T("_auto_file"); 
  95 wxString 
wxFileTypeImpl::GetVerbPath(const wxString
& verb
) const 
  98     path 
<< m_strFileType 
<< _T("\\shell\\") << verb 
<< _T("\\command"); 
 102 size_t wxFileTypeImpl::GetAllCommands(wxArrayString 
*verbs
, 
 103                                       wxArrayString 
*commands
, 
 104                                       const wxFileType::MessageParameters
& params
) const 
 106     wxCHECK_MSG( !m_ext
.IsEmpty(), 0, _T("GetAllCommands() needs an extension") ); 
 108     if ( m_strFileType
.IsEmpty() ) 
 110         // get it from the registry 
 111         wxFileTypeImpl 
*self 
= wxConstCast(this, wxFileTypeImpl
); 
 112         wxRegKey 
rkey(wxRegKey::HKCR
, m_ext
); 
 113         if ( !rkey
.Exists() || !rkey
.QueryValue(_T(""), self
->m_strFileType
) ) 
 115             wxLogDebug(_T("Can't get the filetype for extension '%s'."), 
 122     // enum all subkeys of HKCR\filetype\shell 
 124     wxRegKey 
rkey(wxRegKey::HKCR
, m_strFileType  
+ _T("\\shell")); 
 127     bool ok 
= rkey
.GetFirstKey(verb
, dummy
); 
 130         wxString command 
= wxFileType::ExpandCommand(GetCommand(verb
), params
); 
 132         // we want the open bverb to eb always the first 
 134         if ( verb
.CmpNoCase(_T("open")) == 0 ) 
 137                 verbs
->Insert(verb
, 0); 
 139                 commands
->Insert(command
, 0); 
 141         else // anything else than "open" 
 146                 commands
->Add(command
); 
 151         ok 
= rkey
.GetNextKey(verb
, dummy
); 
 157 // ---------------------------------------------------------------------------- 
 158 // modify the registry database 
 159 // ---------------------------------------------------------------------------- 
 161 bool wxFileTypeImpl::EnsureExtKeyExists() 
 163     wxRegKey 
rkey(wxRegKey::HKCR
, m_ext
); 
 164     if ( !rkey
.Exists() ) 
 166         if ( !rkey
.Create() || !rkey
.SetValue(_T(""), m_strFileType
) ) 
 168             wxLogError(_("Failed to create registry entry for '%s' files."), 
 177 // ---------------------------------------------------------------------------- 
 178 // get the command to use 
 179 // ---------------------------------------------------------------------------- 
 181 wxString 
wxFileTypeImpl::GetCommand(const wxChar 
*verb
) const 
 183     // suppress possible error messages 
 187     if ( wxRegKey(wxRegKey::HKCR
, m_ext 
+ _T("\\shell")).Exists() ) 
 189     if ( wxRegKey(wxRegKey::HKCR
, m_strFileType 
+ _T("\\shell")).Exists() ) 
 190         strKey 
= m_strFileType
; 
 195         return wxEmptyString
; 
 198     strKey 
<< wxT("\\shell\\") << verb
; 
 199     wxRegKey 
key(wxRegKey::HKCR
, strKey 
+ _T("\\command")); 
 202         // it's the default value of the key 
 203         if ( key
.QueryValue(wxT(""), command
) ) { 
 204             // transform it from '%1' to '%s' style format string (now also 
 205             // test for %L - apparently MS started using it as well for the 
 208             // NB: we don't make any attempt to verify that the string is valid, 
 209             //     i.e. doesn't contain %2, or second %1 or .... But we do make 
 210             //     sure that we return a string with _exactly_ one '%s'! 
 211             bool foundFilename 
= FALSE
; 
 212             size_t len 
= command
.Len(); 
 213             for ( size_t n 
= 0; (n 
< len
) && !foundFilename
; n
++ ) { 
 214                 if ( command
[n
] == wxT('%') && 
 216                      (command
[n 
+ 1] == wxT('1') || 
 217                       command
[n 
+ 1] == wxT('L')) ) { 
 218                     // replace it with '%s' 
 219                     command
[n 
+ 1] = wxT('s'); 
 221                     foundFilename 
= TRUE
; 
 226             // look whether we must issue some DDE requests to the application 
 227             // (and not just launch it) 
 228             strKey 
+= _T("\\DDEExec"); 
 229             wxRegKey 
keyDDE(wxRegKey::HKCR
, strKey
); 
 230             if ( keyDDE
.Open() ) { 
 231                 wxString ddeCommand
, ddeServer
, ddeTopic
; 
 232                 keyDDE
.QueryValue(_T(""), ddeCommand
); 
 233                 ddeCommand
.Replace(_T("%1"), _T("%s")); 
 235                 wxRegKey(wxRegKey::HKCR
, strKey 
+ _T("\\Application")). 
 236                     QueryValue(_T(""), ddeServer
); 
 237                 wxRegKey(wxRegKey::HKCR
, strKey 
+ _T("\\Topic")). 
 238                     QueryValue(_T(""), ddeTopic
); 
 240                 if (ddeTopic
.IsEmpty()) 
 241                     ddeTopic 
= wxT("System"); 
 243                 // HACK: we use a special feature of wxExecute which exists 
 244                 //       just because we need it here: it will establish DDE 
 245                 //       conversation with the program it just launched 
 246                 command
.Prepend(_T("WX_DDE#")); 
 247                 command 
<< _T('#') << ddeServer
 
 248                         << _T('#') << ddeTopic
 
 249                         << _T('#') << ddeCommand
; 
 253                 if ( !foundFilename 
) { 
 254                 // we didn't find any '%1' - the application doesn't know which 
 255                 // file to open (note that we only do it if there is no DDEExec 
 258                 // HACK: append the filename at the end, hope that it will do 
 259                 command 
<< wxT(" %s"); 
 263     //else: no such file type or no value, will return empty string 
 269 wxFileTypeImpl::GetOpenCommand(wxString 
*openCmd
, 
 270                                const wxFileType::MessageParameters
& params
) 
 273     wxString cmd 
= GetCommand(wxT("open")); 
 275     *openCmd 
= wxFileType::ExpandCommand(cmd
, params
); 
 277     return !openCmd
->IsEmpty(); 
 281 wxFileTypeImpl::GetPrintCommand(wxString 
*printCmd
, 
 282                                 const wxFileType::MessageParameters
& params
) 
 285     wxString cmd 
= GetCommand(wxT("print")); 
 287     *printCmd 
= wxFileType::ExpandCommand(cmd
, params
); 
 289     return !printCmd
->IsEmpty(); 
 292 // ---------------------------------------------------------------------------- 
 293 // getting other stuff 
 294 // ---------------------------------------------------------------------------- 
 296 // TODO this function is half implemented 
 297 bool wxFileTypeImpl::GetExtensions(wxArrayString
& extensions
) 
 299     if ( m_ext
.IsEmpty() ) { 
 300         // the only way to get the list of extensions from the file type is to 
 301         // scan through all extensions in the registry - too slow... 
 306         extensions
.Add(m_ext
); 
 308         // it's a lie too, we don't return _all_ extensions... 
 313 bool wxFileTypeImpl::GetMimeType(wxString 
*mimeType
) const 
 315     // suppress possible error messages 
 317     wxRegKey 
key(wxRegKey::HKCR
, m_ext
); 
 319     return key
.Open() && key
.QueryValue(wxT("Content Type"), *mimeType
); 
 322 bool wxFileTypeImpl::GetMimeTypes(wxArrayString
& mimeTypes
) const 
 326     if ( !GetMimeType(&s
) ) 
 337 bool wxFileTypeImpl::GetIcon(wxIcon 
*icon
, 
 344     strIconKey 
<< m_strFileType 
<< wxT("\\DefaultIcon"); 
 346     // suppress possible error messages 
 348     wxRegKey 
key(wxRegKey::HKCR
, strIconKey
); 
 352         // it's the default value of the key 
 353         if ( key
.QueryValue(wxT(""), strIcon
) ) { 
 354             // the format is the following: <full path to file>, <icon index> 
 355             // NB: icon index may be negative as well as positive and the full 
 356             //     path may contain the environment variables inside '%' 
 357             wxString strFullPath 
= strIcon
.BeforeLast(wxT(',')), 
 358             strIndex 
= strIcon
.AfterLast(wxT(',')); 
 360             // index may be omitted, in which case BeforeLast(',') is empty and 
 361             // AfterLast(',') is the whole string 
 362             if ( strFullPath
.IsEmpty() ) { 
 363                 strFullPath 
= strIndex
; 
 367             wxString strExpPath 
= wxExpandEnvVars(strFullPath
); 
 368             // here we need C based counting! 
 369             int nIndex 
= wxAtoi(strIndex
); 
 371             HICON hIcon
, hIconLarge
, hIconSmall
; 
 372             ExtractIconEx(strExpPath
, nIndex
, &hIconLarge
, &hIconSmall
, 1); 
 374             hIcon 
= (iconSize 
== wxICON_LARGE
) ? hIconLarge 
: hIconSmall
; 
 377             switch ( (int)hIcon 
) { 
 378                 case 0: // means no icons were found 
 379                 case 1: // means no such file or it wasn't a DLL/EXE/OCX/ICO/... 
 380                     wxLogDebug(wxT("incorrect registry entry '%s': no such icon."), 
 381                                key
.GetName().c_str()); 
 385                     icon
->SetHICON((WXHICON
)hIcon
); 
 386                     wxSize size 
= wxGetHiconSize(hIcon
); 
 391                         *iconFile 
= strFullPath
; 
 397     // no such file type or no value or incorrect icon entry 
 403 bool wxFileTypeImpl::GetDescription(wxString 
*desc
) const 
 405     // suppress possible error messages 
 407     wxRegKey 
key(wxRegKey::HKCR
, m_strFileType
); 
 410         // it's the default value of the key 
 411         if ( key
.QueryValue(wxT(""), *desc
) ) { 
 421 wxMimeTypesManagerImpl::CreateFileType(const wxString
& filetype
, const wxString
& ext
) 
 423     wxFileType 
*fileType 
= new wxFileType
; 
 424     fileType
->m_impl
->Init(filetype
, ext
); 
 428 // extension -> file type 
 430 wxMimeTypesManagerImpl::GetFileTypeFromExtension(const wxString
& ext
) 
 432     // add the leading point if necessary 
 434     if ( ext
[0u] != wxT('.') ) { 
 439     // suppress possible error messages 
 442     bool knownExtension 
= FALSE
; 
 444     wxString strFileType
; 
 445     wxRegKey 
key(wxRegKey::HKCR
, str
); 
 447         // it's the default value of the key 
 448         if ( key
.QueryValue(wxT(""), strFileType
) ) { 
 449             // create the new wxFileType object 
 450             return CreateFileType(strFileType
, ext
); 
 453             // this extension doesn't have a filetype, but it's known to the 
 454             // system and may be has some other useful keys (open command or 
 455             // content-type), so still return a file type object for it 
 456             knownExtension 
= TRUE
; 
 460     if ( !knownExtension 
) 
 466     return CreateFileType(wxEmptyString
, ext
); 
 471 wxMimeTypesManagerImpl::GetOrAllocateFileTypeFromExtension(const wxString& ext) 
 473     wxFileType *fileType = GetFileTypeFromExtension(ext); 
 476         fileType = CreateFileType(wxEmptyString, ext); 
 483 // MIME type -> extension -> file type 
 485 wxMimeTypesManagerImpl::GetFileTypeFromMimeType(const wxString
& mimeType
) 
 487     wxString strKey 
= MIME_DATABASE_KEY
; 
 490     // suppress possible error messages 
 494     wxRegKey 
key(wxRegKey::HKCR
, strKey
); 
 496         if ( key
.QueryValue(wxT("Extension"), ext
) ) { 
 497             return GetFileTypeFromExtension(ext
); 
 505 size_t wxMimeTypesManagerImpl::EnumAllFileTypes(wxArrayString
& mimetypes
) 
 507     // enumerate all keys under MIME_DATABASE_KEY 
 508     wxRegKey 
key(wxRegKey::HKCR
, MIME_DATABASE_KEY
); 
 512     bool cont 
= key
.GetFirstKey(type
, cookie
); 
 517         cont 
= key
.GetNextKey(type
, cookie
); 
 520     return mimetypes
.GetCount(); 
 523 // ---------------------------------------------------------------------------- 
 524 // create a new association 
 525 // ---------------------------------------------------------------------------- 
 527 wxFileType 
*wxMimeTypesManagerImpl::Associate(const wxFileTypeInfo
& ftInfo
) 
 529     wxCHECK_MSG( !ftInfo
.GetExtensions().IsEmpty(), NULL
, 
 530                  _T("Associate() needs extension") ); 
 537     wxString ext 
= ftInfo
.GetExtensions()[iExtCount
]; 
 539     wxCHECK_MSG( !ext
.empty(), NULL
, 
 540                  _T("Associate() needs non empty extension") ); 
 542     if ( ext
[0u] != _T('.') ) 
 543         extWithDot 
= _T('.'); 
 546     // start by setting the HKCR\\.ext entries 
 547     // default is filetype; content type is mimetype 
 548     const wxString
& filetypeOrig 
= ftInfo
.GetShortDesc(); 
 550     wxRegKey 
key(wxRegKey::HKCR
, extWithDot
); 
 553         // create the mapping from the extension to the filetype 
 558             if ( filetypeOrig
.empty() ) 
 560                 // make it up from the extension 
 561                 filetype 
<< extWithDot
.c_str() + 1 << _T("_file"); 
 565                 // just use the provided one 
 566                 filetype 
= filetypeOrig
; 
 569             ok 
= key
.SetValue(_T(""), filetype
); 
 574             // key already exists, maybe we want to change it ?? 
 575             if (!filetypeOrig
.empty()) 
 577                     filetype 
= filetypeOrig
; 
 578                     ok 
= key
.SetValue(_T(""), filetype
); 
 582                     ok 
= key
.QueryValue(_T(""), filetype
); 
 585         // now set a mimetypeif we have it, but ignore it if none 
 586         const wxString
& mimetype 
= ftInfo
.GetMimeType(); 
 587         if ( !mimetype
.empty() ) 
 590             ok 
= key
.SetValue(_T("Content Type"), mimetype
); 
 594                 // create the MIME key 
 595                 wxString strKey 
= MIME_DATABASE_KEY
; 
 597                 wxRegKey 
keyMIME(wxRegKey::HKCR
, strKey
); 
 598                 ok 
= keyMIME
.Create(); 
 602                     // and provide a back link to the extension 
 603                     ok 
= keyMIME
.SetValue(_T("Extension"), extWithDot
); 
 609     // now make other extensions have the same filetype 
 611     for (iExtCount
=1; iExtCount 
< ftInfo
.GetExtensionsCount(); iExtCount
++ ) 
 613             ext 
= ftInfo
.GetExtensions()[iExtCount
]; 
 614             if ( ext
[0u] != _T('.') ) 
 615                extWithDot 
= _T('.'); 
 618             wxRegKey 
key(wxRegKey::HKCR
, extWithDot
); 
 619             if ( !key
.Exists() ) ok 
= key
.Create(); 
 620             ok 
= key
.SetValue(_T(""), filetype
); 
 622         // now set any mimetypes we may have, but ignore it if none 
 623         const wxString
& mimetype 
= ftInfo
.GetMimeType(); 
 624         if ( !mimetype
.empty() ) 
 627             ok 
= key
.SetValue(_T("Content Type"), mimetype
); 
 631                 // create the MIME key 
 632                 wxString strKey 
= MIME_DATABASE_KEY
; 
 634                 wxRegKey 
keyMIME(wxRegKey::HKCR
, strKey
); 
 635                 ok 
= keyMIME
.Create(); 
 639                     // and provide a back link to the extension 
 640                     ok 
= keyMIME
.SetValue(_T("Extension"), extWithDot
); 
 646     } // end of for loop; all extensions now point to HKCR\.ext\Default 
 648     // create the filetype key itself (it will be empty for now, but 
 649     // SetCommand(), SetDefaultIcon() &c will use it later) 
 650     wxRegKey 
keyFT(wxRegKey::HKCR
, filetype
); 
 653     wxFileType 
*ft 
= NULL
; 
 654     ft 
= CreateFileType(filetype
, extWithDot
); 
 658             if (! ftInfo
.GetOpenCommand ().IsEmpty() ) ft
->SetCommand (ftInfo
.GetOpenCommand (), wxT("open"  ) ); 
 659             if (! ftInfo
.GetPrintCommand().IsEmpty() ) ft
->SetCommand (ftInfo
.GetPrintCommand(), wxT("print" ) ); 
 660             // chris: I don't like the ->m_impl-> here FIX this ?? 
 661             if (! ftInfo
.GetDescription ().IsEmpty() ) ft
->m_impl
->SetDescription (ftInfo
.GetDescription ()) ; 
 662             if (! ftInfo
.GetIconFile().IsEmpty() ) ft
->SetDefaultIcon (ftInfo
.GetIconFile(), ftInfo
.GetIconIndex() ); 
 668 bool wxFileTypeImpl::SetCommand(const wxString
& cmd
, 
 669                                 const wxString
& verb
, 
 670                                 bool WXUNUSED(overwriteprompt
)) 
 672     wxCHECK_MSG( !m_ext
.IsEmpty() && !verb
.IsEmpty(), FALSE
, 
 673                  _T("SetCommand() needs an extension and a verb") ); 
 675     if ( !EnsureExtKeyExists() ) 
 678     wxRegKey 
rkey(wxRegKey::HKCR
, GetVerbPath(verb
)); 
 680     if ( rkey
.Exists() && overwriteprompt 
) 
 684         rkey
.QueryValue(wxT(""), old
); 
 688                     _("Do you want to overwrite the command used to %s " 
 689                       "files with extension \"%s\" ?\nCurrent value is \n%s, " 
 690                       "\nNew value is \n%s %1"), // bug here FIX need %1 ?? 
 695                 _("Confirm registry update"), 
 696                 wxYES_NO 
| wxICON_QUESTION
 
 706     // 1. translate '%s' to '%1' instead of always adding it 
 707     // 2. create DDEExec value if needed (undo GetCommand) 
 708     return rkey
.Create() && rkey
.SetValue(_T(""), cmd 
+ _T(" \"%1\"") ); 
 712 bool wxFileTypeImpl::SetMimeType(const wxString& mimeTypeOrig) 
 714     wxCHECK_MSG( !m_ext.IsEmpty(), FALSE, _T("SetMimeType() needs extension") ); 
 716     if ( !EnsureExtKeyExists() ) 
 719     // VZ: is this really useful? (FIXME) 
 723         // make up a default value for it 
 725         wxSplitPath(GetCommand(_T("open")), NULL, &cmd, NULL); 
 726         mimeType << _T("application/x-") << cmd; 
 730         mimeType = mimeTypeOrig; 
 733     wxRegKey rkey(wxRegKey::HKCR, m_ext); 
 734     return rkey.Create() && rkey.SetValue(_T("Content Type"), mimeType); 
 738 bool wxFileTypeImpl::SetDefaultIcon(const wxString
& cmd
, int index
) 
 740     wxCHECK_MSG( !m_ext
.IsEmpty(), FALSE
, _T("SetDefaultIcon() needs extension") ); 
 741     wxCHECK_MSG( !m_strFileType
.IsEmpty(), FALSE
, _T("File key not found") ); 
 742 //    the next line fails on a SMBshare, I think because it is case mangled 
 743 //    wxCHECK_MSG( !wxFileExists(cmd), FALSE, _T("Icon file not found.") ); 
 745     if ( !EnsureExtKeyExists() ) 
 748     wxRegKey 
rkey(wxRegKey::HKCR
, m_strFileType 
+ _T("\\DefaultIcon")); 
 750     return rkey
.Create() && 
 751            rkey
.SetValue(_T(""), 
 752                          wxString::Format(_T("%s,%d"), cmd
.c_str(), index
)); 
 755 bool wxFileTypeImpl::SetDescription (const wxString
& desc
) 
 757     wxCHECK_MSG( !m_strFileType
.IsEmpty(), FALSE
, _T("File key not found") ); 
 758     wxCHECK_MSG( !desc
.IsEmpty(), FALSE
, _T("No file description supplied") ); 
 760     if ( !EnsureExtKeyExists() ) 
 763     wxRegKey 
rkey(wxRegKey::HKCR
, m_strFileType 
); 
 765     return rkey
.Create() && 
 766            rkey
.SetValue(_T(""), desc
); 
 769 // ---------------------------------------------------------------------------- 
 770 // remove file association 
 771 // ---------------------------------------------------------------------------- 
 773 bool wxFileTypeImpl::Unassociate() 
 776     if ( !RemoveOpenCommand() ) 
 778     if ( !RemoveDefaultIcon() ) 
 780     if ( !RemoveMimeType() ) 
 782    if ( !RemoveDescription() ) 
 786     //this might hold other keys, eg some have CSLID keys 
 789         // delete the root key 
 790         wxRegKey key(wxRegKey::HKCR, m_ext); 
 792             result = key.DeleteSelf(); 
 798 bool wxFileTypeImpl::RemoveOpenCommand() 
 800    return RemoveCommand(_T("open")); 
 803 bool wxFileTypeImpl::RemoveCommand(const wxString
& verb
) 
 805     wxCHECK_MSG( !m_ext
.IsEmpty() && !verb
.IsEmpty(), FALSE
, 
 806                  _T("RemoveCommand() needs an extension and a verb") ); 
 808     wxString  sKey 
= m_strFileType
; 
 809     wxRegKey 
rkey(wxRegKey::HKCR
, GetVerbPath(verb
)); 
 811     // if the key already doesn't exist, it's a success 
 812     return !rkey
.Exists() || rkey
.DeleteSelf(); 
 815 bool wxFileTypeImpl::RemoveMimeType() 
 817     wxCHECK_MSG( !m_ext
.IsEmpty(), FALSE
, _T("RemoveMimeType() needs extension") ); 
 819     wxRegKey 
rkey(wxRegKey::HKCR
, m_ext
); 
 820     return !rkey
.Exists() || rkey
.DeleteSelf(); 
 823 bool wxFileTypeImpl::RemoveDefaultIcon() 
 825     wxCHECK_MSG( !m_ext
.IsEmpty(), FALSE
, 
 826                  _T("RemoveDefaultIcon() needs extension") ); 
 828     wxRegKey 
rkey (wxRegKey::HKCR
, m_strFileType  
+ _T("\\DefaultIcon")); 
 829     return !rkey
.Exists() || rkey
.DeleteSelf(); 
 832 bool wxFileTypeImpl::RemoveDescription() 
 834     wxCHECK_MSG( !m_ext
.IsEmpty(), FALSE
, 
 835                  _T("RemoveDescription() needs extension") ); 
 837     wxRegKey 
rkey (wxRegKey::HKCR
, m_strFileType 
); 
 838     return !rkey
.Exists() || rkey
.DeleteSelf(); 
 844 #endif // wxUSE_MIMETYPE