1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/mac/carbon/mimetype.cpp 
   3 // Purpose:     Mac Carbon implementation for wx MIME-related classes 
   8 // Copyright:   (c) 2005 Ryan Norton (<wxprojects@comcast.net>) 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  13 //  TODO: Search Info[-macos](classic).plist dictionary in addition 
  14 //  to Internet Config database. 
  16 //  Maybe try a brainstorm a way to change the wxMimeTypesManager API 
  17 //  to get info from a file instead/addition to current get all stuff 
  18 //  API so that we can use Launch Services to get MIME type info. 
  20 //  Implement GetIcon from one of the FinderInfo functions - or 
  21 //  use Launch Services and search that app's plist for the icon. 
  23 //  Put some special juice in for the print command. 
  26 // for compilers that support precompilation, includes "wx.h". 
  27 #include "wx/wxprec.h" 
  35 #include "wx/osx/mimetype.h" 
  38     #include "wx/dynarray.h" 
  39     #include "wx/string.h" 
  49 #include "wx/confbase.h" 
  51 #include "wx/osx/private.h" 
  53 // other standard headers 
  57     #include <InternetConfig.h> 
  58     #include <CoreServices.h> 
  62 //   START CODE SAMPLE FROM TECHNOTE 1002 (http://developer.apple.com/technotes/tn/tn1002.html) 
  64 // IsRemoteVolume can be used to find out if the 
  65 // volume referred to by vRefNum is a remote volume 
  66 // located somewhere on a network. the volume's attribute 
  67 // flags (copied from the GetVolParmsInfoBuffer structure) 
  68 // are returned in the longword pointed to by vMAttrib. 
  69 OSErr 
IsRemoteVolume(short vRefNum
, Boolean 
*isRemote
, long *vMAttrib
) 
  72     GetVolParmsInfoBuffer volinfo
; 
  75     volPB
.ioParam
.ioVRefNum 
= vRefNum
; 
  76     volPB
.ioParam
.ioNamePtr 
= NULL
; 
  77     volPB
.ioParam
.ioBuffer 
= (Ptr
)&volinfo
; 
  78     volPB
.ioParam
.ioReqCount 
= sizeof(volinfo
); 
  79     err 
= PBHGetVolParmsSync( &volPB 
); 
  82         *isRemote 
= (volinfo
.vMServerAdr 
!= 0); 
  83         *vMAttrib 
= volinfo
.vMAttrib
; 
  89 // BuildVolumeList fills the array pointed to by vols with 
  90 // a list of the currently mounted volumes.  If includeRemote 
  91 // is true, then remote server volumes will be included in 
  92 // the list.  When remote server volumes are included in the 
  93 // list, they will be added to the end of the list.  On entry, 
  94 // *count should contain the size of the array pointed to by 
  95 // vols.  On exit, *count will be set to the number of id numbers 
  96 // placed in the array. If vMAttribMask is non-zero, then 
  97 // only volumes with matching attributes are added to the 
  98 // list of volumes. bits in the vMAttribMask should use the 
  99 // same encoding as bits in the vMAttrib field of 
 100 // the GetVolParmsInfoBuffer structure. 
 101 OSErr 
BuildVolumeList(Boolean includeRemote
, short *vols
, 
 102         long *count
, long vMAttribMask
) 
 104     HParamBlockRec volPB
; 
 107     long nlocal
, nremote
; 
 110     // set up and check parameters 
 111     volPB
.volumeParam
.ioNamePtr 
= NULL
; 
 112     nlocal 
= nremote 
= 0; 
 116     // iterate through volumes 
 117     for (volPB
.volumeParam
.ioVolIndex 
= 1; 
 118         PBHGetVInfoSync(&volPB
) == noErr
; 
 119         volPB
.volumeParam
.ioVolIndex
++) 
 121         // skip remote volumes, if necessary 
 122         err 
= IsRemoteVolume(volPB
.volumeParam
.ioVRefNum
, &isRemote
, &vMAttrib
); 
 126         if ((includeRemote 
|| !isRemote
) && ((vMAttrib 
& vMAttribMask
) == vMAttribMask
)) 
 128             // add local volumes at the front; remote volumes at the end 
 130                 vols
[nlocal 
+ nremote
++] = volPB
.volumeParam
.ioVRefNum
; 
 137                         nremote 
* sizeof(short) ); 
 138                 vols
[nlocal
++] = volPB
.volumeParam
.ioVRefNum
; 
 142             if ((nlocal 
+ nremote
) >= *count
) 
 148     *count 
= (nlocal 
+ nremote
); 
 154 // FindApplication iterates through mounted volumes 
 155 // searching for an application with the given creator 
 156 // type.  If includeRemote is true, then remote volumes 
 157 // will be searched (after local ones) for an application 
 158 // with the creator type. 
 160 // Hacked to output to appName 
 164 OSErr 
FindApplication(OSType appCreator
, Boolean includeRemote
, Str255 appName
, FSSpec
* appSpec
) 
 166     short rRefNums
[kMaxVols
]; 
 171     // get a list of volumes - with desktop files 
 173     err 
= BuildVolumeList(includeRemote
, rRefNums
, &volCount
, (1 << bHasDesktopMgr
) ); 
 177     // iterate through the list 
 178     for (i
=0; i
<volCount
; i
++) 
 180         // has a desktop file? 
 181         desktopPB
.ioCompletion 
= NULL
; 
 182         desktopPB
.ioVRefNum 
= rRefNums
[i
]; 
 183         desktopPB
.ioNamePtr 
= NULL
; 
 184         desktopPB
.ioIndex 
= 0; 
 185         err 
= PBDTGetPath( &desktopPB 
); 
 189         // has the correct app?? 
 190         desktopPB
.ioFileCreator 
= appCreator
; 
 191         desktopPB
.ioNamePtr 
= appName
; 
 192         err 
= PBDTGetAPPLSync( &desktopPB 
); 
 196         // make a file spec referring to it 
 197         err 
= FSMakeFSSpec( rRefNums
[i
], desktopPB
.ioAPPLParID
, appName
, appSpec 
); 
 208 // END CODE SAMPLE FROM TECHNOTE 1002 (http://developer.apple.com/technotes/tn/tn1002.html) 
 210 // yeah, duplicated code 
 211 pascal OSErr 
FSpGetFullPath( const FSSpec 
*spec
, 
 212     short *fullPathLength
, 
 215     OSErr result
, realResult
; 
 223     realResult 
= result 
= noErr
; 
 225   // work around Nav Services "bug" (it returns invalid FSSpecs with empty names) 
 227     if ( spec
->name
[0] == 0 ) 
 229         result 
= FSMakeFSSpecCompat(spec
->vRefNum
, spec
->parID
, spec
->name
, &tempSpec
); 
 235     // Make a copy of the input FSSpec that can be modified 
 236     BlockMoveData( spec
, &tempSpec
, sizeof(FSSpec
) ); 
 238     if ( result 
== noErr 
) 
 240         if ( tempSpec
.parID 
== fsRtParID 
) 
 242             // object is a volume 
 243             // Add a colon to make it a full pathname 
 245             tempSpec
.name
[tempSpec
.name
[0]] = ':'; 
 248             result 
= PtrToHand(&tempSpec
.name
[1], fullPath
, tempSpec
.name
[0]); 
 252             // object isn't a volume 
 254             // Is the object a file or a directory? 
 255             pb
.dirInfo
.ioNamePtr 
= tempSpec
.name
; 
 256             pb
.dirInfo
.ioVRefNum 
= tempSpec
.vRefNum
; 
 257             pb
.dirInfo
.ioDrDirID 
= tempSpec
.parID
; 
 258             pb
.dirInfo
.ioFDirIndex 
= 0; 
 259             result 
= PBGetCatInfoSync( &pb 
); 
 261             // Allow file/directory name at end of path to not exist. 
 263             if ((result 
== noErr
) || (result 
== fnfErr
)) 
 265                 // if the object is a directory, append a colon so full pathname ends with colon 
 266                 if ((result 
== noErr
) && (pb
.hFileInfo
.ioFlAttrib 
& kioFlAttribDirMask
) != 0) 
 269                     tempSpec
.name
[tempSpec
.name
[0]] = ':'; 
 272                 // Put the object name in first 
 273                 result 
= PtrToHand( &tempSpec
.name
[1], fullPath
, tempSpec
.name
[0] ); 
 274                 if ( result 
== noErr 
) 
 276                     // Get the ancestor directory names 
 277                     pb
.dirInfo
.ioNamePtr 
= tempSpec
.name
; 
 278                     pb
.dirInfo
.ioVRefNum 
= tempSpec
.vRefNum
; 
 279                     pb
.dirInfo
.ioDrParID 
= tempSpec
.parID
; 
 281                     // loop until we have an error or find the root directory 
 284                         pb
.dirInfo
.ioFDirIndex 
= -1; 
 285                         pb
.dirInfo
.ioDrDirID 
= pb
.dirInfo
.ioDrParID
; 
 286                         result 
= PBGetCatInfoSync(&pb
); 
 287                         if ( result 
== noErr 
) 
 289                             // Append colon to directory name 
 291                             tempSpec
.name
[tempSpec
.name
[0]] = ':'; 
 293                             // Add directory name to beginning of fullPath 
 294                             (void)Munger(*fullPath
, 0, NULL
, 0, &tempSpec
.name
[1], tempSpec
.name
[0]); 
 298                     while ( (result 
== noErr
) && (pb
.dirInfo
.ioDrDirID 
!= fsRtDirID
) ); 
 304     if ( result 
== noErr 
) 
 307         *fullPathLength 
= GetHandleSize( *fullPath 
); 
 308         result 
= realResult
;  // return realResult in case it was fnfErr 
 312         // Dispose of the handle and return NULL and zero length 
 313         if ( *fullPath 
!= NULL 
) 
 315             DisposeHandle( *fullPath 
); 
 325 // On the mac there are two ways to open a file - one is through apple events and the 
 326 // finder, another is through mime types. 
 328 // So, really there are two ways to implement wxFileType... 
 330 // Mime types are only available on OS 8.1+ through the InternetConfig API 
 332 // Much like the old-style file manager, it has 3 levels of flexibility for its methods - 
 333 // Low - which means you have to iterate yourself through the mime database 
 334 // Medium - which lets you sort of cache the database if you want to use lowlevel functions 
 335 // High - which requires access to the database every time 
 337 // We want to be efficient (i.e. professional :) ) about it, so we use a combo of low 
 338 // and mid-level functions 
 340 // TODO: Should we call ICBegin/ICEnd?  Then where? 
 344 inline void wxLogMimeDebug(const wxChar
* WXUNUSED_UNLESS_DEBUG(szMsg
), OSStatus 
WXUNUSED_UNLESS_DEBUG(status
)) 
 346     wxLogDebug(wxString::Format(wxT("%s  LINE:%i  OSERROR:%i"), szMsg
, __LINE__
, (int)status
)); 
 349 // in case we're compiling in non-GUI mode 
 350 class WXDLLIMPEXP_FWD_CORE wxIcon
; 
 352 bool wxFileTypeImpl::SetCommand(const wxString
& WXUNUSED(cmd
), const wxString
& WXUNUSED(verb
), bool WXUNUSED(overwriteprompt
)) 
 354     wxASSERT_MSG( m_manager 
!= NULL 
, wxT("Bad wxFileType") ); 
 359 bool wxFileTypeImpl::SetDefaultIcon(const wxString
& WXUNUSED(strIcon
), int WXUNUSED(index
)) 
 361     wxASSERT_MSG( m_manager 
!= NULL 
, wxT("Bad wxFileType") ); 
 366 bool wxFileTypeImpl::GetOpenCommand(wxString 
*openCmd
, 
 367                                const wxFileType::MessageParameters
& params
) const 
 369     wxString cmd 
= GetCommand(wxT("open")); 
 371     *openCmd 
= wxFileType::ExpandCommand(cmd
, params
); 
 373     return !openCmd
->empty(); 
 377 wxFileTypeImpl::GetPrintCommand( 
 379     const wxFileType::MessageParameters
& params
) const 
 381     wxString cmd 
= GetCommand(wxT("print")); 
 383     *printCmd 
= wxFileType::ExpandCommand(cmd
, params
); 
 385     return !printCmd
->empty(); 
 389 // Internet Config vs. Launch Services 
 391 // From OS 8 on there was internet config... 
 392 // However, OSX and its finder does not use info 
 393 // from Internet Config at all - the Internet Config 
 394 // database ONLY CONTAINS APPS THAT ARE CLASSIC APPS 
 395 // OR REGISTERED THROUGH INTERNET CONFIG 
 397 // Therefore on OSX in order for the open command to be useful 
 398 // we need to go straight to launch services 
 401 //on darwin, use launch services 
 403 #include <ApplicationServices/ApplicationServices.h> 
 405 wxString 
wxFileTypeImpl::GetCommand(const wxString
& verb
) const 
 407     wxASSERT_MSG( m_manager 
!= NULL 
, wxT("Bad wxFileType") ); 
 409     if (verb 
== wxT("open")) 
 412         ICGetMapEntry( (ICInstance
) m_manager
->m_hIC
, 
 413                        (Handle
) m_manager
->m_hDatabase
, 
 416         wxString sCurrentExtension 
= wxMacMakeStringFromPascal(entry
.extension
); 
 417         sCurrentExtension 
= sCurrentExtension
.Right(sCurrentExtension
.length()-1 ); 
 419         //type, creator, ext, roles, outapp (FSRef), outappurl 
 420         CFURLRef cfurlAppPath
; 
 421         OSStatus status 
= LSGetApplicationForInfo( kLSUnknownType
, 
 423             wxCFStringRef(sCurrentExtension
, wxLocale::GetSystemEncoding()), 
 430             CFStringRef cfsUnixPath 
= CFURLCopyFileSystemPath(cfurlAppPath
, kCFURLPOSIXPathStyle
); 
 431             CFRelease(cfurlAppPath
); 
 434             // Since a filename might have spaces in it, so surround it with quotes 
 441                     + wxCFStringRef(cfsUnixPath
).AsString(wxLocale::GetSystemEncoding()) 
 442                     + wxString(wxT("'")); 
 449             wxLogDebug(wxString::Format(wxT("%i - %s - %i"), 
 451             wxT("LSGetApplicationForInfo failed."), 
 456     return wxEmptyString
; 
 459 bool wxFileTypeImpl::GetDescription(wxString 
*desc
) const 
 461     wxASSERT_MSG( m_manager 
!= NULL 
, wxT("Bad wxFileType") ); 
 464     ICGetMapEntry( (ICInstance
) m_manager
->m_hIC
, 
 465         (Handle
) m_manager
->m_hDatabase
, m_lIndex
, &entry 
); 
 467     *desc 
= wxMacMakeStringFromPascal( entry
.entryName 
); 
 472 bool wxFileTypeImpl::GetExtensions(wxArrayString
& extensions
) 
 474     wxASSERT_MSG( m_manager 
!= NULL 
, wxT("Bad wxFileType") ); 
 477     ICGetMapEntry( (ICInstance
) m_manager
->m_hIC
, 
 478         (Handle
) m_manager
->m_hDatabase
, m_lIndex
, &entry 
); 
 480     //entry has period in it 
 481     wxString sCurrentExtension 
= wxMacMakeStringFromPascal( entry
.extension 
); 
 482     extensions
.Add( sCurrentExtension
.Right( sCurrentExtension
.length() - 1 ) ); 
 487 bool wxFileTypeImpl::GetMimeType(wxString 
*mimeType
) const 
 489     wxASSERT_MSG( m_manager 
!= NULL 
, wxT("Bad wxFileType") ); 
 492     ICGetMapEntry( (ICInstance
) m_manager
->m_hIC
, 
 493         (Handle
) m_manager
->m_hDatabase
, m_lIndex
, &entry 
); 
 495     *mimeType 
= wxMacMakeStringFromPascal(entry
.MIMEType
); 
 500 bool wxFileTypeImpl::GetMimeTypes(wxArrayString
& mimeTypes
) const 
 515 bool wxFileTypeImpl::GetIcon(wxIconLocation 
*WXUNUSED(icon
)) const 
 517     wxASSERT_MSG( m_manager 
!= NULL 
, wxT("Bad wxFileType") ); 
 519     // no such file type or no value or incorrect icon entry 
 523 size_t wxFileTypeImpl::GetAllCommands(wxArrayString 
* verbs
, 
 524     wxArrayString 
* commands
, 
 525     const wxFileType::MessageParameters
& params
) const 
 527     wxASSERT_MSG( m_manager 
!= NULL 
, wxT("Bad wxFileType") ); 
 532     if (GetOpenCommand(&sCommand
, params
)) 
 534         verbs
->Add(wxString(wxT("open"))); 
 535         commands
->Add(sCommand
); 
 542 void wxMimeTypesManagerImpl::Initialize(int WXUNUSED(mailcapStyles
), const wxString
& WXUNUSED(extraDir
)) 
 544     wxASSERT_MSG(m_hIC 
== NULL
, wxT("Already initialized wxMimeTypesManager!")); 
 546     // some apps (non-wx) use the 'plst' resource instead 
 548     CFBundleRef cfbMain 
= CFBundleGetMainBundle(); 
 549     wxCFDictionary 
cfdInfo( CFBundleGetInfoDictionary(cfbMain
), wxCF_RETAIN 
); 
 551     cfdInfo
.PrintOut(sLog
); 
 555     // start Internet Config - log if there's an error 
 556     // the second param is the signature of the application, also known 
 557     // as resource ID 0.  However, as per some recent discussions, we may not 
 558     // have a signature for this app, so a generic 'APPL' which is the executable 
 559     // type will work for now. 
 560     OSStatus status 
= ICStart( (ICInstance
*)&m_hIC
, 'APPL' ); 
 564         wxLogDebug(wxT("Could not initialize wxMimeTypesManager!")); 
 572     m_hDatabase 
= (void**) NewHandle(0); 
 573     status 
= ICFindPrefHandle( (ICInstance
) m_hIC
, kICMapping
, &attr
, (Handle
) m_hDatabase 
); 
 575     //the database file can be corrupt (on OSX its 
 576     //~/Library/Preferences/com.apple.internetconfig.plist) 
 581         wxLogDebug(wxT("Corrupt MIME database!")); 
 585     //obtain the number of entries in the map 
 586     status 
= ICCountMapEntries( (ICInstance
) m_hIC
, (Handle
) m_hDatabase
, &m_lCount 
); 
 587     wxASSERT( status 
== noErr 
); 
 594     for (long i 
= 1; i 
<= m_lCount
; ++i
) 
 596         OSStatus status 
= ICGetIndMapEntry( (ICInstance
) m_hIC
, (Handle
) m_hDatabase
, i
, &pos
, &entry 
); 
 600             wxString sCreator 
= wxMacMakeStringFromPascal(entry
.creatorAppName
); 
 601             wxString sCurrentExtension 
= wxMacMakeStringFromPascal(entry
.extension
); 
 602             wxString sMIMEType 
= wxMacMakeStringFromPascal(entry
.MIMEType
); 
 605             impl
.Init(this, pos
); 
 607             if (sMIMEType 
== wxT("text/html") && sCurrentExtension 
== wxT(".html")) 
 611                 impl
.GetOpenCommand( &cmd
, wxFileType::MessageParameters (wxT("http://www.google.com"))); 
 612                 wxPrintf(wxT("APP: [%s]\n"), cmd
.c_str()); 
 619 void wxMimeTypesManagerImpl::ClearData() 
 623         DisposeHandle( (Handle
)m_hDatabase 
); 
 625         // this can return an error, but we don't really care that much about it 
 626         ICStop( (ICInstance
)m_hIC 
); 
 632 //  Q) Iterating through the map - why does it use if (err == noErr) instead of just asserting? 
 633 //  A) Some intermediate indexes are bad while subsequent ones may be good.  Its wierd, I know. 
 636 // extension -> file type 
 637 wxFileType
* wxMimeTypesManagerImpl::GetFileTypeFromExtension(const wxString
& e
) 
 639     wxASSERT_MSG( m_hIC 
!= NULL
, wxT("wxMimeTypesManager not Initialized!") ); 
 641     //low level functions - iterate through the database 
 645     for (long i 
= 1; i 
<= m_lCount
; ++i
) 
 647         OSStatus status 
= ICGetIndMapEntry( (ICInstance
) m_hIC
, (Handle
) m_hDatabase
, i
, &pos
, &entry 
); 
 651             wxString sCurrentExtension 
= wxMacMakeStringFromPascal(entry
.extension
); 
 652             if ( sCurrentExtension
.Right(sCurrentExtension
.length() - 1) == e 
) // entry has period in it 
 654                 wxFileType
* pFileType 
= new wxFileType(); 
 655                 pFileType
->m_impl
->Init((wxMimeTypesManagerImpl
*)this, pos
); 
 665 // MIME type -> extension -> file type 
 666 wxFileType
* wxMimeTypesManagerImpl::GetFileTypeFromMimeType(const wxString
& mimeType
) 
 668     wxASSERT_MSG( m_hIC 
!= NULL
, wxT("wxMimeTypesManager not Initialized!") ); 
 673     // low level functions - iterate through the database 
 674     for (long i 
= 1; i 
<= m_lCount
; ++i
) 
 676         OSStatus status 
= ICGetIndMapEntry( (ICInstance
) m_hIC
, (Handle
) m_hDatabase
, i
, &pos
, &entry 
); 
 677         wxASSERT_MSG( status 
== noErr
, wxString::Format(wxT("Error: %d"), (int)status
) ); 
 681             if ( wxMacMakeStringFromPascal(entry
.MIMEType
) == mimeType
) 
 683                 wxFileType
* pFileType 
= new wxFileType(); 
 684                 pFileType
->m_impl
->Init((wxMimeTypesManagerImpl
*)this, pos
); 
 694 size_t wxMimeTypesManagerImpl::EnumAllFileTypes(wxArrayString
& mimetypes
) 
 696     wxASSERT_MSG( m_hIC 
!= NULL
, wxT("wxMimeTypesManager not Initialized!") ); 
 699     long pos
, lStartCount
; 
 701     // low level functions - iterate through the database 
 702     lStartCount 
= (long) mimetypes
.GetCount(); 
 703     for (long i 
= 1; i 
<= m_lCount
; ++i
) 
 705         OSStatus status 
= ICGetIndMapEntry( (ICInstance
) m_hIC
, (Handle
) m_hDatabase
, i
, &pos
, &entry 
); 
 706         if ( status 
== noErr 
) 
 707             mimetypes
.Add( wxMacMakeStringFromPascal(entry
.MIMEType
) ); 
 710     return mimetypes
.GetCount() - lStartCount
; 
 713 pascal  OSStatus  
MoreProcGetProcessTypeSignature( 
 714     const ProcessSerialNumberPtr pPSN
, 
 715     OSType 
*pProcessType
, 
 718     OSStatus anErr 
= noErr
; 
 719     ProcessInfoRec infoRec
; 
 720     ProcessSerialNumber localPSN
; 
 722     infoRec
.processInfoLength 
= sizeof(ProcessInfoRec
); 
 723     infoRec
.processName 
= NULL
; 
 725     infoRec
.processAppSpec 
= NULL
; 
 730         localPSN
.highLongOfPSN 
= 0; 
 731         localPSN
.lowLongOfPSN  
= kCurrentProcess
; 
 738     anErr 
= GetProcessInformation(&localPSN
, &infoRec
); 
 741         *pProcessType 
= infoRec
.processType
; 
 742         *pCreator 
= infoRec
.processSignature
; 
 750 //  TODO: clean this up, its messy 
 755 #include "wx/osx/core/cfstring.h" 
 757 #define wxCF_RELEASE true 
 758 #define wxCF_RETAIN  false 
 760 // ---------------------------------------------------------------------------- 
 762 // ---------------------------------------------------------------------------- 
 767     wxCFDictionary(CFTypeRef ref
) 
 769         m_cfmdRef 
= (CFMutableDictionaryRef
) ref
; 
 772     wxCFDictionary(CFIndex cfiSize 
= 0) 
 774         CFDictionaryKeyCallBacks kcbs
; 
 775         CFDictionaryValueCallBacks vcbs
; 
 776         BuildKeyCallbacks(&kcbs
); 
 777         BuildValueCallbacks(&vcbs
); 
 779         m_cfmdRef 
= CFDictionaryCreateMutable( 
 780             kCFAllocatorDefault
, cfiSize
, &kcbs
, &vcbs 
); 
 789             CFRelease(m_cfmdRef
); 
 792     static const void* RetainProc(CFAllocatorRef
, const void* v
) 
 793     { return (const void*) CFRetain(v
); } 
 795     static void ReleaseProc(CFAllocatorRef
, const void* v
) 
 798     void MakeMutable(CFIndex cfiSize 
= 0) 
 800         CFDictionaryRef oldref 
= (CFDictionaryRef
) m_cfmdRef
; 
 802         m_cfmdRef 
= CFDictionaryCreateMutableCopy( 
 803             kCFAllocatorDefault
, cfiSize
, oldref 
); 
 808     void BuildKeyCallbacks(CFDictionaryKeyCallBacks
* pCbs
) 
 811         pCbs
->retain 
= RetainProc
; 
 812         pCbs
->release 
= ReleaseProc
; 
 813         pCbs
->copyDescription 
= NULL
; 
 818     void BuildValueCallbacks(CFDictionaryValueCallBacks
* pCbs
) 
 821         pCbs
->retain 
= RetainProc
; 
 822         pCbs
->release 
= ReleaseProc
; 
 823         pCbs
->copyDescription 
= NULL
; 
 827     operator CFTypeRef () const 
 828     { return (CFTypeRef
)m_cfmdRef
; } 
 830     CFDictionaryRef 
GetCFDictionary() const 
 831     { return (CFDictionaryRef
)m_cfmdRef
; } 
 833     CFMutableDictionaryRef 
GetCFMutableDictionary() 
 834     { return (CFMutableDictionaryRef
) m_cfmdRef
; } 
 836     CFTypeRef 
operator [] (CFTypeRef cftEntry
) const 
 839         return (CFTypeRef
) CFDictionaryGetValue((CFDictionaryRef
)m_cfmdRef
, cftEntry
); 
 842     CFIndex 
GetCount() const 
 845         return CFDictionaryGetCount((CFDictionaryRef
)m_cfmdRef
); 
 848     void Add(CFTypeRef cftKey
, CFTypeRef cftValue
) 
 851         wxASSERT(Exists(cftKey
) == false); 
 852         CFDictionaryAddValue(m_cfmdRef
, cftKey
, cftValue
); 
 855     void Remove(CFTypeRef cftKey
) 
 858         wxASSERT(Exists(cftKey
)); 
 859         CFDictionaryRemoveValue(m_cfmdRef
, cftKey
); 
 862     void Set(CFTypeRef cftKey
, CFTypeRef cftValue
) 
 865         wxASSERT(Exists(cftKey
)); 
 866         CFDictionarySetValue(m_cfmdRef
, cftKey
, cftValue
); 
 869     bool Exists(CFTypeRef cftKey
) const 
 872         return CFDictionaryContainsKey((CFDictionaryRef
)m_cfmdRef
, cftKey
); 
 876     { return m_cfmdRef 
!= NULL
; } 
 879     { return IsOk() && CFGetTypeID((CFTypeRef
)m_cfmdRef
) == CFDictionaryGetTypeID(); } 
 881     void PrintOut(wxString
& sMessage
) 
 883         PrintOutDictionary(sMessage
, m_cfmdRef
); 
 886     static void PrintOutDictionary(wxString
& sMessage
, CFDictionaryRef cfdRef
) 
 888         CFIndex cfiCount 
= CFDictionaryGetCount(cfdRef
); 
 889         CFTypeRef
* pKeys 
= new CFTypeRef
[cfiCount
]; 
 890         CFTypeRef
* pValues 
= new CFTypeRef
[cfiCount
]; 
 892         CFDictionaryGetKeysAndValues(cfdRef
, pKeys
, pValues
); 
 894         for (CFIndex i 
= 0; i 
< cfiCount
; ++i
) 
 896             wxString sKey 
= wxCFStringRef(CFCopyTypeIDDescription(CFGetTypeID(pKeys
[i
]))).AsString(); 
 897             wxString sValue 
= wxCFStringRef(CFCopyTypeIDDescription(CFGetTypeID(pValues
[i
]))).AsString(); 
 900                 wxString::Format(wxT("[{#%d} Key : %s]"), (int) i
, 
 903             PrintOutType(sMessage
, sKey
, pKeys
[i
]); 
 906                 wxString::Format(wxT("\n\t[Value : %s]"), 
 909             PrintOutType(sMessage
, sValue
, pValues
[i
]); 
 911             sMessage 
<< wxT("\n"); 
 918     static void PrintOutArray(wxString
& sMessage
, CFArrayRef cfaRef
) 
 920         for (CFIndex i 
= 0; i 
< CFArrayGetCount(cfaRef
); ++i
) 
 922             wxString sValue 
= wxCFStringRef(CFCopyTypeIDDescription(CFGetTypeID( 
 923                 CFArrayGetValueAtIndex(cfaRef
, i
) 
 927                 wxString::Format(wxT("\t\t[{#%d} ArrayValue : %s]\n"), (int) i
, 
 930             PrintOutType(sMessage
, sValue
, CFArrayGetValueAtIndex(cfaRef
, i
)); 
 934     static void PrintOutType(wxString
& sMessage
, const wxString
& sValue
, CFTypeRef cfRef
) 
 936             sMessage 
<< wxT(" {"); 
 938             if (sValue 
== wxT("CFString")) 
 940                  sMessage 
<< wxCFStringRef(wxCFRetain((CFStringRef
)cfRef
)).AsString(); 
 942             else if (sValue 
== wxT("CFNumber")) 
 945                 CFNumberGetValue((CFNumberRef
)cfRef
, kCFNumberIntType
, &nOut
); 
 948             else if (sValue 
== wxT("CFDictionary")) 
 950                 PrintOutDictionary(sMessage
, (CFDictionaryRef
)cfRef
); 
 952             else if (sValue 
== wxT("CFArray")) 
 954                 PrintOutArray(sMessage
, (CFArrayRef
)cfRef
); 
 956             else if (sValue 
== wxT("CFBoolean")) 
 958                 sMessage 
<< (cfRef 
== kCFBooleanTrue 
? wxT("true") : wxT("false")); 
 960             else if (sValue 
== wxT("CFURL")) 
 962                 sMessage 
<< wxCFStringRef(CFURLCopyPath((CFURLRef
) cfRef
)).AsString(); 
 966                 sMessage 
<< wxT("*****UNKNOWN TYPE******"); 
 969             sMessage 
<< wxT("} "); 
 976     CFTypeRef 
WriteAsXML() 
 978         return CFPropertyListCreateXMLData(kCFAllocatorDefault
, m_cfmdRef
); 
 981     bool ReadAsXML(CFTypeRef cfData
, wxString
* pErrorMsg 
= NULL
) 
 984         CFStringRef cfsError
=NULL
; 
 985         m_cfmdRef 
= (CFMutableDictionaryRef
) CFPropertyListCreateFromXMLData( 
 988                     kCFPropertyListMutableContainersAndLeaves
, 
 993                 *pErrorMsg 
= wxCFStringRef(cfsError
).AsString(); 
 998         return m_cfmdRef 
!= NULL
; 
1002     CFMutableDictionaryRef m_cfmdRef
; 
1005 // ---------------------------------------------------------------------------- 
1007 // ---------------------------------------------------------------------------- 
1012     wxCFArray(CFTypeRef ref
) 
1014         m_cfmaRef 
= (CFMutableArrayRef
)ref
; 
1017     wxCFArray(CFIndex cfiSize 
= 0) : m_cfmaRef(NULL
) 
1018     { Create(cfiSize
); } 
1023     void MakeMutable(CFIndex cfiSize 
= 0) 
1025         wxASSERT(IsValid()); 
1027         CFMutableArrayRef oldref 
= m_cfmaRef
; 
1028         m_cfmaRef 
= CFArrayCreateMutableCopy( 
1029                 kCFAllocatorDefault
, 
1031                 (CFArrayRef
)oldref
); 
1035     void BuildCallbacks(CFArrayCallBacks
* pCbs
) 
1038         pCbs
->retain 
= RetainProc
; 
1039         pCbs
->release 
= ReleaseProc
; 
1040         pCbs
->copyDescription 
= NULL
; 
1044     void Create(CFIndex cfiSize 
= 0) 
1047         CFArrayCallBacks cb
; 
1048         BuildCallbacks(&cb
); 
1050         m_cfmaRef 
= CFArrayCreateMutable(kCFAllocatorDefault
, cfiSize
, &cb
); 
1054     { if (m_cfmaRef
) CFRelease(m_cfmaRef
); } 
1056     static const void* RetainProc(CFAllocatorRef
, const void* v
) 
1057     { return (const void*) CFRetain(v
); } 
1059     static void ReleaseProc(CFAllocatorRef
, const void* v
) 
1062     operator CFTypeRef () const 
1063     { return (CFTypeRef
)m_cfmaRef
; } 
1065     CFArrayRef 
GetCFArray() const 
1066     { return (CFArrayRef
)m_cfmaRef
; } 
1068     CFMutableArrayRef 
GetCFMutableArray() 
1069     { return (CFMutableArrayRef
) m_cfmaRef
; } 
1071     CFTypeRef 
operator [] (CFIndex cfiIndex
) const 
1073         wxASSERT(IsValid()); 
1074         return (CFTypeRef
) CFArrayGetValueAtIndex((CFArrayRef
)m_cfmaRef
, cfiIndex
); 
1079         wxASSERT(IsValid()); 
1080         return CFArrayGetCount((CFArrayRef
)m_cfmaRef
); 
1083     void Add(CFTypeRef cftValue
) 
1085         wxASSERT(IsValid()); 
1086         CFArrayAppendValue(m_cfmaRef
, cftValue
); 
1089     void Remove(CFIndex cfiIndex
) 
1091         wxASSERT(IsValid()); 
1092         wxASSERT(cfiIndex 
< GetCount()); 
1093         CFArrayRemoveValueAtIndex(m_cfmaRef
, cfiIndex
); 
1096     void Set(CFIndex cfiIndex
, CFTypeRef cftValue
) 
1098         wxASSERT(IsValid()); 
1099         wxASSERT(cfiIndex 
< GetCount()); 
1100         CFArraySetValueAtIndex(m_cfmaRef
, cfiIndex
, cftValue
); 
1104     { return m_cfmaRef 
!= NULL
; } 
1106     bool IsValid() const 
1108         return IsOk() && CFGetTypeID((CFTypeRef
)m_cfmaRef
) == CFArrayGetTypeID(); 
1112     void MakeValidXML(); 
1116     CFMutableArrayRef m_cfmaRef
; 
1119 // ---------------------------------------------------------------------------- 
1121 // ---------------------------------------------------------------------------- 
1126     wxCFNumber(int nValue
) 
1128         m_cfnRef 
= CFNumberCreate(kCFAllocatorDefault
, kCFNumberIntType
, &nValue
); 
1131     wxCFNumber(CFTypeRef ref
) : m_cfnRef((CFNumberRef
)ref
) 
1135     virtual ~wxCFNumber() 
1138             CFRelease(m_cfnRef
); 
1141     operator CFTypeRef() const 
1142     { return (CFTypeRef
) m_cfnRef
; } 
1147         CFNumberGetValue( m_cfnRef
, kCFNumberIntType
, &nOut 
); 
1153     { return m_cfnRef 
!= NULL
; } 
1156     CFNumberRef m_cfnRef
; 
1159 // ---------------------------------------------------------------------------- 
1161 // ---------------------------------------------------------------------------- 
1166     wxCFURL(CFTypeRef ref 
= NULL
) : m_cfurlRef((CFURLRef
)ref
) 
1170     wxCFURL(const wxCFStringRef
& URLString
, CFTypeRef BaseURL 
= NULL
) 
1172         Create(URLString
, BaseURL
); 
1175     void Create(const wxCFStringRef
& URLString
, CFTypeRef BaseURL 
= NULL
) 
1177         m_cfurlRef 
= CFURLCreateWithString( 
1178             kCFAllocatorDefault
, 
1179             (CFStringRef
)(CFTypeRef
)URLString
, 
1180             (CFURLRef
) BaseURL
); 
1186             CFRelease(m_cfurlRef
); 
1189     wxString 
BuildWXString() 
1191         return wxCFStringRef(CFURLCopyPath(m_cfurlRef
)).AsString(); 
1194     operator CFTypeRef() const 
1195     { return (CFTypeRef
)m_cfurlRef
; } 
1198     { return m_cfurlRef 
!= NULL
; } 
1201     CFURLRef m_cfurlRef
; 
1204 // ---------------------------------------------------------------------------- 
1206 // ---------------------------------------------------------------------------- 
1208 #define wxCFDATA_RELEASEBUFFER  1 
1213     wxCFData(CFTypeRef ref
) : m_cfdaRef((CFDataRef
)ref
) 
1217     wxCFData(const UInt8
* pBytes
, CFIndex len
, bool bKeep 
= wxCFDATA_RELEASEBUFFER
) 
1219         if (bKeep 
== wxCFDATA_RELEASEBUFFER
) 
1221             m_cfdaRef 
= CFDataCreateWithBytesNoCopy
 
1222                             (kCFAllocatorDefault
, pBytes
, len
, kCFAllocatorDefault
); 
1226             m_cfdaRef 
= CFDataCreate(kCFAllocatorDefault
, pBytes
, len
); 
1233             CFRelease(m_cfdaRef
); 
1236     const UInt8
* GetValue() 
1237     { return CFDataGetBytePtr(m_cfdaRef
); } 
1240     { return CFDataGetLength(m_cfdaRef
); } 
1242     operator CFTypeRef() const 
1243     { return (CFTypeRef
)m_cfdaRef
; } 
1246     { return m_cfdaRef 
!= NULL
; } 
1249     CFDataRef m_cfdaRef
; 
1252 void wxCFDictionary::MakeValidXML() 
1254         CFIndex cfiCount 
= GetCount(); 
1255         CFTypeRef
* pKeys 
= new CFTypeRef
[cfiCount
]; 
1256         CFTypeRef
* pValues 
= new CFTypeRef
[cfiCount
]; 
1258         CFDictionaryGetKeysAndValues(m_cfmdRef
, pKeys
, pValues
); 
1260         // for plist xml format, all dictionary keys must be cfstrings and 
1261         // no values in the dictionary or subkeys/values can be NULL; 
1262         // additionally, CFURLs are not allowed 
1263         for (CFIndex i 
= 0; i 
< cfiCount
; ++i
) 
1265             // must be an array, dictionary, string, bool, or int and cannot be null 
1266             // and dictionaries can only contain cfstring keys 
1267             CFTypeRef cfRef 
= pValues
[i
]; 
1269                 CFGetTypeID(pKeys
[i
]) != CFStringGetTypeID() || 
1277                 pKeys 
= new CFTypeRef
[cfiCount
]; 
1278                 pValues 
= new CFTypeRef
[cfiCount
]; 
1279                 CFDictionaryGetKeysAndValues(m_cfmdRef
, pKeys
, pValues
); 
1281             else if (CFGetTypeID(cfRef
) == CFArrayGetTypeID()) 
1284                 wxCFArray 
cfaCurrent(cfRef
); 
1285                 cfaCurrent
.MakeMutable(); 
1286                 cfaCurrent
.MakeValidXML(); 
1287                 Set(pKeys
[i
], cfaCurrent
); 
1289             else if (CFGetTypeID(cfRef
) == CFDictionaryGetTypeID()) 
1292                 wxCFDictionary 
cfdCurrent(cfRef
); 
1293                 cfdCurrent
.MakeMutable(); 
1294                 cfdCurrent
.MakeValidXML(); 
1295                 Set(pKeys
[i
], cfdCurrent
); 
1297             else if ( CFGetTypeID(cfRef
) != CFStringGetTypeID() && 
1298                       CFGetTypeID(cfRef
) != CFNumberGetTypeID() && 
1299                       CFGetTypeID(cfRef
) != CFBooleanGetTypeID() ) 
1306                 pKeys 
= new CFTypeRef
[cfiCount
]; 
1307                 pValues 
= new CFTypeRef
[cfiCount
]; 
1308                 CFDictionaryGetKeysAndValues(m_cfmdRef
, pKeys
, pValues
); 
1316 void wxCFArray::MakeValidXML() 
1318         for (CFIndex i 
= 0; i 
< GetCount(); ++i
) 
1320             //must be an array, dictionary, string, bool, or int and cannot be null 
1321             //and dictionaries can only contain cfstring keys 
1322             CFTypeRef cfRef 
= (*this)[i
]; 
1328             else if (CFGetTypeID(cfRef
) == CFArrayGetTypeID()) 
1331                 wxCFArray 
cfaCurrent(cfRef
); 
1332                 cfaCurrent
.MakeMutable(); 
1333                 cfaCurrent
.MakeValidXML(); 
1336             else if (CFGetTypeID(cfRef
) == CFDictionaryGetTypeID()) 
1339                 wxCFDictionary 
cfdCurrent(cfRef
); 
1340                 cfdCurrent
.MakeMutable(); 
1341                 cfdCurrent
.MakeValidXML(); 
1344             else if ( CFGetTypeID(cfRef
) != CFStringGetTypeID() && 
1345                       CFGetTypeID(cfRef
) != CFNumberGetTypeID() && 
1346                       CFGetTypeID(cfRef
) != CFBooleanGetTypeID() ) 
1362 wxFileType
* wxMimeTypesManagerImpl::Associate(const wxFileTypeInfo
& ftInfo
) 
1364     bool bInfoSuccess 
= false; 
1366     const wxArrayString
& asExtensions 
= ftInfo
.GetExtensions(); 
1367     size_t dwFoundIndex 
= 0; 
1368     if (!asExtensions
.GetCount()) 
1370         wxLogDebug(wxT("Must have extension to associate with")); 
1373     // Find and write to Info.plist in main bundle (note that some other 
1374     // apps have theirs named differently, i.e. IE's is named Info-macos.plist 
1375     // some apps (non-wx) use the 'plst' resource instead 
1376     CFBundleRef cfbMain 
= CFBundleGetMainBundle(); 
1379         UInt32 dwBundleType
, dwBundleCreator
; 
1380         CFBundleGetPackageInfo(cfbMain
, &dwBundleType
, &dwBundleCreator
); 
1382         // if launching terminal non-app, version will be 'BNDL' (generic bundle, maybe in other cases too), 
1383         // which will give us the incorrect info.plist path 
1384         // otherwise it will be 'APPL', or in the case of a framework, 'FMWK' 
1385         if (dwBundleType 
== 'APPL') 
1387             wxCFURL 
cfurlBundleLoc((CFTypeRef
)CFBundleCopyBundleURL(cfbMain
)); 
1388 //             wxCFURL cfurlBundleLoc((CFTypeRef)CFBundleCopyExecutableURL(cfbMain)); 
1390 //            sInfoPath << wxT("file://"); 
1391             sInfoPath 
<< cfurlBundleLoc
.BuildWXString(); 
1392             sInfoPath 
<< wxT("Contents/Info.plist"); 
1394 //        wxCFDictionary cfdInfo( CFBundleGetInfoDictionary(cfbMain), wxCF_RETAIN ); 
1395             wxCFDictionary cfdInfo
; 
1396             bool bInfoOpenSuccess 
= false; 
1398             if (indictfile
.Open(sInfoPath
, wxFile::read
)) 
1400                 CFIndex cfiBufLen 
= (CFIndex
) indictfile
.Length(); 
1401                 const UInt8
* pBuffer 
= new UInt8
[cfiBufLen
]; 
1402                 indictfile
.Read((void*)pBuffer
, cfiBufLen
); 
1403                 wxCFData 
cfdaInDict(pBuffer
, cfiBufLen
); 
1405                 bInfoOpenSuccess 
= cfdInfo
.ReadAsXML(cfdaInDict
, &sError
); 
1406                 if (!bInfoOpenSuccess
) 
1411             if (bInfoOpenSuccess
) 
1413                 cfdInfo
.MakeMutable( cfdInfo
.GetCount() + 1 ); 
1415                 wxCFArray 
cfaDocTypes( wxCFRetain( cfdInfo
[ wxCFStringRef(wxT("CFBundleDocumentTypes")) ] ) ); 
1417                 bool bAddDocTypesArrayToDictionary 
= !cfaDocTypes
.IsOk(); 
1418                 if (bAddDocTypesArrayToDictionary
) 
1419                     cfaDocTypes
.Create(); 
1421                     cfaDocTypes
.MakeMutable( cfaDocTypes
.GetCount() + 1 ); 
1423                 bool bEntryFound 
= false; 
1425                 // search for duplicates 
1427                 for (i 
= 0; i 
< cfaDocTypes
.GetCount(); ++i
) 
1429                     wxCFDictionary 
cfdDocTypeEntry( wxCFRetain( cfaDocTypes
[i
] ) ); 
1431                     // A lot of apps don't support MIME types for some reason 
1432                     // so we go by extensions only 
1433                     wxCFArray 
cfaExtensions( wxCFRetain( cfdDocTypeEntry
[ wxCFStringRef(wxT("CFBundleTypeExtensions")) ] ) ); 
1435                     if (!cfaExtensions
.IsOk()) 
1438                     for (CFIndex iExt 
= 0; iExt 
< cfaExtensions
.GetCount(); ++iExt
) 
1440                         for (size_t iWXExt 
= 0; iWXExt 
< asExtensions
.GetCount(); ++iWXExt
) 
1442                             if (asExtensions
[iWXExt
] == 
1443                                     wxCFStringRef( wxCFRetain( (CFStringRef
) cfaExtensions
[iExt
] ) ).AsString()) 
1446                                 dwFoundIndex 
= iWXExt
; 
1450                         } //end of wxstring array 
1454                     } //end for cf array 
1458                 } //end for doctypes 
1460                 wxCFDictionary cfdNewEntry
; 
1462                 if (!ftInfo
.GetDescription().empty()) 
1464                     cfdNewEntry
.Add( wxCFStringRef(wxT("CFBundleTypeName")), 
1465                                 wxCFStringRef(ftInfo
.GetDescription()) ); 
1468                 if (!ftInfo
.GetIconFile().empty()) 
1470                     cfdNewEntry
.Add( wxCFStringRef(wxT("CFBundleTypeIconFile")), 
1471                                     wxCFStringRef(ftInfo
.GetIconFile()) ); 
1474                 wxCFArray cfaOSTypes
; 
1475                 wxCFArray cfaExtensions
; 
1476                 wxCFArray cfaMimeTypes
; 
1478                 //OSTypes is a cfarray of four-char-codes - '****' for unrestricted 
1479                 cfaOSTypes
.Add( wxCFStringRef(wxT("****")) ); 
1480                 cfdNewEntry
.Add( wxCFStringRef(wxT("CFBundleTypeOSTypes")), cfaOSTypes 
); 
1482                 //'*' for unrestricted 
1483                 if (ftInfo
.GetExtensionsCount() != 0) 
1485                     for (size_t iExtension 
= 0; iExtension 
< ftInfo
.GetExtensionsCount(); ++iExtension
) 
1487                         cfaExtensions
.Add( wxCFStringRef( asExtensions
[iExtension
] ) ); 
1490                     cfdNewEntry
.Add( wxCFStringRef(wxT("CFBundleTypeExtensions")), cfaExtensions 
); 
1493                 if (!ftInfo
.GetMimeType().empty()) 
1495                     cfaMimeTypes
.Add( wxCFStringRef(ftInfo
.GetMimeType()) ); 
1496                     cfdNewEntry
.Add( wxCFStringRef(wxT("CFBundleTypeMIMETypes")), cfaMimeTypes 
); 
1499                 // Editor - can perform all actions 
1500                 // Viewer - all actions except manipulation/saving 
1501                 // None - can perform no actions 
1502                 cfdNewEntry
.Add( wxCFStringRef(wxT("CFBundleTypeRole")), wxCFStringRef(wxT("Editor")) ); 
1504                 // Is application bundled? 
1505                 cfdNewEntry
.Add( wxCFStringRef(wxT("LSTypeIsPackage")), kCFBooleanTrue 
); 
1508                     cfaDocTypes
.Set(i
, cfdNewEntry
); 
1510                     cfaDocTypes
.Add(cfdNewEntry
); 
1512                 // set the doc types array in the muted dictionary 
1513                 if (bAddDocTypesArrayToDictionary
) 
1514                     cfdInfo
.Add(wxCFStringRef(wxT("CFBundleDocumentTypes")), cfaDocTypes
); 
1516                     cfdInfo
.Set(wxCFStringRef(wxT("CFBundleDocumentTypes")), cfaDocTypes
); 
1518                 cfdInfo
.MakeValidXML(); 
1521                 if (outdictfile
.Open(sInfoPath
, wxFile::write
)) 
1523                     wxCFData 
cfdaInfo(cfdInfo
.WriteAsXML()); 
1524                     if (cfdaInfo
.IsOk()) 
1526                         if (outdictfile
.Write(cfdaInfo
.GetValue(), cfdaInfo
.GetCount()) != 
1527                             (wxFileOffset
)cfdaInfo
.GetCount()) 
1529                             wxLogDebug(wxT("error in writing to file")); 
1533                             bInfoSuccess 
= true; 
1535 //#if defined(__DARWIN__) 
1536 //                //force launch services to update its database for the finder 
1537 //                OSStatus status = LSRegisterURL((CFURLRef)(CFTypeRef)cfurlBundleLoc, true); 
1538 //                if (status != noErr) 
1540 //                    wxLogDebug(wxT("LSRegisterURL Failed.")); 
1544                         outdictfile
.Close(); 
1548                         outdictfile
.Close(); 
1549                         wxLogDebug(wxT("Could not read in new dictionary")); 
1554                     wxLogDebug(wxString(wxT("Could not open [")) + 
1555                     sInfoPath 
+ wxT("] for writing.")); 
1560                 wxLogDebug(wxT("No info dictionary in main bundle")); 
1565             wxLogDebug(wxT("Can only call associate from bundled app within XXX.app")); 
1570         wxLogDebug(wxT("No main bundle")); 
1576     // on mac you have to embed it into the mac's file reference resource ('FREF' I believe) 
1577     // or, alternately, you could just add an entry to m_hDatabase, but you'd need to get 
1578     // the app's signature somehow... 
1580     OSType processType
, creator
; 
1581     OSStatus status 
= MoreProcGetProcessTypeSignature(NULL
, &processType
, &creator
); 
1583     if (status 
== noErr
) 
1585         Str255 psCreatorName
; 
1588         status 
= FindApplication(creator
, false, psCreatorName
, &dummySpec
); 
1591         status 
= LSFindApplicationForInfo( creator
, NULL
, NULL
, &fsref 
,NULL
); 
1593         status 
= FSGetCatalogInfo(&fsref
, kFSCatInfoNone
, NULL
, &name
, NULL
, NULL
); 
1594         CFStringRef str 
= FSCreateStringFromHFSUniStr( 0 , &name 
); 
1595         CFStringGetPascalString(str
, psCreatorName
, 256, CFStringGetSystemEncoding()); 
1599         if (status 
== noErr
) 
1601             //get the file type if it exists - 
1602             //if it really does then modify the database then save it, 
1603             //otherwise we need to create a whole new entry 
1604             wxFileType
* pFileType 
= GetFileTypeFromExtension(asExtensions
[dwFoundIndex
]); 
1608                 ICGetMapEntry( (ICInstance
) m_hIC
, (Handle
) m_hDatabase
, 
1609                     pFileType
->m_impl
->m_lIndex
, &entry 
); 
1611                 memcpy(entry
.creatorAppName
, psCreatorName
, sizeof(Str255
)); 
1612                 entry
.fileCreator 
= creator
; 
1614                 status 
= ICSetMapEntry( (ICInstance
) m_hIC
, (Handle
) m_hDatabase
, 
1615                     pFileType
->m_impl
->m_lIndex
, &entry 
); 
1618                 if (status 
== noErr
) 
1622                     //kICAttrNoChange means we don't care about attributes such as 
1623                     //locking in the database 
1624 //                    status = ICSetPrefHandle((ICInstance) m_hIC, kICMapping, 
1625 //                                             kICAttrNoChange, (Handle) m_hDatabase); 
1626 //                    if (status == noErr) 
1627 //                        return pFileType; 
1630 //                        wxLogDebug(wxString::Format(wxT("%i - %s"), (int)status, wxT("ICSetPrefHandle failed."))); 
1635                     wxLogDebug(wxString::Format(wxT("%i - %s"), __LINE__
, wxT("ICSetMapEntry failed."))); 
1638                 // failure - cleanup 
1643                 // TODO: Maybe force all 3 of these to be non-empty? 
1644                 Str255 psExtension
, psMimeType
, psDescription
; 
1646                 wxMacStringToPascal(wxString(wxT(".")) + ftInfo
.GetExtensions()[0], psExtension
); 
1647                 wxMacStringToPascal(ftInfo
.GetMimeType(), psMimeType
); 
1648                 wxMacStringToPascal(ftInfo
.GetDescription(), psDescription
); 
1650                 Str255 psPostCreatorName
; 
1651                 wxMacStringToPascal(wxEmptyString
, psPostCreatorName
); 
1653                 //add the entry to the database 
1655                 entry
.totalLength 
= sizeof(ICMapEntry
); 
1656                 entry
.fixedLength 
= kICMapFixedLength
; 
1658                 entry
.fileType 
= 0; //TODO:  File type? 
1659                 entry
.fileCreator 
= creator
; 
1660                 entry
.postCreator 
= 0; 
1661                 entry
.flags 
= kICMapDataForkBit
; //TODO:  Maybe resource is valid by default too? 
1662                 PLstrcpy( entry
.extension 
, psExtension 
) ; 
1663                 memcpy(entry
.creatorAppName
, psCreatorName
, sizeof(Str255
)); 
1664                 memcpy(entry
.postAppName
, psPostCreatorName
, sizeof(Str255
)); 
1665                 memcpy(entry
.MIMEType
, psMimeType
, sizeof(Str255
)); 
1666                 memcpy(entry
.entryName
, psDescription
, sizeof(Str255
)); 
1668                 status 
= ICAddMapEntry( (ICInstance
) m_hIC
, (Handle
) m_hDatabase
, &entry
); 
1669                 if (status 
== noErr
) 
1671                     return GetFileTypeFromExtension(ftInfo
.GetMimeType()); 
1673 //                    kICAttrNoChange means we don't care about attributes such as 
1674 //                    locking in the database 
1675 //                    status = ICSetPrefHandle((ICInstance) m_hIC, kICMapping, 
1676 //                                             kICAttrNoChange, (Handle) m_hDatabase); 
1678                     // return the entry in the database if successful 
1679 //                    if (status == noErr) 
1680 //                        return GetFileTypeFromExtension(ftInfo.GetMimeType()); 
1683 //                        wxLogDebug(wxString::Format(wxT("%i - %s"), __LINE__, wxT("ICSetPrefHandle failed."))); 
1688                     wxLogDebug(wxString::Format(wxT("%i - %s"), __LINE__
, wxT("ICAppMapEntry failed."))); 
1691         } // end if FindApplcation was successful 
1694             wxLogDebug(wxString::Format(wxT("%i - %s"), __LINE__
, wxT("FindApplication failed."))); 
1696     } // end if it could obtain app's signature 
1699         wxLogDebug(wxString::Format(wxT("%i - %s"), __LINE__
, wxT("GetProcessSignature failed."))); 
1706 wxMimeTypesManagerImpl::Unassociate(wxFileType 
*pFileType
) 
1708     wxASSERT(pFileType
); 
1709     bool bInfoSuccess 
= false; 
1711     wxArrayString asExtensions
; 
1712     pFileType
->GetExtensions(asExtensions
); 
1714     if (!asExtensions
.GetCount()) 
1716         wxLogDebug(wxT("Must have extension to disassociate")); 
1720     // Find and write to Info.plist in main bundle (note that some other 
1721     // apps have theirs named differently, i.e. IE's is named Info-macos.plist 
1722     // some apps (non-wx) use the 'plst' resource instead 
1723     CFBundleRef cfbMain 
= CFBundleGetMainBundle(); 
1726         UInt32 dwBundleType
, dwBundleCreator
; 
1727         CFBundleGetPackageInfo(cfbMain
, &dwBundleType
, &dwBundleCreator
); 
1729         // if launching terminal non-app, version will be 'BNDL' (generic bundle, maybe in other cases too), 
1730         // which will give us the incorrect info.plist path 
1731         // otherwise it will be 'APPL', or in the case of a framework, 'FMWK' 
1732         if (dwBundleType 
== 'APPL') 
1735             wxCFURL 
cfurlBundleLoc((CFTypeRef
)CFBundleCopyBundleURL(cfbMain
)); 
1736 //             wxCFURL cfurlBundleLoc((CFTypeRef)CFBundleCopyExecutableURL(cfbMain)); 
1738 //            sInfoPath << wxT("file://"); 
1739             sInfoPath 
<< cfurlBundleLoc
.BuildWXString(); 
1740             sInfoPath 
<< wxT("Contents/Info.plist"); 
1742 //        wxCFDictionary cfdInfo( (CFTypeRef) CFBundleGetInfoDictionary(cfbMain), wxCF_RETAIN ); 
1743             wxCFDictionary cfdInfo
; 
1744             bool bInfoOpenSuccess 
= false; 
1746             if (indictfile
.Open(sInfoPath
, wxFile::read
)) 
1748                 CFIndex cfiBufLen 
= (CFIndex
) indictfile
.Length(); 
1749                 const UInt8
* pBuffer 
= new UInt8
[cfiBufLen
]; 
1750                 indictfile
.Read((void*)pBuffer
, cfiBufLen
); 
1751                 wxCFData 
cfdaInDict(pBuffer
, cfiBufLen
); 
1753                 bInfoOpenSuccess 
= cfdInfo
.ReadAsXML(cfdaInDict
, &sError
); 
1754                 if (!bInfoOpenSuccess
) 
1759             if (bInfoOpenSuccess
) 
1761                 cfdInfo
.MakeMutable( cfdInfo
.GetCount() + 1 ); 
1763                 wxCFArray 
cfaDocTypes( wxCFRetain( cfdInfo
[ wxCFStringRef(wxT("CFBundleDocumentTypes")) ] ) ); 
1765                 if (cfaDocTypes
.IsOk()) 
1767                     bool bEntryFound 
= false; 
1769                     //search for duplicate 
1771                     for (i 
= 0; i 
< cfaDocTypes
.GetCount(); ++i
) 
1773                         wxCFDictionary 
cfdDocTypeEntry( wxCFRetain( cfaDocTypes
[i
] ) ); 
1775                         //A lot of apps dont do to mime types for some reason 
1776                         //so we go by extensions only 
1777                         wxCFArray 
cfaExtensions( wxCFRetain( cfdDocTypeEntry
[ wxCFStringRef(wxT("CFBundleTypeExtensions")) ]) ); 
1779                         if (!cfaExtensions
.IsOk()) 
1782                         for (CFIndex iExt 
= 0; iExt 
< cfaExtensions
.GetCount(); ++iExt
) 
1784                             for (size_t iWXExt 
= 0; iWXExt 
< asExtensions
.GetCount(); ++iWXExt
) 
1786                                 if (asExtensions
[iWXExt
] == 
1787                                     wxCFStringRef( wxCFRetain( (CFStringRef
) cfaExtensions
[iExt
] ) ).AsString()) 
1790                                     cfaDocTypes
.Remove(i
); 
1791                                     cfdInfo
.Set( wxCFStringRef(wxT("CFBundleDocumentTypes")) , cfaDocTypes 
); 
1794                             } //end of wxstring array 
1798                         } //end for cf array 
1806                         cfdInfo
.MakeValidXML(); 
1809                         if (outdictfile
.Open(sInfoPath
, wxFile::write
)) 
1811                             wxCFData 
cfdaInfo(cfdInfo
.WriteAsXML()); 
1812                             if (cfdaInfo
.IsOk()) 
1814                                 if (outdictfile
.Write(cfdaInfo
.GetValue(), cfdaInfo
.GetCount()) != 
1815                                     (wxFileOffset
)cfdaInfo
.GetCount()) 
1817                                     wxLogDebug(wxT("error in writing to file")); 
1821                                     bInfoSuccess 
= true; 
1823 //#if defined(__DARWIN__) 
1824 //                //force launch services to update its database for the finder 
1825 //                OSStatus status = LSRegisterURL((CFURLRef)(CFTypeRef)cfurlBundleLoc, true); 
1826 //                if (status != noErr) 
1828 //                    wxLogDebug(wxT("LSRegisterURL Failed.")); 
1832                                 outdictfile
.Close(); 
1836                                 outdictfile
.Close(); 
1837                                 wxLogDebug(wxT("Could not read in new dictionary")); 
1843                                 wxString(wxT("Could not open [")) + 
1844                                 sInfoPath 
+ wxT("] for writing.")); 
1849                         wxLogDebug(wxT("Entry not found to remove")); 
1852                         wxCFDictionary::PrintOutArray(sPrintOut
, (CFArrayRef
)(CFTypeRef
)cfaDocTypes
); 
1853                         wxLogDebug(sPrintOut
); 
1855                         for (size_t i 
= 0; i 
< asExtensions
.GetCount(); ++i
) 
1856                             wxLogDebug(asExtensions
[i
]); 
1861                     wxLogDebug(wxT("No doc types array found")); 
1862                     wxString sPrintOut
;  cfdInfo
.PrintOut(sPrintOut
);  wxLogDebug(sPrintOut
); 
1867                 wxLogDebug(wxT("No info dictionary in main bundle")); 
1872             wxLogDebug(wxT("Can only call associate from bundled app within XXX.app")); 
1877         wxLogDebug(wxT("No main bundle")); 
1883     // this should be as easy as removing the entry from the database 
1884     // and then saving the database 
1885     OSStatus status 
= ICDeleteMapEntry( (ICInstance
) m_hIC
, (Handle
) m_hDatabase
, 
1886                             pFileType
->m_impl
->m_lIndex
); 
1888     if (status 
== noErr
) 
1892         //kICAttrNoChange means we don't care about attributes such as 
1893         //locking in the database 
1894 //        status = ICSetPrefHandle((ICInstance) m_hIC, kICMapping, 
1895 //                    kICAttrNoChange, (Handle) m_hDatabase); 
1897 //        if (status == noErr) 
1903 //            wxLogDebug(wxString::Format(wxT("%i - %s"), __LINE__, wxT("ICSetPrefHandle failed."))); 
1908         wxLogDebug(wxString::Format(wxT("%i - %s"), __LINE__
, wxT("ICDeleteMapEntry failed."))); 
1915     CFWriteStreamRef cfwsInfo 
= CFWriteStreamCreateWithFile( 
1916            kCFAllocatorDefault
, 
1917            (CFURLRef
) (CFTypeRef
)cfurlInfoLoc 
); 
1922             Boolean bOpened 
= CFWriteStreamOpen(cfwsInfo
); 
1925                 CFStringRef cfsError
; 
1926                 CFIndex cfiWritten 
= CFPropertyListWriteToStream((CFPropertyListRef
)(CFTypeRef
)cfdInfo
, 
1928                                     kCFPropertyListXMLFormat_v1_0
, //100 
1930                 if (cfsError 
&& cfiWritten 
== 0) 
1932                     wxLogDebug(wxCFStringRef(cfsError
).BuildWXString()); 
1934                     cfdInfo
.PrintOut(sMessage
); 
1935                     wxLogDebug(sMessage
); 
1939                     bInfoSuccess 
= true; 
1940 //#if defined(__DARWIN__) 
1941 //                //force launch services to update its database for the finder 
1942 //                OSStatus status = LSRegisterURL((CFURLRef)(CFTypeRef)cfurlBundleLoc, true); 
1943 //                if (status != noErr) 
1945 //                    wxLogDebug(wxT("LSRegisterURL Failed.")); 
1950                 CFWriteStreamClose(cfwsInfo
); 
1953 #endif //wxUSE_MIMETYPE