1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     wxGenericDirCtrl 
   4 // Author:      Harm van der Heijden, Robert Roebling, Julian Smart 
   8 // Copyright:   (c) Harm van der Heijden, Robert Roebling and Julian Smart 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA) 
  13 #pragma implementation "dirctrlg.h" 
  16 // For compilers that support precompilation, includes "wx.h". 
  17 #include "wx/wxprec.h" 
  25 #include "wx/generic/dirctrlg.h" 
  27 #include "wx/module.h" 
  29 #include "wx/button.h" 
  30 #include "wx/layout.h" 
  31 #include "wx/msgdlg.h" 
  32 #include "wx/textctrl.h" 
  33 #include "wx/textdlg.h" 
  34 #include "wx/filefn.h" 
  35 #include "wx/cmndata.h" 
  36 #include "wx/gdicmn.h" 
  38 #include "wx/imaglist.h" 
  42 #include "wx/tokenzr.h" 
  44 #include "wx/settings.h" 
  45 #include "wx/artprov.h" 
  47 #include "wx/mimetype.h" 
  49 #include "wx/choice.h" 
  50 #include "wx/filedlg.h"  // for wxFileDialogBase::ParseWildcard 
  53     #include "wx/statline.h" 
  56 #if defined(__WXMAC__) 
  57   #include  "wx/mac/private.h"  // includes mac headers 
  63 // FIXME - Mingw32 1.0 has both _getdrive() and _chdrive(). For now, let's assume 
  64 //         older releases don't, but it should be verified and the checks modified 
  66 #if !defined(__GNUWIN32__) || (defined(__MINGW32_MAJOR_VERSION) && __MINGW32_MAJOR_VERSION >= 1) 
  67 #if !defined(__WXWINCE__) 
  85 extern bool wxIsDriveAvailable(const wxString
& dirName
); 
  88 #if defined(__WXMAC__) 
  90 #    include "MoreFilesX.h" 
  92 #    include "MoreFilesExtras.h" 
 100 // If compiled under Windows, this macro can cause problems 
 105 // ---------------------------------------------------------------------------- 
 106 // wxGetAvailableDrives, for WINDOWS, DOS, WXPM, MAC, UNIX (returns "/") 
 107 // ---------------------------------------------------------------------------- 
 109 size_t wxGetAvailableDrives(wxArrayString 
&paths
, wxArrayString 
&names
, wxArrayInt 
&icon_ids
) 
 111 #if defined(__WINDOWS__) || defined(__DOS__) || defined(__WXPM__) 
 114     // No logical drives; return "\" 
 115     paths
.Add(wxT("\\")); 
 116     names
.Add(wxT("\\")); 
 118 #elif defined(__WIN32__) 
 119     wxChar driveBuffer
[256]; 
 120     size_t n 
= (size_t) GetLogicalDriveStrings(255, driveBuffer
); 
 125         path
.Printf(wxT("%c:\\"), driveBuffer
[i
]); 
 126         name
.Printf(wxT("%c:"), driveBuffer
[i
]); 
 128         int imageId 
= wxFileIconsTable::drive
; 
 129         int driveType 
= ::GetDriveType(path
); 
 132             case DRIVE_REMOVABLE
: 
 133                 if (path 
== wxT("a:\\") || path 
== wxT("b:\\")) 
 134                     imageId 
= wxFileIconsTable::floppy
; 
 136                     imageId 
= wxFileIconsTable::removeable
; 
 139                 imageId 
= wxFileIconsTable::cdrom
; 
 144                 imageId 
= wxFileIconsTable::drive
; 
 150         icon_ids
.Add(imageId
); 
 152         while (driveBuffer
[i
] != wxT('\0')) 
 155         if (driveBuffer
[i
] == wxT('\0')) 
 161     /* If we can switch to the drive, it exists. */ 
 162     for( drive 
= 1; drive 
<= 26; drive
++ ) 
 165         path
.Printf(wxT("%c:\\"), (char) (drive 
+ 'a' - 1)); 
 166         name
.Printf(wxT("%c:"), (char) (drive 
+ 'A' - 1)); 
 168         if (wxIsDriveAvailable(path
)) 
 172             icon_ids
.Add((drive 
<= 2) ? wxFileIconsTable::floppy 
: wxFileIconsTable::drive
); 
 175 #endif // __WIN32__/!__WIN32__ 
 177 #elif defined(__WXMAC__) 
 180     ItemCount   theVolCount
; 
 181     char        thePath
[FILENAME_MAX
]; 
 183     if (FSGetMountedVolumes(&theVolRefs
, &theVolCount
) == noErr
) { 
 185         ::HLock( (Handle
)theVolRefs 
) ; 
 186         for (index 
= 0; index 
< theVolCount
; ++index
) { 
 187             // get the POSIX path associated with the FSRef 
 188             if ( FSRefMakePath(&((*theVolRefs
)[index
]), 
 189                                  (UInt8 
*)thePath
, sizeof(thePath
)) != noErr 
) { 
 192             // add path separator at end if necessary 
 193             wxString 
path( thePath 
, wxConvLocal
) ; 
 194             if (path
.Last() != wxFILE_SEP_PATH
) { 
 195                 path 
+= wxFILE_SEP_PATH
; 
 197             // get Mac volume name for display 
 198             FSVolumeRefNum vRefNum 
; 
 199             HFSUniStr255 volumeName 
; 
 201             if ( FSGetVRefNum(&((*theVolRefs
)[index
]), &vRefNum
) != noErr 
) { 
 204             if ( FSGetVInfo(vRefNum
, &volumeName
, NULL
, NULL
) != noErr 
) { 
 207             // get C string from Unicode HFS name 
 208             //   see: http://developer.apple.com/carbon/tipsandtricks.html 
 209             CFStringRef cfstr 
= CFStringCreateWithCharacters( kCFAllocatorDefault
, 
 212             //  Do something with str 
 213             char *cstr 
= NewPtr(CFStringGetLength(cfstr
) + 1); 
 214             if (( cstr 
== NULL 
) || 
 215                 !CFStringGetCString(cfstr
, cstr
, CFStringGetLength(cfstr
) + 1, 
 216                                     kCFStringEncodingMacRoman
)) 
 221             wxString 
name( cstr 
, wxConvLocal 
); 
 225             GetVolParmsInfoBuffer volParmsInfo
; 
 227             if ( FSGetVolParms(vRefNum
, sizeof(volParmsInfo
), &volParmsInfo
, &actualSize
) != noErr 
) { 
 234             if ( VolIsEjectable(&volParmsInfo
) ) 
 235                 icon_ids
.Add(wxFileIconsTable::cdrom
); 
 237                 icon_ids
.Add(wxFileIconsTable::drive
); 
 239         ::HUnlock( (Handle
)theVolRefs 
); 
 240         ::DisposeHandle( (Handle
)theVolRefs 
); 
 247         short actualCount 
= 0 ; 
 248         if (OnLine(&volume
, 1, &actualCount
, &index 
) != noErr 
|| actualCount
==0) 
 253         wxString name 
= wxMacFSSpec2MacFilename( &volume 
); 
 254         paths
.Add(name 
+ wxFILE_SEP_PATH
); 
 256         icon_ids
.Add(wxFileIconsTable::drive
); 
 260 #elif defined(__UNIX__) 
 263     icon_ids
.Add(wxFileIconsTable::computer
); 
 265     #error "Unsupported platform in wxGenericDirCtrl!" 
 267     return paths
.GetCount(); 
 270 // ---------------------------------------------------------------------------- 
 271 // wxIsDriveAvailable 
 272 // ---------------------------------------------------------------------------- 
 276 bool wxIsDriveAvailable(const wxString
& dirName
) 
 278     // FIXME_MGL - this method leads to hang up under Watcom for some reason 
 280     if ( dirName
.Len() == 3 && dirName
[1u] == wxT(':') ) 
 282         wxString 
dirNameLower(dirName
.Lower()); 
 283         // VS: always return TRUE for removable media, since Win95 doesn't 
 284         //     like it when MS-DOS app accesses empty floppy drive 
 285         return (dirNameLower
[0u] == wxT('a') || 
 286                 dirNameLower
[0u] == wxT('b') || 
 287                 wxPathExists(dirNameLower
)); 
 294 #elif defined(__WINDOWS__) || defined(__WXPM__) 
 296 int setdrive(int drive
) 
 300 #elif defined(__GNUWIN32__) && \ 
 301     (defined(__MINGW32_MAJOR_VERSION) && __MINGW32_MAJOR_VERSION >= 1) 
 302     return _chdrive(drive
); 
 306         if (drive 
< 1 || drive 
> 31) 
 308         newdrive
[0] = (wxChar
)(wxT('A') + drive 
- 1); 
 309         newdrive
[1] = wxT(':'); 
 310         newdrive
[2] = wxT('\0'); 
 311 #if defined(__WXMSW__) 
 312         if (::SetCurrentDirectory(newdrive
)) 
 314     // VA doesn't know what LPSTR is and has its own set 
 315         if (DosSetCurrentDir((PSZ
)newdrive
)) 
 323 bool wxIsDriveAvailable(const wxString
& dirName
) 
 329     UINT errorMode 
= SetErrorMode(SEM_FAILCRITICALERRORS 
| SEM_NOOPENFILEERRORBOX
); 
 333     // Check if this is a root directory and if so, 
 334     // whether the drive is available. 
 335     if (dirName
.Len() == 3 && dirName
[(size_t)1] == wxT(':')) 
 337         wxString 
dirNameLower(dirName
.Lower()); 
 338 #if defined(__GNUWIN32__) && !(defined(__MINGW32_MAJOR_VERSION) && __MINGW32_MAJOR_VERSION >= 1) 
 339         success 
= wxPathExists(dirNameLower
); 
 341         int currentDrive 
= _getdrive(); 
 342         int thisDrive 
= (int) (dirNameLower
[(size_t)0] - 'a' + 1) ; 
 343         int err 
= setdrive( thisDrive 
) ; 
 344         setdrive( currentDrive 
); 
 353     (void) SetErrorMode(errorMode
); 
 359 #endif // __WINDOWS__ || __WXPM__ 
 362 // Function which is called by quick sort. We want to override the default wxArrayString behaviour, 
 363 // and sort regardless of case. 
 364 static int LINKAGEMODE 
wxDirCtrlStringCompareFunction(wxString
* strFirst
, wxString
* strSecond
) 
 366     return strFirst
->CmpNoCase(*strSecond
); 
 369 //----------------------------------------------------------------------------- 
 371 //----------------------------------------------------------------------------- 
 373 wxDirItemData::wxDirItemData(const wxString
& path
, const wxString
& name
, 
 378     /* Insert logic to detect hidden files here 
 379      * In UnixLand we just check whether the first char is a dot 
 380      * For FileNameFromPath read LastDirNameInThisPath ;-) */ 
 381     // m_isHidden = (bool)(wxFileNameFromPath(*m_path)[0] == '.'); 
 383     m_isExpanded 
= FALSE
; 
 387 wxDirItemData::~wxDirItemData() 
 391 void wxDirItemData::SetNewDirName(const wxString
& path
) 
 394     m_name 
= wxFileNameFromPath(path
); 
 397 bool wxDirItemData::HasSubDirs() const 
 399     if (m_path
.IsEmpty()) 
 405         if ( !dir
.Open(m_path
) ) 
 409     return dir
.HasSubDirs(); 
 412 bool wxDirItemData::HasFiles(const wxString
& WXUNUSED(spec
)) const 
 414     if (m_path
.IsEmpty()) 
 420         if ( !dir
.Open(m_path
) ) 
 424     return dir
.HasFiles(); 
 427 //----------------------------------------------------------------------------- 
 429 //----------------------------------------------------------------------------- 
 431 IMPLEMENT_DYNAMIC_CLASS(wxGenericDirCtrl
, wxControl
) 
 437         (long) defaultfilter) 
 440 BEGIN_EVENT_TABLE(wxGenericDirCtrl
, wxControl
) 
 441   EVT_TREE_ITEM_EXPANDING     (-1, wxGenericDirCtrl::OnExpandItem
) 
 442   EVT_TREE_ITEM_COLLAPSED     (-1, wxGenericDirCtrl::OnCollapseItem
) 
 443   EVT_TREE_BEGIN_LABEL_EDIT   (-1, wxGenericDirCtrl::OnBeginEditItem
) 
 444   EVT_TREE_END_LABEL_EDIT     (-1, wxGenericDirCtrl::OnEndEditItem
) 
 445   EVT_SIZE                    (wxGenericDirCtrl::OnSize
) 
 448 wxGenericDirCtrl::wxGenericDirCtrl(void) 
 453 bool wxGenericDirCtrl::Create(wxWindow 
*parent
, 
 459                               const wxString
& filter
, 
 461                               const wxString
& name
) 
 463     if (!wxControl::Create(parent
, id
, pos
, size
, style
, wxDefaultValidator
, name
)) 
 466     SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE
)); 
 470     long treeStyle 
= wxTR_HAS_BUTTONS 
| wxTR_HIDE_ROOT
; 
 472     if (style 
& wxDIRCTRL_EDIT_LABELS
) 
 473         treeStyle 
|= wxTR_EDIT_LABELS
; 
 475     if ((style 
& wxDIRCTRL_3D_INTERNAL
) == 0) 
 476         treeStyle 
|= wxNO_BORDER
; 
 478         treeStyle 
|= wxBORDER_SUNKEN
; 
 480     long filterStyle 
= 0; 
 481     if ((style 
& wxDIRCTRL_3D_INTERNAL
) == 0) 
 482         filterStyle 
|= wxNO_BORDER
; 
 484         filterStyle 
|= wxBORDER_SUNKEN
; 
 486     m_treeCtrl 
= new wxTreeCtrl(this, wxID_TREECTRL
, pos
, size
, treeStyle
); 
 488     if (!filter
.IsEmpty() && (style 
& wxDIRCTRL_SHOW_FILTERS
)) 
 489         m_filterListCtrl 
= new wxDirFilterListCtrl(this, wxID_FILTERLISTCTRL
, wxDefaultPosition
, wxDefaultSize
, filterStyle
); 
 494     SetFilterIndex(defaultFilter
); 
 496     if (m_filterListCtrl
) 
 497         m_filterListCtrl
->FillFilterList(filter
, defaultFilter
); 
 499     m_treeCtrl
->SetImageList(wxTheFileIconsTable
->GetSmallImageList()); 
 501     m_showHidden 
= FALSE
; 
 502     wxDirItemData
* rootData 
= new wxDirItemData(wxT(""), wxT(""), TRUE
); 
 506 #if defined(__WINDOWS__) || defined(__WXPM__) || defined(__DOS__) 
 507     rootName 
= _("Computer"); 
 509     rootName 
= _("Sections"); 
 512     m_rootId 
= m_treeCtrl
->AddRoot( rootName
, 3, -1, rootData
); 
 513     m_treeCtrl
->SetItemHasChildren(m_rootId
); 
 514     ExpandDir(m_rootId
); // automatically expand first level 
 516     // Expand and select the default path 
 517     if (!m_defaultPath
.IsEmpty()) 
 518         ExpandPath(m_defaultPath
); 
 525 wxGenericDirCtrl::~wxGenericDirCtrl() 
 529 void wxGenericDirCtrl::Init() 
 531     m_showHidden 
= FALSE
; 
 533     m_currentFilterStr 
= wxEmptyString
; // Default: any file 
 535     m_filterListCtrl 
= NULL
; 
 538 void wxGenericDirCtrl::ShowHidden( bool show 
) 
 542     wxString path 
= GetPath(); 
 548 wxGenericDirCtrl::AddSection(const wxString
& path
, const wxString
& name
, int imageId
) 
 550     wxDirItemData 
*dir_item 
= new wxDirItemData(path
,name
,TRUE
); 
 552     wxTreeItemId id 
= AppendItem( m_rootId
, name
, imageId
, -1, dir_item
); 
 554     m_treeCtrl
->SetItemHasChildren(id
); 
 559 void wxGenericDirCtrl::SetupSections() 
 561     wxArrayString paths
, names
; 
 564     size_t n
, count 
= wxGetAvailableDrives(paths
, names
, icons
); 
 566     for (n 
= 0; n 
< count
; n
++) 
 568         AddSection(paths
[n
], names
[n
], icons
[n
]); 
 572 void wxGenericDirCtrl::OnBeginEditItem(wxTreeEvent 
&event
) 
 574     // don't rename the main entry "Sections" 
 575     if (event
.GetItem() == m_rootId
) 
 581     // don't rename the individual sections 
 582     if (m_treeCtrl
->GetItemParent( event
.GetItem() ) == m_rootId
) 
 589 void wxGenericDirCtrl::OnEndEditItem(wxTreeEvent 
&event
) 
 591     if ((event
.GetLabel().IsEmpty()) || 
 592         (event
.GetLabel() == _(".")) || 
 593         (event
.GetLabel() == _("..")) || 
 594         (event
.GetLabel().Find(wxT('/')) != wxNOT_FOUND
) || 
 595         (event
.GetLabel().Find(wxT('\\')) != wxNOT_FOUND
) || 
 596         (event
.GetLabel().Find(wxT('|')) != wxNOT_FOUND
)) 
 598         wxMessageDialog 
dialog(this, _("Illegal directory name."), _("Error"), wxOK 
| wxICON_ERROR 
); 
 604     wxTreeItemId id 
= event
.GetItem(); 
 605     wxDirItemData 
*data 
= (wxDirItemData
*)m_treeCtrl
->GetItemData( id 
); 
 608     wxString 
new_name( wxPathOnly( data
->m_path 
) ); 
 609     new_name 
+= wxString(wxFILE_SEP_PATH
); 
 610     new_name 
+= event
.GetLabel(); 
 614     if (wxFileExists(new_name
)) 
 616         wxMessageDialog 
dialog(this, _("File name exists already."), _("Error"), wxOK 
| wxICON_ERROR 
); 
 621     if (wxRenameFile(data
->m_path
,new_name
)) 
 623         data
->SetNewDirName( new_name 
); 
 627         wxMessageDialog 
dialog(this, _("Operation not permitted."), _("Error"), wxOK 
| wxICON_ERROR 
); 
 633 void wxGenericDirCtrl::OnExpandItem(wxTreeEvent 
&event
) 
 635     wxTreeItemId parentId 
= event
.GetItem(); 
 637     // VS: this is needed because the event handler is called from wxTreeCtrl 
 638     //     ctor when wxTR_HIDE_ROOT was specified 
 640     if (!m_rootId
.IsOk()) 
 642         m_rootId 
= m_treeCtrl
->GetRootItem(); 
 647 void wxGenericDirCtrl::OnCollapseItem(wxTreeEvent 
&event 
) 
 649     CollapseDir(event
.GetItem()); 
 652 void wxGenericDirCtrl::CollapseDir(wxTreeItemId parentId
) 
 656     wxDirItemData 
*data 
= (wxDirItemData 
*) m_treeCtrl
->GetItemData(parentId
); 
 657     if (!data
->m_isExpanded
) 
 660     data
->m_isExpanded 
= FALSE
; 
 661     wxTreeItemIdValue cookie
; 
 662     /* Workaround because DeleteChildren has disapeared (why?) and 
 663      * CollapseAndReset doesn't work as advertised (deletes parent too) */ 
 664     child 
= m_treeCtrl
->GetFirstChild(parentId
, cookie
); 
 667         m_treeCtrl
->Delete(child
); 
 668         /* Not GetNextChild below, because the cookie mechanism can't 
 669          * handle disappearing children! */ 
 670         child 
= m_treeCtrl
->GetFirstChild(parentId
, cookie
); 
 674 void wxGenericDirCtrl::ExpandDir(wxTreeItemId parentId
) 
 676     wxDirItemData 
*data 
= (wxDirItemData 
*) m_treeCtrl
->GetItemData(parentId
); 
 678     if (data
->m_isExpanded
) 
 681     data
->m_isExpanded 
= TRUE
; 
 683     if (parentId 
== m_treeCtrl
->GetRootItem()) 
 691     wxString search
,path
,filename
; 
 693     wxString 
dirName(data
->m_path
); 
 695 #if defined(__WINDOWS__) || defined(__DOS__) || defined(__WXPM__) 
 696     // Check if this is a root directory and if so, 
 697     // whether the drive is avaiable. 
 698     if (!wxIsDriveAvailable(dirName
)) 
 700         data
->m_isExpanded 
= FALSE
; 
 701         //wxMessageBox(wxT("Sorry, this drive is not available.")); 
 706     // This may take a longish time. Go to busy cursor 
 709 #if defined(__WINDOWS__) || defined(__DOS__) || defined(__WXPM__) 
 710     if (dirName
.Last() == ':') 
 711         dirName 
+= wxString(wxFILE_SEP_PATH
); 
 715     wxArrayString filenames
; 
 718     wxString eachFilename
; 
 725         int style 
= wxDIR_DIRS
; 
 726         if (m_showHidden
) style 
|= wxDIR_HIDDEN
; 
 727         if (d
.GetFirst(& eachFilename
, wxEmptyString
, style
)) 
 731                 if ((eachFilename 
!= wxT(".")) && (eachFilename 
!= wxT(".."))) 
 733                     dirs
.Add(eachFilename
); 
 736             while (d
.GetNext(& eachFilename
)); 
 739     dirs
.Sort(wxDirCtrlStringCompareFunction
); 
 741     // Now do the filenames -- but only if we're allowed to 
 742     if ((GetWindowStyle() & wxDIRCTRL_DIR_ONLY
) == 0) 
 750             if (d
.GetFirst(& eachFilename
, m_currentFilterStr
, wxDIR_FILES
)) 
 754                     if ((eachFilename 
!= wxT(".")) && (eachFilename 
!= wxT(".."))) 
 756                         filenames
.Add(eachFilename
); 
 759                 while (d
.GetNext(& eachFilename
)); 
 762         filenames
.Sort(wxDirCtrlStringCompareFunction
); 
 765     // Add the sorted dirs 
 767     for (i 
= 0; i 
< dirs
.Count(); i
++) 
 769         wxString 
eachFilename(dirs
[i
]); 
 771         if (!wxEndsWithPathSeparator(path
)) 
 772             path 
+= wxString(wxFILE_SEP_PATH
); 
 773         path 
+= eachFilename
; 
 775         wxDirItemData 
*dir_item 
= new wxDirItemData(path
,eachFilename
,TRUE
); 
 776         wxTreeItemId id 
= AppendItem( parentId
, eachFilename
, 
 777                                       wxFileIconsTable::folder
, -1, dir_item
); 
 778         m_treeCtrl
->SetItemImage( id
, wxFileIconsTable::folder_open
, 
 779                                   wxTreeItemIcon_Expanded 
); 
 781         // Has this got any children? If so, make it expandable. 
 782         // (There are two situations when a dir has children: either it 
 783         // has subdirectories or it contains files that weren't filtered 
 784         // out. The latter only applies to dirctrl with files.) 
 785         if ( dir_item
->HasSubDirs() || 
 786              (((GetWindowStyle() & wxDIRCTRL_DIR_ONLY
) == 0) && 
 787                dir_item
->HasFiles(m_currentFilterStr
)) ) 
 789             m_treeCtrl
->SetItemHasChildren(id
); 
 793     // Add the sorted filenames 
 794     if ((GetWindowStyle() & wxDIRCTRL_DIR_ONLY
) == 0) 
 796         for (i 
= 0; i 
< filenames
.Count(); i
++) 
 798             wxString 
eachFilename(filenames
[i
]); 
 800             if (!wxEndsWithPathSeparator(path
)) 
 801                 path 
+= wxString(wxFILE_SEP_PATH
); 
 802             path 
+= eachFilename
; 
 803             //path = dirName + wxString(wxT("/")) + eachFilename; 
 804             wxDirItemData 
*dir_item 
= new wxDirItemData(path
,eachFilename
,FALSE
); 
 805             int image_id 
= wxFileIconsTable::file
; 
 806             if (eachFilename
.Find(wxT('.')) != wxNOT_FOUND
) 
 807                 image_id 
= wxTheFileIconsTable
->GetIconID(eachFilename
.AfterLast(wxT('.'))); 
 808             (void) AppendItem( parentId
, eachFilename
, image_id
, -1, dir_item
); 
 813 void wxGenericDirCtrl::ReCreateTree() 
 815     CollapseDir(m_treeCtrl
->GetRootItem()); 
 816     ExpandDir(m_treeCtrl
->GetRootItem()); 
 819 // Find the child that matches the first part of 'path'. 
 820 // E.g. if a child path is "/usr" and 'path' is "/usr/include" 
 821 // then the child for /usr is returned. 
 822 wxTreeItemId 
wxGenericDirCtrl::FindChild(wxTreeItemId parentId
, const wxString
& path
, bool& done
) 
 824     wxString 
path2(path
); 
 826     // Make sure all separators are as per the current platform 
 827     path2
.Replace(wxT("\\"), wxString(wxFILE_SEP_PATH
)); 
 828     path2
.Replace(wxT("/"), wxString(wxFILE_SEP_PATH
)); 
 830     // Append a separator to foil bogus substring matching 
 831     path2 
+= wxString(wxFILE_SEP_PATH
); 
 833     // In MSW or PM, case is not significant 
 834 #if defined(__WINDOWS__) || defined(__DOS__) || defined(__WXPM__) 
 838     wxTreeItemIdValue cookie
; 
 839     wxTreeItemId childId 
= m_treeCtrl
->GetFirstChild(parentId
, cookie
); 
 840     while (childId
.IsOk()) 
 842         wxDirItemData
* data 
= (wxDirItemData
*) m_treeCtrl
->GetItemData(childId
); 
 844         if (data 
&& !data
->m_path
.IsEmpty()) 
 846             wxString 
childPath(data
->m_path
); 
 847             if (!wxEndsWithPathSeparator(childPath
)) 
 848                 childPath 
+= wxString(wxFILE_SEP_PATH
); 
 850             // In MSW and PM, case is not significant 
 851 #if defined(__WINDOWS__) || defined(__DOS__) || defined(__WXPM__) 
 852             childPath
.MakeLower(); 
 855             if (childPath
.Len() <= path2
.Len()) 
 857                 wxString path3 
= path2
.Mid(0, childPath
.Len()); 
 858                 if (childPath 
== path3
) 
 860                     if (path3
.Len() == path2
.Len()) 
 869         childId 
= m_treeCtrl
->GetNextChild(parentId
, cookie
); 
 871     wxTreeItemId invalid
; 
 875 // Try to expand as much of the given path as possible, 
 876 // and select the given tree item. 
 877 bool wxGenericDirCtrl::ExpandPath(const wxString
& path
) 
 880     wxTreeItemId id 
= FindChild(m_rootId
, path
, done
); 
 881     wxTreeItemId lastId 
= id
; // The last non-zero id 
 882     while (id
.IsOk() && !done
) 
 886         id 
= FindChild(id
, path
, done
); 
 892         wxDirItemData 
*data 
= (wxDirItemData 
*) m_treeCtrl
->GetItemData(lastId
); 
 895             m_treeCtrl
->Expand(lastId
); 
 897         if ((GetWindowStyle() & wxDIRCTRL_SELECT_FIRST
) && data
->m_isDir
) 
 899             // Find the first file in this directory 
 900             wxTreeItemIdValue cookie
; 
 901             wxTreeItemId childId 
= m_treeCtrl
->GetFirstChild(lastId
, cookie
); 
 902             bool selectedChild 
= FALSE
; 
 903             while (childId
.IsOk()) 
 905                 wxDirItemData
* data 
= (wxDirItemData
*) m_treeCtrl
->GetItemData(childId
); 
 907                 if (data 
&& data
->m_path 
!= wxT("") && !data
->m_isDir
) 
 909                     m_treeCtrl
->SelectItem(childId
); 
 910                     m_treeCtrl
->EnsureVisible(childId
); 
 911                     selectedChild 
= TRUE
; 
 914                 childId 
= m_treeCtrl
->GetNextChild(lastId
, cookie
); 
 918                 m_treeCtrl
->SelectItem(lastId
); 
 919                 m_treeCtrl
->EnsureVisible(lastId
); 
 924             m_treeCtrl
->SelectItem(lastId
); 
 925             m_treeCtrl
->EnsureVisible(lastId
); 
 934 wxString 
wxGenericDirCtrl::GetPath() const 
 936     wxTreeItemId id 
= m_treeCtrl
->GetSelection(); 
 939         wxDirItemData
* data 
= (wxDirItemData
*) m_treeCtrl
->GetItemData(id
); 
 943         return wxEmptyString
; 
 946 wxString 
wxGenericDirCtrl::GetFilePath() const 
 948     wxTreeItemId id 
= m_treeCtrl
->GetSelection(); 
 951         wxDirItemData
* data 
= (wxDirItemData
*) m_treeCtrl
->GetItemData(id
); 
 953             return wxEmptyString
; 
 958         return wxEmptyString
; 
 961 void wxGenericDirCtrl::SetPath(const wxString
& path
) 
 963     m_defaultPath 
= path
; 
 970 void wxGenericDirCtrl::FindChildFiles(wxTreeItemId id
, int dirFlags
, wxArrayString
& filenames
) 
 972     wxDirItemData 
*data 
= (wxDirItemData 
*) m_treeCtrl
->GetItemData(id
); 
 974     // This may take a longish time. Go to busy cursor 
 979     wxString search
,path
,filename
; 
 981     wxString 
dirName(data
->m_path
); 
 983 #if defined(__WXMSW__) || defined(__WXPM__) 
 984     if (dirName
.Last() == ':') 
 985         dirName 
+= wxString(wxFILE_SEP_PATH
); 
 989     wxString eachFilename
; 
 996         if (d
.GetFirst(& eachFilename
, m_currentFilterStr
, dirFlags
)) 
1000                 if ((eachFilename 
!= wxT(".")) && (eachFilename 
!= wxT(".."))) 
1002                     filenames
.Add(eachFilename
); 
1005             while (d
.GetNext(& eachFilename
)) ; 
1011 void wxGenericDirCtrl::SetFilterIndex(int n
) 
1013     m_currentFilter 
= n
; 
1016     if (ExtractWildcard(m_filter
, n
, f
, d
)) 
1017         m_currentFilterStr 
= f
; 
1019         m_currentFilterStr 
= wxT("*.*"); 
1022 void wxGenericDirCtrl::SetFilter(const wxString
& filter
) 
1027     if (ExtractWildcard(m_filter
, m_currentFilter
, f
, d
)) 
1028         m_currentFilterStr 
= f
; 
1030         m_currentFilterStr 
= wxT("*.*"); 
1033 // Extract description and actual filter from overall filter string 
1034 bool wxGenericDirCtrl::ExtractWildcard(const wxString
& filterStr
, int n
, wxString
& filter
, wxString
& description
) 
1036     wxArrayString filters
, descriptions
; 
1037     int count 
= ParseFilter(filterStr
, filters
, descriptions
); 
1038     if (count 
> 0 && n 
< count
) 
1040         filter 
= filters
[n
]; 
1041         description 
= descriptions
[n
]; 
1048 // Parses the global filter, returning the number of filters. 
1049 // Returns 0 if none or if there's a problem. 
1050 // filterStr is in the form: "All files (*.*)|*.*|JPEG Files (*.jpeg)|*.jpg" 
1052 int wxGenericDirCtrl::ParseFilter(const wxString
& filterStr
, wxArrayString
& filters
, wxArrayString
& descriptions
) 
1054     return wxFileDialogBase::ParseWildcard(filterStr
, descriptions
, filters 
); 
1057 void wxGenericDirCtrl::DoResize() 
1059     wxSize sz 
= GetClientSize(); 
1060     int verticalSpacing 
= 3; 
1064         if (m_filterListCtrl
) 
1067             // For some reason, this is required in order for the 
1068             // correct control height to always be returned, rather 
1069             // than the drop-down list height which is sometimes returned. 
1070             wxSize oldSize 
= m_filterListCtrl
->GetSize(); 
1071             m_filterListCtrl
->SetSize(-1, -1, oldSize
.x
+10, -1, wxSIZE_USE_EXISTING
); 
1072             m_filterListCtrl
->SetSize(-1, -1, oldSize
.x
, -1, wxSIZE_USE_EXISTING
); 
1074             filterSz 
= m_filterListCtrl
->GetSize(); 
1075             sz
.y 
-= (filterSz
.y 
+ verticalSpacing
); 
1077         m_treeCtrl
->SetSize(0, 0, sz
.x
, sz
.y
); 
1078         if (m_filterListCtrl
) 
1080             m_filterListCtrl
->SetSize(0, sz
.y 
+ verticalSpacing
, sz
.x
, filterSz
.y
); 
1081             // Don't know why, but this needs refreshing after a resize (wxMSW) 
1082             m_filterListCtrl
->Refresh(); 
1088 void wxGenericDirCtrl::OnSize(wxSizeEvent
& WXUNUSED(event
)) 
1093 wxTreeItemId 
wxGenericDirCtrl::AppendItem (const wxTreeItemId 
& parent
, 
1094                                            const wxString 
& text
, 
1095                                            int image
, int selectedImage
, 
1096                                            wxTreeItemData 
* data
) 
1098   wxTreeCtrl 
*treeCtrl 
= GetTreeCtrl (); 
1100   wxASSERT (treeCtrl
); 
1104     return treeCtrl
->AppendItem (parent
, text
, image
, selectedImage
, data
); 
1108     return wxTreeItemId(); 
1113 //----------------------------------------------------------------------------- 
1114 // wxDirFilterListCtrl 
1115 //----------------------------------------------------------------------------- 
1117 IMPLEMENT_CLASS(wxDirFilterListCtrl
, wxChoice
) 
1119 BEGIN_EVENT_TABLE(wxDirFilterListCtrl
, wxChoice
) 
1120     EVT_CHOICE(-1, wxDirFilterListCtrl::OnSelFilter
) 
1123 bool wxDirFilterListCtrl::Create(wxGenericDirCtrl
* parent
, const wxWindowID id
, 
1129     return wxChoice::Create(parent
, id
, pos
, size
, 0, NULL
, style
); 
1132 void wxDirFilterListCtrl::Init() 
1137 void wxDirFilterListCtrl::OnSelFilter(wxCommandEvent
& WXUNUSED(event
)) 
1139     int sel 
= GetSelection(); 
1141     wxString currentPath 
= m_dirCtrl
->GetPath(); 
1143     m_dirCtrl
->SetFilterIndex(sel
); 
1145     // If the filter has changed, the view is out of date, so 
1146     // collapse the tree. 
1147     m_dirCtrl
->ReCreateTree(); 
1149     // Try to restore the selection, or at least the directory 
1150     m_dirCtrl
->ExpandPath(currentPath
); 
1153 void wxDirFilterListCtrl::FillFilterList(const wxString
& filter
, int defaultFilter
) 
1156     wxArrayString descriptions
, filters
; 
1157     size_t n 
= (size_t) m_dirCtrl
->ParseFilter(filter
, filters
, descriptions
); 
1159     if (n 
> 0 && defaultFilter 
< (int) n
) 
1162         for (i 
= 0; i 
< n
; i
++) 
1163             Append(descriptions
[i
]); 
1164         SetSelection(defaultFilter
); 
1169 // ---------------------------------------------------------------------------- 
1170 // wxFileIconsTable icons 
1171 // ---------------------------------------------------------------------------- 
1174 static const char * file_icons_tbl_folder_open_xpm
[] = { 
1175 /* width height ncolors chars_per_pixel */ 
1203 static const char * file_icons_tbl_computer_xpm
[] = { 
1230 static const char * file_icons_tbl_drive_xpm
[] = { 
1257 static const char *file_icons_tbl_cdrom_xpm
[] = { 
1287 static const char * file_icons_tbl_floppy_xpm
[] = { 
1314 static const char * file_icons_tbl_removeable_xpm
[] = { 
1340 // ---------------------------------------------------------------------------- 
1341 // wxFileIconsTable & friends 
1342 // ---------------------------------------------------------------------------- 
1344 // global instance of a wxFileIconsTable 
1345 wxFileIconsTable
* wxTheFileIconsTable 
= (wxFileIconsTable 
*)NULL
; 
1347 // A module to allow icons table cleanup 
1349 class wxFileIconsTableModule
: public wxModule
 
1351 DECLARE_DYNAMIC_CLASS(wxFileIconsTableModule
) 
1353     wxFileIconsTableModule() {} 
1354     bool OnInit() { wxTheFileIconsTable 
= new wxFileIconsTable
; return TRUE
; } 
1357         if (wxTheFileIconsTable
) 
1359             delete wxTheFileIconsTable
; 
1360             wxTheFileIconsTable 
= NULL
; 
1365 IMPLEMENT_DYNAMIC_CLASS(wxFileIconsTableModule
, wxModule
) 
1367 class wxFileIconEntry 
: public wxObject
 
1370     wxFileIconEntry(int i
) { id 
= i
; } 
1375 wxFileIconsTable::wxFileIconsTable() 
1378     m_smallImageList 
= NULL
; 
1381 wxFileIconsTable::~wxFileIconsTable() 
1385         WX_CLEAR_HASH_TABLE(*m_HashTable
); 
1388     if (m_smallImageList
) delete m_smallImageList
; 
1391 // delayed initialization - wait until first use (wxArtProv not created yet) 
1392 void wxFileIconsTable::Create() 
1394     wxCHECK_RET(!m_smallImageList 
&& !m_HashTable
, wxT("creating icons twice")); 
1395     m_HashTable 
= new wxHashTable(wxKEY_STRING
); 
1396     m_smallImageList 
= new wxImageList(16, 16); 
1399     m_smallImageList
->Add(wxArtProvider::GetBitmap(wxART_FOLDER
, wxART_CMN_DIALOG
)); 
1401     m_smallImageList
->Add(wxIcon(file_icons_tbl_folder_open_xpm
)); 
1403     m_smallImageList
->Add(wxIcon(file_icons_tbl_computer_xpm
)); 
1405     m_smallImageList
->Add(wxIcon(file_icons_tbl_drive_xpm
)); 
1407     m_smallImageList
->Add(wxIcon(file_icons_tbl_cdrom_xpm
)); 
1409     m_smallImageList
->Add(wxIcon(file_icons_tbl_floppy_xpm
)); 
1411     m_smallImageList
->Add(wxIcon(file_icons_tbl_removeable_xpm
)); 
1413     m_smallImageList
->Add(wxArtProvider::GetBitmap(wxART_NORMAL_FILE
, wxART_CMN_DIALOG
)); 
1415     if (GetIconID(wxEmptyString
, _T("application/x-executable")) == file
) 
1417         m_smallImageList
->Add(wxArtProvider::GetBitmap(wxART_EXECUTABLE_FILE
, wxART_CMN_DIALOG
)); 
1418         delete m_HashTable
->Get(_T("exe")); 
1419         m_HashTable
->Delete(_T("exe")); 
1420         m_HashTable
->Put(_T("exe"), new wxFileIconEntry(executable
)); 
1422     /* else put into list by GetIconID 
1423        (KDE defines application/x-executable for *.exe and has nice icon) 
1427 wxImageList 
*wxFileIconsTable::GetSmallImageList() 
1429     if (!m_smallImageList
) 
1432     return m_smallImageList
; 
1436 // VS: we don't need this function w/o wxMimeTypesManager because we'll only have 
1437 //     one icon and we won't resize it 
1439 static wxBitmap 
CreateAntialiasedBitmap(const wxImage
& img
) 
1441     wxImage 
smallimg (16, 16); 
1442     unsigned char *p1
, *p2
, *ps
; 
1443     unsigned char mr 
= img
.GetMaskRed(), 
1444                   mg 
= img
.GetMaskGreen(), 
1445                   mb 
= img
.GetMaskBlue(); 
1448     unsigned sr
, sg
, sb
, smask
; 
1450     p1 
= img
.GetData(), p2 
= img
.GetData() + 3 * 32, ps 
= smallimg
.GetData(); 
1451     smallimg
.SetMaskColour(mr
, mr
, mr
); 
1453     for (y 
= 0; y 
< 16; y
++) 
1455         for (x 
= 0; x 
< 16; x
++) 
1457             sr 
= sg 
= sb 
= smask 
= 0; 
1458             if (p1
[0] != mr 
|| p1
[1] != mg 
|| p1
[2] != mb
) 
1459                 sr 
+= p1
[0], sg 
+= p1
[1], sb 
+= p1
[2]; 
1462             if (p1
[0] != mr 
|| p1
[1] != mg 
|| p1
[2] != mb
) 
1463                 sr 
+= p1
[0], sg 
+= p1
[1], sb 
+= p1
[2]; 
1466             if (p2
[0] != mr 
|| p2
[1] != mg 
|| p2
[2] != mb
) 
1467                 sr 
+= p2
[0], sg 
+= p2
[1], sb 
+= p2
[2]; 
1470             if (p2
[0] != mr 
|| p2
[1] != mg 
|| p2
[2] != mb
) 
1471                 sr 
+= p2
[0], sg 
+= p2
[1], sb 
+= p2
[2]; 
1476                 ps
[0] = ps
[1] = ps
[2] = mr
; 
1478                 ps
[0] = sr 
>> 2, ps
[1] = sg 
>> 2, ps
[2] = sb 
>> 2; 
1481         p1 
+= 32 * 3, p2 
+= 32 * 3; 
1484     return wxBitmap(smallimg
); 
1487 // finds empty borders and return non-empty area of image: 
1488 static wxImage 
CutEmptyBorders(const wxImage
& img
) 
1490     unsigned char mr 
= img
.GetMaskRed(), 
1491                   mg 
= img
.GetMaskGreen(), 
1492                   mb 
= img
.GetMaskBlue(); 
1493     unsigned char *dt 
= img
.GetData(), *dttmp
; 
1494     unsigned w 
= img
.GetWidth(), h 
= img
.GetHeight(); 
1496     unsigned top
, bottom
, left
, right
, i
; 
1499 #define MK_DTTMP(x,y)      dttmp = dt + ((x + y * w) * 3) 
1500 #define NOEMPTY_PIX(empt)  if (dttmp[0] != mr || dttmp[1] != mg || dttmp[2] != mb) {empt = FALSE; break;} 
1502     for (empt 
= TRUE
, top 
= 0; empt 
&& top 
< h
; top
++) 
1505         for (i 
= 0; i 
< w
; i
++, dttmp
+=3) 
1508     for (empt 
= TRUE
, bottom 
= h
-1; empt 
&& bottom 
> top
; bottom
--) 
1510         MK_DTTMP(0, bottom
); 
1511         for (i 
= 0; i 
< w
; i
++, dttmp
+=3) 
1514     for (empt 
= TRUE
, left 
= 0; empt 
&& left 
< w
; left
++) 
1517         for (i 
= 0; i 
< h
; i
++, dttmp
+=3*w
) 
1520     for (empt 
= TRUE
, right 
= w
-1; empt 
&& right 
> left
; right
--) 
1523         for (i 
= 0; i 
< h
; i
++, dttmp
+=3*w
) 
1526     top
--, left
--, bottom
++, right
++; 
1528     return img
.GetSubImage(wxRect(left
, top
, right 
- left 
+ 1, bottom 
- top 
+ 1)); 
1530 #endif // wxUSE_MIMETYPE 
1532 int wxFileIconsTable::GetIconID(const wxString
& extension
, const wxString
& mime
) 
1534     if (!m_smallImageList
) 
1538     if (!extension
.IsEmpty()) 
1540         wxFileIconEntry 
*entry 
= (wxFileIconEntry
*) m_HashTable
->Get(extension
); 
1541         if (entry
) return (entry 
-> id
); 
1544     wxFileType 
*ft 
= (mime
.IsEmpty()) ? 
1545                    wxTheMimeTypesManager 
-> GetFileTypeFromExtension(extension
) : 
1546                    wxTheMimeTypesManager 
-> GetFileTypeFromMimeType(mime
); 
1548     wxIconLocation iconLoc
; 
1550     if ( ft 
&& ft
->GetIcon(&iconLoc
) ) 
1552         ic 
= wxIcon(iconLoc
); 
1560         m_HashTable
->Put(extension
, new wxFileIconEntry(newid
)); 
1565     tmpBmp
.CopyFromIcon(ic
); 
1566     wxImage img 
= tmpBmp
.ConvertToImage(); 
1568     int id 
= m_smallImageList
->GetImageCount(); 
1569     if (img
.GetWidth() == 16 && img
.GetHeight() == 16) 
1570         m_smallImageList
->Add(wxBitmap(img
)); 
1573         if (img
.GetWidth() != 32 || img
.GetHeight() != 32) 
1574             m_smallImageList
->Add(CreateAntialiasedBitmap(CutEmptyBorders(img
).Rescale(32, 32))); 
1576             m_smallImageList
->Add(CreateAntialiasedBitmap(img
)); 
1578     m_HashTable
->Put(extension
, new wxFileIconEntry(id
)); 
1581 #else // !wxUSE_MIMETYPE 
1583     if (extension 
== wxT("exe")) 
1587 #endif // wxUSE_MIMETYPE/!wxUSE_MIMETYPE 
1590 #endif // wxUSE_DIRDLG