1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/msw/treectrl.cpp 
   4 // Author:      Julian Smart 
   5 // Modified by: Vadim Zeitlin to be less MSW-specific on 10.10.98 
   8 // Copyright:   (c) Julian Smart 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  21     #pragma implementation "treectrl.h" 
  24 // For compilers that support precompilation, includes "wx.h". 
  25 #include "wx/wxprec.h" 
  33 #include "wx/msw/private.h" 
  35 // Set this to 1 to be _absolutely_ sure that repainting will work for all 
  36 // comctl32.dll versions 
  37 #define wxUSE_COMCTL32_SAFELY 0 
  39 // Mingw32 is a bit mental even though this is done in winundef 
  48 #if defined(__WIN95__) 
  52 #include "wx/dynarray.h" 
  53 #include "wx/imaglist.h" 
  54 #include "wx/settings.h" 
  55 #include "wx/msw/treectrl.h" 
  56 #include "wx/msw/dragimag.h" 
  58 #ifdef __GNUWIN32_OLD__ 
  59     #include "wx/msw/gnuwin32/extra.h" 
  62 #if defined(__WIN95__) && !(defined(__GNUWIN32_OLD__) && !defined(__CYGWIN10__)) 
  66 // Bug in headers, sometimes 
  68     #define TVIS_FOCUSED            0x0001 
  72     #define TV_FIRST                0x1100 
  75 #ifndef TVS_CHECKBOXES 
  76     #define TVS_CHECKBOXES          0x0100 
  79 #ifndef TVS_FULLROWSELECT 
  80     #define TVS_FULLROWSELECT       0x1000 
  83 // old headers might miss these messages (comctl32.dll 4.71+ only) 
  84 #ifndef TVM_SETBKCOLOR 
  85     #define TVM_SETBKCOLOR          (TV_FIRST + 29) 
  86     #define TVM_SETTEXTCOLOR        (TV_FIRST + 30) 
  89 // a macro to hide the ugliness of nested casts 
  90 #define HITEM(item)     (HTREEITEM)(WXHTREEITEM)(item) 
  92 // the native control doesn't support multiple selections under MSW and we 
  93 // have 2 ways to emulate them: either using TVS_CHECKBOXES style and let 
  94 // checkboxes be the selection status (checked == selected) or by really 
  95 // emulating everything, i.e. intercepting mouse and key events &c. The first 
  96 // approach is much easier but doesn't work with comctl32.dll < 4.71 and also 
  98 #define wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE 0 
 100 // ---------------------------------------------------------------------------- 
 102 // ---------------------------------------------------------------------------- 
 104 // wrapper for TreeView_HitTest 
 105 static HTREEITEM 
GetItemFromPoint(HWND hwndTV
, int x
, int y
) 
 111     return (HTREEITEM
)TreeView_HitTest(hwndTV
, &tvht
); 
 114 #if !wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE 
 116 // wrappers for TreeView_GetItem/TreeView_SetItem 
 117 static bool IsItemSelected(HWND hwndTV
, HTREEITEM hItem
) 
 121     tvi
.mask 
= TVIF_STATE 
| TVIF_HANDLE
; 
 122     tvi
.stateMask 
= TVIS_SELECTED
; 
 125     if ( !TreeView_GetItem(hwndTV
, &tvi
) ) 
 127         wxLogLastError(wxT("TreeView_GetItem")); 
 130     return (tvi
.state 
& TVIS_SELECTED
) != 0; 
 133 static void SelectItem(HWND hwndTV
, HTREEITEM hItem
, bool select 
= true) 
 136     tvi
.mask 
= TVIF_STATE 
| TVIF_HANDLE
; 
 137     tvi
.stateMask 
= TVIS_SELECTED
; 
 138     tvi
.state 
= select 
? TVIS_SELECTED 
: 0; 
 141     if ( TreeView_SetItem(hwndTV
, &tvi
) == -1 ) 
 143         wxLogLastError(wxT("TreeView_SetItem")); 
 147 static inline void UnselectItem(HWND hwndTV
, HTREEITEM htItem
) 
 149     SelectItem(hwndTV
, htItem
, false); 
 152 static inline void ToggleItemSelection(HWND hwndTV
, HTREEITEM htItem
) 
 154     SelectItem(hwndTV
, htItem
, !IsItemSelected(hwndTV
, htItem
)); 
 157 // helper function which selects all items in a range and, optionally, 
 158 // unselects all others 
 159 static void SelectRange(HWND hwndTV
, 
 162                         bool unselectOthers 
= true) 
 164     // find the first (or last) item and select it 
 166     HTREEITEM htItem 
= (HTREEITEM
)TreeView_GetRoot(hwndTV
); 
 167     while ( htItem 
&& cont 
) 
 169         if ( (htItem 
== htFirst
) || (htItem 
== htLast
) ) 
 171             if ( !IsItemSelected(hwndTV
, htItem
) ) 
 173                 SelectItem(hwndTV
, htItem
); 
 180             if ( unselectOthers 
&& IsItemSelected(hwndTV
, htItem
) ) 
 182                 UnselectItem(hwndTV
, htItem
); 
 186         htItem 
= (HTREEITEM
)TreeView_GetNextVisible(hwndTV
, htItem
); 
 189     // select the items in range 
 190     cont 
= htFirst 
!= htLast
; 
 191     while ( htItem 
&& cont 
) 
 193         if ( !IsItemSelected(hwndTV
, htItem
) ) 
 195             SelectItem(hwndTV
, htItem
); 
 198         cont 
= (htItem 
!= htFirst
) && (htItem 
!= htLast
); 
 200         htItem 
= (HTREEITEM
)TreeView_GetNextVisible(hwndTV
, htItem
); 
 204     if ( unselectOthers 
) 
 208             if ( IsItemSelected(hwndTV
, htItem
) ) 
 210                 UnselectItem(hwndTV
, htItem
); 
 213             htItem 
= (HTREEITEM
)TreeView_GetNextVisible(hwndTV
, htItem
); 
 217     // seems to be necessary - otherwise the just selected items don't always 
 218     // appear as selected 
 219     UpdateWindow(hwndTV
); 
 222 // helper function which tricks the standard control into changing the focused 
 223 // item without changing anything else (if someone knows why Microsoft doesn't 
 224 // allow to do it by just setting TVIS_FOCUSED flag, please tell me!) 
 225 static void SetFocus(HWND hwndTV
, HTREEITEM htItem
) 
 228     HTREEITEM htFocus 
= (HTREEITEM
)TreeView_GetSelection(hwndTV
); 
 233         if ( htItem 
!= htFocus 
) 
 235             // remember the selection state of the item 
 236             bool wasSelected 
= IsItemSelected(hwndTV
, htItem
); 
 238             if ( htFocus 
&& IsItemSelected(hwndTV
, htFocus
) ) 
 240                 // prevent the tree from unselecting the old focus which it 
 241                 // would do by default (TreeView_SelectItem unselects the 
 243                 TreeView_SelectItem(hwndTV
, 0); 
 244                 SelectItem(hwndTV
, htFocus
); 
 247             TreeView_SelectItem(hwndTV
, htItem
); 
 251                 // need to clear the selection which TreeView_SelectItem() gave 
 253                 UnselectItem(hwndTV
, htItem
); 
 255             //else: was selected, still selected - ok 
 257         //else: nothing to do, focus already there 
 263             bool wasFocusSelected 
= IsItemSelected(hwndTV
, htFocus
); 
 265             // just clear the focus 
 266             TreeView_SelectItem(hwndTV
, 0); 
 268             if ( wasFocusSelected 
) 
 270                 // restore the selection state 
 271                 SelectItem(hwndTV
, htFocus
); 
 274         //else: nothing to do, no focus already 
 278 #endif // wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE 
 280 // ---------------------------------------------------------------------------- 
 282 // ---------------------------------------------------------------------------- 
 284 // a convenient wrapper around TV_ITEM struct which adds a ctor 
 286 #pragma warning( disable : 4097 ) // inheriting from typedef 
 289 struct wxTreeViewItem 
: public TV_ITEM
 
 291     wxTreeViewItem(const wxTreeItemId
& item
,    // the item handle 
 292                    UINT mask_
,                  // fields which are valid 
 293                    UINT stateMask_ 
= 0)         // for TVIF_STATE only 
 297         // hItem member is always valid 
 298         mask 
= mask_ 
| TVIF_HANDLE
; 
 299         stateMask 
= stateMask_
; 
 304 // wxVirutalNode is used in place of a single root when 'hidden' root is 
 306 class wxVirtualNode 
: public wxTreeViewItem
 
 309     wxVirtualNode(wxTreeItemData 
*data
) 
 310         : wxTreeViewItem(TVI_ROOT
, 0) 
 320     wxTreeItemData 
*GetData() const { return m_data
; } 
 321     void SetData(wxTreeItemData 
*data
) { delete m_data
; m_data 
= data
; } 
 324     wxTreeItemData 
*m_data
; 
 326     DECLARE_NO_COPY_CLASS(wxVirtualNode
) 
 330 #pragma warning( default : 4097 ) 
 333 // a macro to get the virtual root, returns NULL if none 
 334 #define GET_VIRTUAL_ROOT() ((wxVirtualNode *)m_pVirtualRoot) 
 336 // returns true if the item is the virtual root 
 337 #define IS_VIRTUAL_ROOT(item) (HITEM(item) == TVI_ROOT) 
 339 // a class which encapsulates the tree traversal logic: it vists all (unless 
 340 // OnVisit() returns false) items under the given one 
 341 class wxTreeTraversal
 
 344     wxTreeTraversal(const wxTreeCtrl 
*tree
) 
 349     // do traverse the tree: visit all items (recursively by default) under the 
 350     // given one; return true if all items were traversed or false if the 
 351     // traversal was aborted because OnVisit returned false 
 352     bool DoTraverse(const wxTreeItemId
& root
, bool recursively 
= true); 
 354     // override this function to do whatever is needed for each item, return 
 355     // false to stop traversing 
 356     virtual bool OnVisit(const wxTreeItemId
& item
) = 0; 
 359     const wxTreeCtrl 
*GetTree() const { return m_tree
; } 
 362     bool Traverse(const wxTreeItemId
& root
, bool recursively
); 
 364     const wxTreeCtrl 
*m_tree
; 
 366     DECLARE_NO_COPY_CLASS(wxTreeTraversal
) 
 369 // internal class for getting the selected items 
 370 class TraverseSelections 
: public wxTreeTraversal
 
 373     TraverseSelections(const wxTreeCtrl 
*tree
, 
 374                        wxArrayTreeItemIds
& selections
) 
 375         : wxTreeTraversal(tree
), m_selections(selections
) 
 377             m_selections
.Empty(); 
 379             DoTraverse(tree
->GetRootItem()); 
 382     virtual bool OnVisit(const wxTreeItemId
& item
) 
 384         // can't visit a virtual node. 
 385         if ( (GetTree()->GetRootItem() == item
) && (GetTree()->GetWindowStyle() & wxTR_HIDE_ROOT
)) 
 390 #if wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE 
 391         if ( GetTree()->IsItemChecked(item
) ) 
 393         if ( ::IsItemSelected(GetHwndOf(GetTree()), HITEM(item
)) ) 
 396             m_selections
.Add(item
); 
 402     size_t GetCount() const { return m_selections
.GetCount(); } 
 405     wxArrayTreeItemIds
& m_selections
; 
 408 // internal class for counting tree items 
 409 class TraverseCounter 
: public wxTreeTraversal
 
 412     TraverseCounter(const wxTreeCtrl 
*tree
, 
 413                     const wxTreeItemId
& root
, 
 415         : wxTreeTraversal(tree
) 
 419             DoTraverse(root
, recursively
); 
 422     virtual bool OnVisit(const wxTreeItemId
& WXUNUSED(item
)) 
 429     size_t GetCount() const { return m_count
; } 
 435 // ---------------------------------------------------------------------------- 
 436 // This class is needed for support of different images: the Win32 common 
 437 // control natively supports only 2 images (the normal one and another for the 
 438 // selected state). We wish to provide support for 2 more of them for folder 
 439 // items (i.e. those which have children): for expanded state and for expanded 
 440 // selected state. For this we use this structure to store the additional items 
 443 // There is only one problem with this: when we retrieve the item's data, we 
 444 // don't know whether we get a pointer to wxTreeItemData or 
 445 // wxTreeItemIndirectData. So we always set the item id to an invalid value 
 446 // in this class and the code using the client data checks for it and retrieves 
 447 // the real client data in this case. 
 448 // ---------------------------------------------------------------------------- 
 450 class wxTreeItemIndirectData 
: public wxTreeItemData
 
 453     // ctor associates this data with the item and the real item data becomes 
 454     // available through our GetData() method 
 455     wxTreeItemIndirectData(wxTreeCtrl 
*tree
, const wxTreeItemId
& item
) 
 457         for ( size_t n 
= 0; n 
< WXSIZEOF(m_images
); n
++ ) 
 463         m_data 
= tree
->GetItemData(item
); 
 465         // and set ourselves as the new one 
 466         tree
->SetIndirectItemData(item
, this); 
 468         // we must have the invalid value for the item 
 472     // dtor deletes the associated data as well 
 473     virtual ~wxTreeItemIndirectData() { delete m_data
; } 
 476         // get the real data associated with the item 
 477     wxTreeItemData 
*GetData() const { return m_data
; } 
 479     void SetData(wxTreeItemData 
*data
) { m_data 
= data
; } 
 481         // do we have such image? 
 482     bool HasImage(wxTreeItemIcon which
) const { return m_images
[which
] != -1; } 
 484     int GetImage(wxTreeItemIcon which
) const { return m_images
[which
]; } 
 486     void SetImage(int image
, wxTreeItemIcon which
) { m_images
[which
] = image
; } 
 489     // all the images associated with the item 
 490     int m_images
[wxTreeItemIcon_Max
]; 
 492     // the real client data 
 493     wxTreeItemData 
*m_data
; 
 495     DECLARE_NO_COPY_CLASS(wxTreeItemIndirectData
) 
 498 // ---------------------------------------------------------------------------- 
 500 // ---------------------------------------------------------------------------- 
 502 IMPLEMENT_DYNAMIC_CLASS(wxTreeCtrl
, wxControl
) 
 504 // ---------------------------------------------------------------------------- 
 506 // ---------------------------------------------------------------------------- 
 508 // indices in gs_expandEvents table below 
 523 // handy table for sending events - it has to be initialized during run-time 
 524 // now so can't be const any more 
 525 static /* const */ wxEventType gs_expandEvents
[IDX_WHAT_MAX
][IDX_HOW_MAX
]; 
 528    but logically it's a const table with the following entries: 
 531     { wxEVT_COMMAND_TREE_ITEM_COLLAPSED, wxEVT_COMMAND_TREE_ITEM_COLLAPSING }, 
 532     { wxEVT_COMMAND_TREE_ITEM_EXPANDED,  wxEVT_COMMAND_TREE_ITEM_EXPANDING  } 
 536 // ============================================================================ 
 538 // ============================================================================ 
 540 // ---------------------------------------------------------------------------- 
 542 // ---------------------------------------------------------------------------- 
 544 bool wxTreeTraversal::DoTraverse(const wxTreeItemId
& root
, bool recursively
) 
 546     if ( !OnVisit(root
) ) 
 549     return Traverse(root
, recursively
); 
 552 bool wxTreeTraversal::Traverse(const wxTreeItemId
& root
, bool recursively
) 
 555     wxTreeItemId child 
= m_tree
->GetFirstChild(root
, cookie
); 
 556     while ( child
.IsOk() ) 
 558         // depth first traversal 
 559         if ( recursively 
&& !Traverse(child
, true) ) 
 562         if ( !OnVisit(child
) ) 
 565         child 
= m_tree
->GetNextChild(root
, cookie
); 
 571 // ---------------------------------------------------------------------------- 
 572 // construction and destruction 
 573 // ---------------------------------------------------------------------------- 
 575 void wxTreeCtrl::Init() 
 577     m_imageListNormal 
= NULL
; 
 578     m_imageListState 
= NULL
; 
 579     m_ownsImageListNormal 
= m_ownsImageListState 
= false; 
 581     m_hasAnyAttr 
= false; 
 584     m_pVirtualRoot 
= NULL
; 
 586     // initialize the global array of events now as it can't be done statically 
 587     // with the wxEVT_XXX values being allocated during run-time only 
 588     gs_expandEvents
[IDX_COLLAPSE
][IDX_DONE
] = wxEVT_COMMAND_TREE_ITEM_COLLAPSED
; 
 589     gs_expandEvents
[IDX_COLLAPSE
][IDX_DOING
] = wxEVT_COMMAND_TREE_ITEM_COLLAPSING
; 
 590     gs_expandEvents
[IDX_EXPAND
][IDX_DONE
] = wxEVT_COMMAND_TREE_ITEM_EXPANDED
; 
 591     gs_expandEvents
[IDX_EXPAND
][IDX_DOING
] = wxEVT_COMMAND_TREE_ITEM_EXPANDING
; 
 594 bool wxTreeCtrl::Create(wxWindow 
*parent
, 
 599                         const wxValidator
& validator
, 
 600                         const wxString
& name
) 
 604     if ( (style 
& wxBORDER_MASK
) == wxBORDER_DEFAULT 
) 
 605         style 
|= wxBORDER_SUNKEN
; 
 607     if ( !CreateControl(parent
, id
, pos
, size
, style
, validator
, name
) ) 
 611     DWORD wstyle 
= MSWGetStyle(m_windowStyle
, & exStyle
); 
 612     wstyle 
|= WS_TABSTOP 
| TVS_SHOWSELALWAYS
; 
 614     if ((m_windowStyle 
& wxTR_NO_LINES
) == 0) 
 615         wstyle 
|= TVS_HASLINES
; 
 616     if ( m_windowStyle 
& wxTR_HAS_BUTTONS 
) 
 617         wstyle 
|= TVS_HASBUTTONS
; 
 619     if ( m_windowStyle 
& wxTR_EDIT_LABELS 
) 
 620         wstyle 
|= TVS_EDITLABELS
; 
 622     if ( m_windowStyle 
& wxTR_LINES_AT_ROOT 
) 
 623         wstyle 
|= TVS_LINESATROOT
; 
 625     if ( m_windowStyle 
& wxTR_FULL_ROW_HIGHLIGHT 
) 
 627         if ( wxTheApp
->GetComCtl32Version() >= 471 ) 
 628             wstyle 
|= TVS_FULLROWSELECT
; 
 631     // using TVS_CHECKBOXES for emulation of a multiselection tree control 
 632     // doesn't work without the new enough headers 
 633 #if wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE && \ 
 634     !defined( __GNUWIN32_OLD__ ) && \ 
 635     !defined( __BORLANDC__ ) && \ 
 636     !defined( __WATCOMC__ ) && \ 
 637     (!defined(__VISUALC__) || (__VISUALC__ > 1010)) 
 639     // we emulate the multiple selection tree controls by using checkboxes: set 
 640     // up the image list we need for this if we do have multiple selections 
 641     if ( m_windowStyle 
& wxTR_MULTIPLE 
) 
 642         wstyle 
|= TVS_CHECKBOXES
; 
 643 #endif // wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE 
 645     // Create the tree control. 
 646     if ( !MSWCreateControl(WC_TREEVIEW
, wstyle
) ) 
 649 #if wxUSE_COMCTL32_SAFELY 
 650     wxWindow::SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW
)); 
 651     wxWindow::SetForegroundColour(wxWindow::GetParent()->GetForegroundColour()); 
 653     SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW
)); 
 654     SetForegroundColour(wxWindow::GetParent()->GetForegroundColour()); 
 656     // This works around a bug in the Windows tree control whereby for some versions 
 657     // of comctrl32, setting any colour actually draws the background in black. 
 658     // This will initialise the background to the system colour. 
 659     // THIS FIX NOW REVERTED since it caused problems on _other_ systems. 
 660     // Assume the user has an updated comctl32.dll. 
 661     ::SendMessage(GetHwnd(), TVM_SETBKCOLOR
, 0,-1); 
 662     wxWindow::SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW
)); 
 663     SetForegroundColour(wxWindow::GetParent()->GetForegroundColour()); 
 667     // VZ: this is some experimental code which may be used to get the 
 668     //     TVS_CHECKBOXES style functionality for comctl32.dll < 4.71. 
 669     //     AFAIK, the standard DLL does about the same thing anyhow. 
 671     if ( m_windowStyle 
& wxTR_MULTIPLE 
) 
 675         // create the DC compatible with the current screen 
 676         HDC hdcMem 
= CreateCompatibleDC(NULL
); 
 678         // create a mono bitmap of the standard size 
 679         int x 
= GetSystemMetrics(SM_CXMENUCHECK
); 
 680         int y 
= GetSystemMetrics(SM_CYMENUCHECK
); 
 681         wxImageList 
imagelistCheckboxes(x
, y
, false, 2); 
 682         HBITMAP hbmpCheck 
= CreateBitmap(x
, y
,   // bitmap size 
 683                                          1,      // # of color planes 
 684                                          1,      // # bits needed for one pixel 
 685                                          0);     // array containing colour data 
 686         SelectObject(hdcMem
, hbmpCheck
); 
 688         // then draw a check mark into it 
 689         RECT rect 
= { 0, 0, x
, y 
}; 
 690         if ( !::DrawFrameControl(hdcMem
, &rect
, 
 692                                  DFCS_BUTTONCHECK 
| DFCS_CHECKED
) ) 
 694             wxLogLastError(wxT("DrawFrameControl(check)")); 
 697         bmp
.SetHBITMAP((WXHBITMAP
)hbmpCheck
); 
 698         imagelistCheckboxes
.Add(bmp
); 
 700         if ( !::DrawFrameControl(hdcMem
, &rect
, 
 704             wxLogLastError(wxT("DrawFrameControl(uncheck)")); 
 707         bmp
.SetHBITMAP((WXHBITMAP
)hbmpCheck
); 
 708         imagelistCheckboxes
.Add(bmp
); 
 714         SetStateImageList(&imagelistCheckboxes
); 
 718     SetSize(pos
.x
, pos
.y
, size
.x
, size
.y
); 
 723 wxTreeCtrl::~wxTreeCtrl() 
 725     // delete any attributes 
 728         for ( wxNode 
*node 
= m_attrs
.Next(); node
; node 
= m_attrs
.Next() ) 
 730             delete (wxTreeItemAttr 
*)node
->GetData(); 
 733         // prevent TVN_DELETEITEM handler from deleting the attributes again! 
 734         m_hasAnyAttr 
= false; 
 739     // delete user data to prevent memory leaks 
 740     // also deletes hidden root node storage. 
 743     if (m_ownsImageListNormal
) delete m_imageListNormal
; 
 744     if (m_ownsImageListState
) delete m_imageListState
; 
 747 // ---------------------------------------------------------------------------- 
 749 // ---------------------------------------------------------------------------- 
 751 // simple wrappers which add error checking in debug mode 
 753 bool wxTreeCtrl::DoGetItem(wxTreeViewItem
* tvItem
) const 
 755     wxCHECK_MSG( tvItem
->hItem 
!= TVI_ROOT
, false, 
 756                  _T("can't retrieve virtual root item") ); 
 758     if ( !TreeView_GetItem(GetHwnd(), tvItem
) ) 
 760         wxLogLastError(wxT("TreeView_GetItem")); 
 768 void wxTreeCtrl::DoSetItem(wxTreeViewItem
* tvItem
) 
 770     if ( TreeView_SetItem(GetHwnd(), tvItem
) == -1 ) 
 772         wxLogLastError(wxT("TreeView_SetItem")); 
 776 size_t wxTreeCtrl::GetCount() const 
 778     return (size_t)TreeView_GetCount(GetHwnd()); 
 781 unsigned int wxTreeCtrl::GetIndent() const 
 783     return TreeView_GetIndent(GetHwnd()); 
 786 void wxTreeCtrl::SetIndent(unsigned int indent
) 
 788     TreeView_SetIndent(GetHwnd(), indent
); 
 791 wxImageList 
*wxTreeCtrl::GetImageList() const 
 793     return m_imageListNormal
; 
 796 wxImageList 
*wxTreeCtrl::GetStateImageList() const 
 798     return m_imageListNormal
; 
 801 void wxTreeCtrl::SetAnyImageList(wxImageList 
*imageList
, int which
) 
 804     TreeView_SetImageList(GetHwnd(), 
 805                           imageList 
? imageList
->GetHIMAGELIST() : 0, 
 809 void wxTreeCtrl::SetImageList(wxImageList 
*imageList
) 
 811     if (m_ownsImageListNormal
) 
 812         delete m_imageListNormal
; 
 814     SetAnyImageList(m_imageListNormal 
= imageList
, TVSIL_NORMAL
); 
 815     m_ownsImageListNormal 
= false; 
 818 void wxTreeCtrl::SetStateImageList(wxImageList 
*imageList
) 
 820     if (m_ownsImageListState
) delete m_imageListState
; 
 821     SetAnyImageList(m_imageListState 
= imageList
, TVSIL_STATE
); 
 822     m_ownsImageListState 
= false; 
 825 void wxTreeCtrl::AssignImageList(wxImageList 
*imageList
) 
 827     SetImageList(imageList
); 
 828     m_ownsImageListNormal 
= true; 
 831 void wxTreeCtrl::AssignStateImageList(wxImageList 
*imageList
) 
 833     SetStateImageList(imageList
); 
 834     m_ownsImageListState 
= true; 
 837 size_t wxTreeCtrl::GetChildrenCount(const wxTreeItemId
& item
, 
 838                                     bool recursively
) const 
 840     TraverseCounter 
counter(this, item
, recursively
); 
 842     return counter
.GetCount() - 1; 
 845 // ---------------------------------------------------------------------------- 
 847 // ---------------------------------------------------------------------------- 
 849 bool wxTreeCtrl::SetBackgroundColour(const wxColour 
&colour
) 
 851 #if !wxUSE_COMCTL32_SAFELY 
 852     if ( !wxWindowBase::SetBackgroundColour(colour
) ) 
 855     SendMessage(GetHwnd(), TVM_SETBKCOLOR
, 0, colour
.GetPixel()); 
 861 bool wxTreeCtrl::SetForegroundColour(const wxColour 
&colour
) 
 863 #if !wxUSE_COMCTL32_SAFELY 
 864     if ( !wxWindowBase::SetForegroundColour(colour
) ) 
 867     SendMessage(GetHwnd(), TVM_SETTEXTCOLOR
, 0, colour
.GetPixel()); 
 873 // ---------------------------------------------------------------------------- 
 875 // ---------------------------------------------------------------------------- 
 877 wxString 
wxTreeCtrl::GetItemText(const wxTreeItemId
& item
) const 
 879     wxChar buf
[512];  // the size is arbitrary... 
 881     wxTreeViewItem 
tvItem(item
, TVIF_TEXT
); 
 882     tvItem
.pszText 
= buf
; 
 883     tvItem
.cchTextMax 
= WXSIZEOF(buf
); 
 884     if ( !DoGetItem(&tvItem
) ) 
 886         // don't return some garbage which was on stack, but an empty string 
 890     return wxString(buf
); 
 893 void wxTreeCtrl::SetItemText(const wxTreeItemId
& item
, const wxString
& text
) 
 895     if ( IS_VIRTUAL_ROOT(item
) ) 
 898     wxTreeViewItem 
tvItem(item
, TVIF_TEXT
); 
 899     tvItem
.pszText 
= (wxChar 
*)text
.c_str();  // conversion is ok 
 902     // when setting the text of the item being edited, the text control should 
 903     // be updated to reflect the new text as well, otherwise calling 
 904     // SetItemText() in the OnBeginLabelEdit() handler doesn't have any effect 
 906     // don't use GetEditControl() here because m_textCtrl is not set yet 
 907     HWND hwndEdit 
= TreeView_GetEditControl(GetHwnd()); 
 910         if ( item 
== GetSelection() ) 
 912             ::SetWindowText(hwndEdit
, text
); 
 917 int wxTreeCtrl::DoGetItemImageFromData(const wxTreeItemId
& item
, 
 918                                        wxTreeItemIcon which
) const 
 920     wxTreeViewItem 
tvItem(item
, TVIF_PARAM
); 
 921     if ( !DoGetItem(&tvItem
) ) 
 926     return ((wxTreeItemIndirectData 
*)tvItem
.lParam
)->GetImage(which
); 
 929 void wxTreeCtrl::DoSetItemImageFromData(const wxTreeItemId
& item
, 
 931                                         wxTreeItemIcon which
) const 
 933     wxTreeViewItem 
tvItem(item
, TVIF_PARAM
); 
 934     if ( !DoGetItem(&tvItem
) ) 
 939     wxTreeItemIndirectData 
*data 
= ((wxTreeItemIndirectData 
*)tvItem
.lParam
); 
 941     data
->SetImage(image
, which
); 
 943     // make sure that we have selected images as well 
 944     if ( which 
== wxTreeItemIcon_Normal 
&& 
 945          !data
->HasImage(wxTreeItemIcon_Selected
) ) 
 947         data
->SetImage(image
, wxTreeItemIcon_Selected
); 
 950     if ( which 
== wxTreeItemIcon_Expanded 
&& 
 951          !data
->HasImage(wxTreeItemIcon_SelectedExpanded
) ) 
 953         data
->SetImage(image
, wxTreeItemIcon_SelectedExpanded
); 
 957 void wxTreeCtrl::DoSetItemImages(const wxTreeItemId
& item
, 
 961     wxTreeViewItem 
tvItem(item
, TVIF_IMAGE 
| TVIF_SELECTEDIMAGE
); 
 962     tvItem
.iSelectedImage 
= imageSel
; 
 963     tvItem
.iImage 
= image
; 
 967 int wxTreeCtrl::GetItemImage(const wxTreeItemId
& item
, 
 968                              wxTreeItemIcon which
) const 
 970     if ( (HITEM(item
) == TVI_ROOT
) && (m_windowStyle 
& wxTR_HIDE_ROOT
) ) 
 972         // TODO: Maybe a hidden root can still provide images? 
 976     if ( HasIndirectData(item
) ) 
 978         return DoGetItemImageFromData(item
, which
); 
 985             wxFAIL_MSG( wxT("unknown tree item image type") ); 
 987         case wxTreeItemIcon_Normal
: 
 991         case wxTreeItemIcon_Selected
: 
 992             mask 
= TVIF_SELECTEDIMAGE
; 
 995         case wxTreeItemIcon_Expanded
: 
 996         case wxTreeItemIcon_SelectedExpanded
: 
1000     wxTreeViewItem 
tvItem(item
, mask
); 
1003     return mask 
== TVIF_IMAGE 
? tvItem
.iImage 
: tvItem
.iSelectedImage
; 
1006 void wxTreeCtrl::SetItemImage(const wxTreeItemId
& item
, int image
, 
1007                               wxTreeItemIcon which
) 
1009     if ( IS_VIRTUAL_ROOT(item
) ) 
1011         // TODO: Maybe a hidden root can still store images? 
1015     int imageNormal
, imageSel
; 
1019             wxFAIL_MSG( wxT("unknown tree item image type") ); 
1021         case wxTreeItemIcon_Normal
: 
1022             imageNormal 
= image
; 
1023             imageSel 
= GetItemSelectedImage(item
); 
1026         case wxTreeItemIcon_Selected
: 
1027             imageNormal 
= GetItemImage(item
); 
1031         case wxTreeItemIcon_Expanded
: 
1032         case wxTreeItemIcon_SelectedExpanded
: 
1033             if ( !HasIndirectData(item
) ) 
1035                 // we need to get the old images first, because after we create 
1036                 // the wxTreeItemIndirectData GetItemXXXImage() will use it to 
1038                 imageNormal 
= GetItemImage(item
); 
1039                 imageSel 
= GetItemSelectedImage(item
); 
1041                 // if it doesn't have it yet, add it 
1042                 wxTreeItemIndirectData 
*data 
= new 
1043                     wxTreeItemIndirectData(this, item
); 
1045                 // copy the data to the new location 
1046                 data
->SetImage(imageNormal
, wxTreeItemIcon_Normal
); 
1047                 data
->SetImage(imageSel
, wxTreeItemIcon_Selected
); 
1050             DoSetItemImageFromData(item
, image
, which
); 
1052             // reset the normal/selected images because we won't use them any 
1053             // more - now they're stored inside the indirect data 
1055             imageSel 
= I_IMAGECALLBACK
; 
1059     // NB: at least in version 5.00.0518.9 of comctl32.dll we need to always 
1060     //     change both normal and selected image - otherwise the change simply 
1061     //     doesn't take place! 
1062     DoSetItemImages(item
, imageNormal
, imageSel
); 
1065 wxTreeItemData 
*wxTreeCtrl::GetItemData(const wxTreeItemId
& item
) const 
1067     wxTreeViewItem 
tvItem(item
, TVIF_PARAM
); 
1069     // Hidden root may have data. 
1070     if ( IS_VIRTUAL_ROOT(item
) ) 
1072         return GET_VIRTUAL_ROOT()->GetData(); 
1076     if ( !DoGetItem(&tvItem
) ) 
1081     wxTreeItemData 
*data 
= (wxTreeItemData 
*)tvItem
.lParam
; 
1082     if ( IsDataIndirect(data
) ) 
1084         data 
= ((wxTreeItemIndirectData 
*)data
)->GetData(); 
1090 void wxTreeCtrl::SetItemData(const wxTreeItemId
& item
, wxTreeItemData 
*data
) 
1092     if ( IS_VIRTUAL_ROOT(item
) ) 
1094         GET_VIRTUAL_ROOT()->SetData(data
); 
1097     // first, associate this piece of data with this item 
1103     wxTreeViewItem 
tvItem(item
, TVIF_PARAM
); 
1105     if ( HasIndirectData(item
) ) 
1107         if ( DoGetItem(&tvItem
) ) 
1109             ((wxTreeItemIndirectData 
*)tvItem
.lParam
)->SetData(data
); 
1113             wxFAIL_MSG( wxT("failed to change tree items data") ); 
1118         tvItem
.lParam 
= (LPARAM
)data
; 
1123 void wxTreeCtrl::SetIndirectItemData(const wxTreeItemId
& item
, 
1124                                      wxTreeItemIndirectData 
*data
) 
1126     // this should never happen because it's unnecessary and will probably lead 
1127     // to crash too because the code elsewhere supposes that the pointer the 
1128     // wxTreeItemIndirectData has is a real wxItemData and not 
1129     // wxTreeItemIndirectData as well 
1130     wxASSERT_MSG( !HasIndirectData(item
), wxT("setting indirect data twice?") ); 
1132     SetItemData(item
, data
); 
1135 bool wxTreeCtrl::HasIndirectData(const wxTreeItemId
& item
) const 
1137     // query the item itself 
1138     wxTreeViewItem 
tvItem(item
, TVIF_PARAM
); 
1139     if ( !DoGetItem(&tvItem
) ) 
1144     wxTreeItemData 
*data 
= (wxTreeItemData 
*)tvItem
.lParam
; 
1146     return data 
&& IsDataIndirect(data
); 
1149 void wxTreeCtrl::SetItemHasChildren(const wxTreeItemId
& item
, bool has
) 
1151     if ( IS_VIRTUAL_ROOT(item
) ) 
1154     wxTreeViewItem 
tvItem(item
, TVIF_CHILDREN
); 
1155     tvItem
.cChildren 
= (int)has
; 
1159 void wxTreeCtrl::SetItemBold(const wxTreeItemId
& item
, bool bold
) 
1161     if ( IS_VIRTUAL_ROOT(item
) ) 
1164     wxTreeViewItem 
tvItem(item
, TVIF_STATE
, TVIS_BOLD
); 
1165     tvItem
.state 
= bold 
? TVIS_BOLD 
: 0; 
1169 void wxTreeCtrl::SetItemDropHighlight(const wxTreeItemId
& item
, bool highlight
) 
1171     if ( IS_VIRTUAL_ROOT(item
) ) 
1174     wxTreeViewItem 
tvItem(item
, TVIF_STATE
, TVIS_DROPHILITED
); 
1175     tvItem
.state 
= highlight 
? TVIS_DROPHILITED 
: 0; 
1179 void wxTreeCtrl::RefreshItem(const wxTreeItemId
& item
) 
1181     if ( IS_VIRTUAL_ROOT(item
) ) 
1185     if ( GetBoundingRect(item
, rect
) ) 
1191 wxColour 
wxTreeCtrl::GetItemTextColour(const wxTreeItemId
& item
) const 
1193     long id 
= (long)(WXHTREEITEM
)item
; 
1194     wxTreeItemAttr 
*attr 
= (wxTreeItemAttr 
*)m_attrs
.Get(id
); 
1197         return wxNullColour
; 
1200     return attr
->GetTextColour(); 
1203 wxColour 
wxTreeCtrl::GetItemBackgroundColour(const wxTreeItemId
& item
) const 
1205     long id 
= (long)(WXHTREEITEM
)item
; 
1206     wxTreeItemAttr 
*attr 
= (wxTreeItemAttr 
*)m_attrs
.Get(id
); 
1209         return wxNullColour
; 
1212     return attr
->GetBackgroundColour(); 
1215 wxFont 
wxTreeCtrl::GetItemFont(const wxTreeItemId
& item
) const 
1217     long id 
= (long)(WXHTREEITEM
)item
; 
1218     wxTreeItemAttr 
*attr 
= (wxTreeItemAttr 
*)m_attrs
.Get(id
); 
1224     return attr
->GetFont(); 
1227 void wxTreeCtrl::SetItemTextColour(const wxTreeItemId
& item
, 
1228                                    const wxColour
& col
) 
1230     m_hasAnyAttr 
= true; 
1232     long id 
= (long)(WXHTREEITEM
)item
; 
1233     wxTreeItemAttr 
*attr 
= (wxTreeItemAttr 
*)m_attrs
.Get(id
); 
1236         attr 
= new wxTreeItemAttr
; 
1237         m_attrs
.Put(id
, (wxObject 
*)attr
); 
1240     attr
->SetTextColour(col
); 
1245 void wxTreeCtrl::SetItemBackgroundColour(const wxTreeItemId
& item
, 
1246                                          const wxColour
& col
) 
1248     m_hasAnyAttr 
= true; 
1250     long id 
= (long)(WXHTREEITEM
)item
; 
1251     wxTreeItemAttr 
*attr 
= (wxTreeItemAttr 
*)m_attrs
.Get(id
); 
1254         attr 
= new wxTreeItemAttr
; 
1255         m_attrs
.Put(id
, (wxObject 
*)attr
); 
1258     attr
->SetBackgroundColour(col
); 
1263 void wxTreeCtrl::SetItemFont(const wxTreeItemId
& item
, const wxFont
& font
) 
1265     m_hasAnyAttr 
= true; 
1267     long id 
= (long)(WXHTREEITEM
)item
; 
1268     wxTreeItemAttr 
*attr 
= (wxTreeItemAttr 
*)m_attrs
.Get(id
); 
1271         attr 
= new wxTreeItemAttr
; 
1272         m_attrs
.Put(id
, (wxObject 
*)attr
); 
1275     attr
->SetFont(font
); 
1280 // ---------------------------------------------------------------------------- 
1282 // ---------------------------------------------------------------------------- 
1284 bool wxTreeCtrl::IsVisible(const wxTreeItemId
& item
) const 
1286     if ( item 
== wxTreeItemId(TVI_ROOT
) ) 
1288         // virtual (hidden) root is never visible 
1292     // Bug in Gnu-Win32 headers, so don't use the macro TreeView_GetItemRect 
1295     // this ugliness comes directly from MSDN - it *is* the correct way to pass 
1296     // the HTREEITEM with TVM_GETITEMRECT 
1297     *(WXHTREEITEM 
*)&rect 
= (WXHTREEITEM
)item
; 
1299     // false means get item rect for the whole item, not only text 
1300     return SendMessage(GetHwnd(), TVM_GETITEMRECT
, false, (LPARAM
)&rect
) != 0; 
1303 bool wxTreeCtrl::ItemHasChildren(const wxTreeItemId
& item
) const 
1305     wxTreeViewItem 
tvItem(item
, TVIF_CHILDREN
); 
1308     return tvItem
.cChildren 
!= 0; 
1311 bool wxTreeCtrl::IsExpanded(const wxTreeItemId
& item
) const 
1313     // probably not a good idea to put it here 
1314     //wxASSERT( ItemHasChildren(item) ); 
1316     wxTreeViewItem 
tvItem(item
, TVIF_STATE
, TVIS_EXPANDED
); 
1319     return (tvItem
.state 
& TVIS_EXPANDED
) != 0; 
1322 bool wxTreeCtrl::IsSelected(const wxTreeItemId
& item
) const 
1324     wxTreeViewItem 
tvItem(item
, TVIF_STATE
, TVIS_SELECTED
); 
1327     return (tvItem
.state 
& TVIS_SELECTED
) != 0; 
1330 bool wxTreeCtrl::IsBold(const wxTreeItemId
& item
) const 
1332     wxTreeViewItem 
tvItem(item
, TVIF_STATE
, TVIS_BOLD
); 
1335     return (tvItem
.state 
& TVIS_BOLD
) != 0; 
1338 // ---------------------------------------------------------------------------- 
1340 // ---------------------------------------------------------------------------- 
1342 wxTreeItemId 
wxTreeCtrl::GetRootItem() const 
1344     // Root may be real (visible) or virtual (hidden). 
1345     if ( GET_VIRTUAL_ROOT() ) 
1348     return wxTreeItemId((WXHTREEITEM
) TreeView_GetRoot(GetHwnd())); 
1351 wxTreeItemId 
wxTreeCtrl::GetSelection() const 
1353     wxCHECK_MSG( !(m_windowStyle 
& wxTR_MULTIPLE
), (long)(WXHTREEITEM
)0, 
1354                  wxT("this only works with single selection controls") ); 
1356     return wxTreeItemId((WXHTREEITEM
) TreeView_GetSelection(GetHwnd())); 
1359 wxTreeItemId 
wxTreeCtrl::GetItemParent(const wxTreeItemId
& item
) const 
1363     if ( IS_VIRTUAL_ROOT(item
) ) 
1365         // no parent for the virtual root 
1370         hItem 
= TreeView_GetParent(GetHwnd(), HITEM(item
)); 
1371         if ( !hItem 
&& HasFlag(wxTR_HIDE_ROOT
) ) 
1373             // the top level items should have the virtual root as their parent 
1378     return wxTreeItemId((WXHTREEITEM
)hItem
); 
1381 wxTreeItemId 
wxTreeCtrl::GetFirstChild(const wxTreeItemId
& item
, 
1382                                        long& _cookie
) const 
1384     // remember the last child returned in 'cookie' 
1385     _cookie 
= (long)TreeView_GetChild(GetHwnd(), HITEM(item
)); 
1387     return wxTreeItemId((WXHTREEITEM
)_cookie
); 
1390 wxTreeItemId 
wxTreeCtrl::GetNextChild(const wxTreeItemId
& WXUNUSED(item
), 
1391                                       long& _cookie
) const 
1393     wxTreeItemId l 
= wxTreeItemId((WXHTREEITEM
)TreeView_GetNextSibling(GetHwnd(), 
1400 wxTreeItemId 
wxTreeCtrl::GetLastChild(const wxTreeItemId
& item
) const 
1402     // can this be done more efficiently? 
1405     wxTreeItemId childLast
, 
1406     child 
= GetFirstChild(item
, cookie
); 
1407     while ( child
.IsOk() ) 
1410         child 
= GetNextChild(item
, cookie
); 
1416 wxTreeItemId 
wxTreeCtrl::GetNextSibling(const wxTreeItemId
& item
) const 
1418     return wxTreeItemId((WXHTREEITEM
) TreeView_GetNextSibling(GetHwnd(), HITEM(item
))); 
1421 wxTreeItemId 
wxTreeCtrl::GetPrevSibling(const wxTreeItemId
& item
) const 
1423     return wxTreeItemId((WXHTREEITEM
) TreeView_GetPrevSibling(GetHwnd(), HITEM(item
))); 
1426 wxTreeItemId 
wxTreeCtrl::GetFirstVisibleItem() const 
1428     return wxTreeItemId((WXHTREEITEM
) TreeView_GetFirstVisible(GetHwnd())); 
1431 wxTreeItemId 
wxTreeCtrl::GetNextVisible(const wxTreeItemId
& item
) const 
1433     wxASSERT_MSG( IsVisible(item
), wxT("The item you call GetNextVisible() for must be visible itself!")); 
1435     return wxTreeItemId((WXHTREEITEM
) TreeView_GetNextVisible(GetHwnd(), HITEM(item
))); 
1438 wxTreeItemId 
wxTreeCtrl::GetPrevVisible(const wxTreeItemId
& item
) const 
1440     wxASSERT_MSG( IsVisible(item
), wxT("The item you call GetPrevVisible() for must be visible itself!")); 
1442     return wxTreeItemId((WXHTREEITEM
) TreeView_GetPrevVisible(GetHwnd(), HITEM(item
))); 
1445 // ---------------------------------------------------------------------------- 
1446 // multiple selections emulation 
1447 // ---------------------------------------------------------------------------- 
1449 bool wxTreeCtrl::IsItemChecked(const wxTreeItemId
& item
) const 
1451     // receive the desired information. 
1452     wxTreeViewItem 
tvItem(item
, TVIF_STATE
, TVIS_STATEIMAGEMASK
); 
1455     // state image indices are 1 based 
1456     return ((tvItem
.state 
>> 12) - 1) == 1; 
1459 void wxTreeCtrl::SetItemCheck(const wxTreeItemId
& item
, bool check
) 
1461     // receive the desired information. 
1462     wxTreeViewItem 
tvItem(item
, TVIF_STATE
, TVIS_STATEIMAGEMASK
); 
1466     // state images are one-based 
1467     tvItem
.state 
= (check 
? 2 : 1) << 12; 
1472 size_t wxTreeCtrl::GetSelections(wxArrayTreeItemIds
& selections
) const 
1474     TraverseSelections 
selector(this, selections
); 
1476     return selector
.GetCount(); 
1479 // ---------------------------------------------------------------------------- 
1481 // ---------------------------------------------------------------------------- 
1483 wxTreeItemId 
wxTreeCtrl::DoInsertItem(const wxTreeItemId
& parent
, 
1484                                       wxTreeItemId hInsertAfter
, 
1485                                       const wxString
& text
, 
1486                                       int image
, int selectedImage
, 
1487                                       wxTreeItemData 
*data
) 
1489     wxCHECK_MSG( parent
.IsOk() || !TreeView_GetRoot(GetHwnd()), 
1491                  _T("can't have more than one root in the tree") ); 
1493     TV_INSERTSTRUCT tvIns
; 
1494     tvIns
.hParent 
= HITEM(parent
); 
1495     tvIns
.hInsertAfter 
= HITEM(hInsertAfter
); 
1497     // this is how we insert the item as the first child: supply a NULL 
1499     if ( !tvIns
.hInsertAfter 
) 
1501         tvIns
.hInsertAfter 
= TVI_FIRST
; 
1505     if ( !text
.IsEmpty() ) 
1508         tvIns
.item
.pszText 
= (wxChar 
*)text
.c_str();  // cast is ok 
1512         tvIns
.item
.pszText 
= NULL
; 
1513         tvIns
.item
.cchTextMax 
= 0; 
1519         tvIns
.item
.iImage 
= image
; 
1521         if ( selectedImage 
== -1 ) 
1523             // take the same image for selected icon if not specified 
1524             selectedImage 
= image
; 
1528     if ( selectedImage 
!= -1 ) 
1530         mask 
|= TVIF_SELECTEDIMAGE
; 
1531         tvIns
.item
.iSelectedImage 
= selectedImage
; 
1537         tvIns
.item
.lParam 
= (LPARAM
)data
; 
1540     tvIns
.item
.mask 
= mask
; 
1542     HTREEITEM id 
= (HTREEITEM
) TreeView_InsertItem(GetHwnd(), &tvIns
); 
1545         wxLogLastError(wxT("TreeView_InsertItem")); 
1550         // associate the application tree item with Win32 tree item handle 
1551         data
->SetId((WXHTREEITEM
)id
); 
1554     return wxTreeItemId((WXHTREEITEM
)id
); 
1557 // for compatibility only 
1558 wxTreeItemId 
wxTreeCtrl::InsertItem(const wxTreeItemId
& parent
, 
1559                                     const wxString
& text
, 
1560                                     int image
, int selImage
, 
1563     return DoInsertItem(parent
, (WXHTREEITEM
)insertAfter
, text
, 
1564                         image
, selImage
, NULL
); 
1567 wxTreeItemId 
wxTreeCtrl::AddRoot(const wxString
& text
, 
1568                                  int image
, int selectedImage
, 
1569                                  wxTreeItemData 
*data
) 
1572     if ( m_windowStyle 
& wxTR_HIDE_ROOT 
) 
1574         // create a virtual root item, the parent for all the others 
1575         m_pVirtualRoot 
= new wxVirtualNode(data
); 
1580     return DoInsertItem(wxTreeItemId((long)(WXHTREEITEM
) 0), (long)(WXHTREEITEM
) 0, 
1581                         text
, image
, selectedImage
, data
); 
1584 wxTreeItemId 
wxTreeCtrl::PrependItem(const wxTreeItemId
& parent
, 
1585                                      const wxString
& text
, 
1586                                      int image
, int selectedImage
, 
1587                                      wxTreeItemData 
*data
) 
1589     return DoInsertItem(parent
, (WXHTREEITEM
) TVI_FIRST
, 
1590                         text
, image
, selectedImage
, data
); 
1593 wxTreeItemId 
wxTreeCtrl::InsertItem(const wxTreeItemId
& parent
, 
1594                                     const wxTreeItemId
& idPrevious
, 
1595                                     const wxString
& text
, 
1596                                     int image
, int selectedImage
, 
1597                                     wxTreeItemData 
*data
) 
1599     return DoInsertItem(parent
, idPrevious
, text
, image
, selectedImage
, data
); 
1602 wxTreeItemId 
wxTreeCtrl::InsertItem(const wxTreeItemId
& parent
, 
1604                                     const wxString
& text
, 
1605                                     int image
, int selectedImage
, 
1606                                     wxTreeItemData 
*data
) 
1608     // find the item from index 
1610     wxTreeItemId idPrev
, idCur 
= GetFirstChild(parent
, cookie
); 
1611     while ( index 
!= 0 && idCur
.IsOk() ) 
1616         idCur 
= GetNextChild(parent
, cookie
); 
1619     // assert, not check: if the index is invalid, we will append the item 
1621     wxASSERT_MSG( index 
== 0, _T("bad index in wxTreeCtrl::InsertItem") ); 
1623     return DoInsertItem(parent
, idPrev
, text
, image
, selectedImage
, data
); 
1626 wxTreeItemId 
wxTreeCtrl::AppendItem(const wxTreeItemId
& parent
, 
1627                                     const wxString
& text
, 
1628                                     int image
, int selectedImage
, 
1629                                     wxTreeItemData 
*data
) 
1631     return DoInsertItem(parent
, (WXHTREEITEM
) TVI_LAST
, 
1632                         text
, image
, selectedImage
, data
); 
1635 void wxTreeCtrl::Delete(const wxTreeItemId
& item
) 
1637     if ( !TreeView_DeleteItem(GetHwnd(), HITEM(item
)) ) 
1639         wxLogLastError(wxT("TreeView_DeleteItem")); 
1643 // delete all children (but don't delete the item itself) 
1644 void wxTreeCtrl::DeleteChildren(const wxTreeItemId
& item
) 
1648     wxArrayLong children
; 
1649     wxTreeItemId child 
= GetFirstChild(item
, cookie
); 
1650     while ( child
.IsOk() ) 
1652         children
.Add((long)(WXHTREEITEM
)child
); 
1654         child 
= GetNextChild(item
, cookie
); 
1657     size_t nCount 
= children
.Count(); 
1658     for ( size_t n 
= 0; n 
< nCount
; n
++ ) 
1660         if ( !TreeView_DeleteItem(GetHwnd(), (HTREEITEM
)children
[n
]) ) 
1662             wxLogLastError(wxT("TreeView_DeleteItem")); 
1667 void wxTreeCtrl::DeleteAllItems() 
1669     // delete the "virtual" root item. 
1670     if ( GET_VIRTUAL_ROOT() ) 
1672         delete GET_VIRTUAL_ROOT(); 
1673         m_pVirtualRoot 
= NULL
; 
1676     // and all the real items 
1678     if ( !TreeView_DeleteAllItems(GetHwnd()) ) 
1680         wxLogLastError(wxT("TreeView_DeleteAllItems")); 
1684 void wxTreeCtrl::DoExpand(const wxTreeItemId
& item
, int flag
) 
1686     wxASSERT_MSG( flag 
== TVE_COLLAPSE 
|| 
1687                   flag 
== (TVE_COLLAPSE 
| TVE_COLLAPSERESET
) || 
1688                   flag 
== TVE_EXPAND   
|| 
1690                   wxT("Unknown flag in wxTreeCtrl::DoExpand") ); 
1692     // A hidden root can be neither expanded nor collapsed. 
1693     wxCHECK_RET( !(m_windowStyle 
& wxTR_HIDE_ROOT
) || (HITEM(item
) != TVI_ROOT
), 
1694                  wxT("Can't expand/collapse hidden root node!") ) 
1696     // TreeView_Expand doesn't send TVN_ITEMEXPAND(ING) messages, so we must 
1697     // emulate them. This behaviour has changed slightly with comctl32.dll 
1698     // v 4.70 - now it does send them but only the first time. To maintain 
1699     // compatible behaviour and also in order to not have surprises with the 
1700     // future versions, don't rely on this and still do everything ourselves. 
1701     // To avoid that the messages be sent twice when the item is expanded for 
1702     // the first time we must clear TVIS_EXPANDEDONCE style manually. 
1704     wxTreeViewItem 
tvItem(item
, TVIF_STATE
, TVIS_EXPANDEDONCE
); 
1708     if ( TreeView_Expand(GetHwnd(), HITEM(item
), flag
) != 0 ) 
1710         wxTreeEvent 
event(wxEVT_NULL
, m_windowId
); 
1711         event
.m_item 
= item
; 
1712         event
.SetEventObject(this); 
1714         // note that the {EXPAND|COLLAPS}ING event is sent by TreeView_Expand() 
1716         event
.SetEventType(gs_expandEvents
[IsExpanded(item
) ? IDX_EXPAND
 
1720         (void)GetEventHandler()->ProcessEvent(event
); 
1722     //else: change didn't took place, so do nothing at all 
1725 void wxTreeCtrl::Expand(const wxTreeItemId
& item
) 
1727     DoExpand(item
, TVE_EXPAND
); 
1730 void wxTreeCtrl::Collapse(const wxTreeItemId
& item
) 
1732     DoExpand(item
, TVE_COLLAPSE
); 
1735 void wxTreeCtrl::CollapseAndReset(const wxTreeItemId
& item
) 
1737     DoExpand(item
, TVE_COLLAPSE 
| TVE_COLLAPSERESET
); 
1740 void wxTreeCtrl::Toggle(const wxTreeItemId
& item
) 
1742     DoExpand(item
, TVE_TOGGLE
); 
1745 void wxTreeCtrl::ExpandItem(const wxTreeItemId
& item
, int action
) 
1747     DoExpand(item
, action
); 
1750 void wxTreeCtrl::Unselect() 
1752     wxASSERT_MSG( !(m_windowStyle 
& wxTR_MULTIPLE
), 
1753                   wxT("doesn't make sense, may be you want UnselectAll()?") ); 
1755     // just remove the selection 
1756     SelectItem(wxTreeItemId((long) (WXHTREEITEM
) 0)); 
1759 void wxTreeCtrl::UnselectAll() 
1761     if ( m_windowStyle 
& wxTR_MULTIPLE 
) 
1763         wxArrayTreeItemIds selections
; 
1764         size_t count 
= GetSelections(selections
); 
1765         for ( size_t n 
= 0; n 
< count
; n
++ ) 
1767 #if wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE 
1768             SetItemCheck(selections
[n
], false); 
1769 #else // !wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE 
1770             ::UnselectItem(GetHwnd(), HITEM(selections
[n
])); 
1771 #endif // wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE/!wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE 
1776         // just remove the selection 
1781 void wxTreeCtrl::SelectItem(const wxTreeItemId
& item
) 
1783     if ( m_windowStyle 
& wxTR_MULTIPLE 
) 
1785 #if wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE 
1786         // selecting the item means checking it 
1788 #else // !wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE 
1789         ::SelectItem(GetHwnd(), HITEM(item
)); 
1790 #endif // wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE/!wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE 
1794         // inspite of the docs (MSDN Jan 99 edition), we don't seem to receive 
1795         // the notification from the control (i.e. TVN_SELCHANG{ED|ING}), so 
1796         // send them ourselves 
1798         wxTreeEvent 
event(wxEVT_NULL
, m_windowId
); 
1799         event
.m_item 
= item
; 
1800         event
.SetEventObject(this); 
1802         event
.SetEventType(wxEVT_COMMAND_TREE_SEL_CHANGING
); 
1803         if ( !GetEventHandler()->ProcessEvent(event
) || event
.IsAllowed() ) 
1805             if ( !TreeView_SelectItem(GetHwnd(), HITEM(item
)) ) 
1807                 wxLogLastError(wxT("TreeView_SelectItem")); 
1811                 event
.SetEventType(wxEVT_COMMAND_TREE_SEL_CHANGED
); 
1812                 (void)GetEventHandler()->ProcessEvent(event
); 
1815         //else: program vetoed the change 
1819 void wxTreeCtrl::EnsureVisible(const wxTreeItemId
& item
) 
1822     TreeView_EnsureVisible(GetHwnd(), HITEM(item
)); 
1825 void wxTreeCtrl::ScrollTo(const wxTreeItemId
& item
) 
1827     if ( !TreeView_SelectSetFirstVisible(GetHwnd(), HITEM(item
)) ) 
1829         wxLogLastError(wxT("TreeView_SelectSetFirstVisible")); 
1833 wxTextCtrl 
*wxTreeCtrl::GetEditControl() const 
1838 void wxTreeCtrl::DeleteTextCtrl() 
1842         // the HWND corresponding to this control is deleted by the tree 
1843         // control itself and we don't know when exactly this happens, so check 
1844         // if the window still exists before calling UnsubclassWin() 
1845         if ( !::IsWindow(GetHwndOf(m_textCtrl
)) ) 
1847             m_textCtrl
->SetHWND(0); 
1850         m_textCtrl
->UnsubclassWin(); 
1851         m_textCtrl
->SetHWND(0); 
1857 wxTextCtrl
* wxTreeCtrl::EditLabel(const wxTreeItemId
& item
, 
1858                                   wxClassInfo
* textControlClass
) 
1860     wxASSERT( textControlClass
->IsKindOf(CLASSINFO(wxTextCtrl
)) ); 
1864     m_textCtrl 
= (wxTextCtrl 
*)textControlClass
->CreateObject(); 
1865     HWND hWnd 
= (HWND
) TreeView_EditLabel(GetHwnd(), HITEM(item
)); 
1867     // this is not an error - the TVN_BEGINLABELEDIT handler might have 
1876     // textctrl is subclassed in MSWOnNotify 
1880 // End label editing, optionally cancelling the edit 
1881 void wxTreeCtrl::EndEditLabel(const wxTreeItemId
& WXUNUSED(item
), bool discardChanges
) 
1883     TreeView_EndEditLabelNow(GetHwnd(), discardChanges
); 
1888 wxTreeItemId 
wxTreeCtrl::HitTest(const wxPoint
& point
, int& flags
) 
1890     TV_HITTESTINFO hitTestInfo
; 
1891     hitTestInfo
.pt
.x 
= (int)point
.x
; 
1892     hitTestInfo
.pt
.y 
= (int)point
.y
; 
1894     TreeView_HitTest(GetHwnd(), &hitTestInfo
); 
1899     #define TRANSLATE_FLAG(flag) if ( hitTestInfo.flags & TVHT_##flag ) \ 
1900                                     flags |= wxTREE_HITTEST_##flag 
1902     TRANSLATE_FLAG(ABOVE
); 
1903     TRANSLATE_FLAG(BELOW
); 
1904     TRANSLATE_FLAG(NOWHERE
); 
1905     TRANSLATE_FLAG(ONITEMBUTTON
); 
1906     TRANSLATE_FLAG(ONITEMICON
); 
1907     TRANSLATE_FLAG(ONITEMINDENT
); 
1908     TRANSLATE_FLAG(ONITEMLABEL
); 
1909     TRANSLATE_FLAG(ONITEMRIGHT
); 
1910     TRANSLATE_FLAG(ONITEMSTATEICON
); 
1911     TRANSLATE_FLAG(TOLEFT
); 
1912     TRANSLATE_FLAG(TORIGHT
); 
1914     #undef TRANSLATE_FLAG 
1916     return wxTreeItemId((WXHTREEITEM
) hitTestInfo
.hItem
); 
1919 bool wxTreeCtrl::GetBoundingRect(const wxTreeItemId
& item
, 
1921                                  bool textOnly
) const 
1924     if ( TreeView_GetItemRect(GetHwnd(), HITEM(item
), 
1927         rect 
= wxRect(wxPoint(rc
.left
, rc
.top
), wxPoint(rc
.right
, rc
.bottom
)); 
1933         // couldn't retrieve rect: for example, item isn't visible 
1938 // ---------------------------------------------------------------------------- 
1940 // ---------------------------------------------------------------------------- 
1942 // this is just a tiny namespace which is friend to wxTreeCtrl and so can use 
1943 // functions such as IsDataIndirect() 
1944 class wxTreeSortHelper
 
1947     static int CALLBACK 
Compare(LPARAM data1
, LPARAM data2
, LPARAM tree
); 
1950     static wxTreeItemId 
GetIdFromData(wxTreeCtrl 
*tree
, LPARAM item
) 
1952         wxTreeItemData 
*data 
= (wxTreeItemData 
*)item
; 
1953         if ( tree
->IsDataIndirect(data
) ) 
1955             data 
= ((wxTreeItemIndirectData 
*)data
)->GetData(); 
1958         return data
->GetId(); 
1962 int CALLBACK 
wxTreeSortHelper::Compare(LPARAM pItem1
, 
1966     wxCHECK_MSG( pItem1 
&& pItem2
, 0, 
1967                  wxT("sorting tree without data doesn't make sense") ); 
1969     wxTreeCtrl 
*tree 
= (wxTreeCtrl 
*)htree
; 
1971     return tree
->OnCompareItems(GetIdFromData(tree
, pItem1
), 
1972                                 GetIdFromData(tree
, pItem2
)); 
1975 int wxTreeCtrl::OnCompareItems(const wxTreeItemId
& item1
, 
1976                                const wxTreeItemId
& item2
) 
1978     return wxStrcmp(GetItemText(item1
), GetItemText(item2
)); 
1981 void wxTreeCtrl::SortChildren(const wxTreeItemId
& item
) 
1983     // rely on the fact that TreeView_SortChildren does the same thing as our 
1984     // default behaviour, i.e. sorts items alphabetically and so call it 
1985     // directly if we're not in derived class (much more efficient!) 
1986     if ( GetClassInfo() == CLASSINFO(wxTreeCtrl
) ) 
1988         TreeView_SortChildren(GetHwnd(), HITEM(item
), 0); 
1993         tvSort
.hParent 
= HITEM(item
); 
1994         tvSort
.lpfnCompare 
= wxTreeSortHelper::Compare
; 
1995         tvSort
.lParam 
= (LPARAM
)this; 
1996         TreeView_SortChildrenCB(GetHwnd(), &tvSort
, 0 /* reserved */); 
2000 // ---------------------------------------------------------------------------- 
2002 // ---------------------------------------------------------------------------- 
2004 bool wxTreeCtrl::MSWCommand(WXUINT cmd
, WXWORD id
) 
2006     if ( cmd 
== EN_UPDATE 
) 
2008         wxCommandEvent 
event(wxEVT_COMMAND_TEXT_UPDATED
, id
); 
2009         event
.SetEventObject( this ); 
2010         ProcessCommand(event
); 
2012     else if ( cmd 
== EN_KILLFOCUS 
) 
2014         wxCommandEvent 
event(wxEVT_KILL_FOCUS
, id
); 
2015         event
.SetEventObject( this ); 
2016         ProcessCommand(event
); 
2024     // command processed 
2028 // we hook into WndProc to process WM_MOUSEMOVE/WM_BUTTONUP messages - as we 
2029 // only do it during dragging, minimize wxWin overhead (this is important for 
2030 // WM_MOUSEMOVE as they're a lot of them) by catching Windows messages directly 
2031 // instead of passing by wxWin events 
2032 long wxTreeCtrl::MSWWindowProc(WXUINT nMsg
, WXWPARAM wParam
, WXLPARAM lParam
) 
2034     bool processed 
= false; 
2036     bool isMultiple 
= (GetWindowStyle() & wxTR_MULTIPLE
) != 0; 
2038     if ( (nMsg 
>= WM_MOUSEFIRST
) && (nMsg 
<= WM_MOUSELAST
) ) 
2040         // we only process mouse messages here and these parameters have the 
2041         // same meaning for all of them 
2042         int x 
= GET_X_LPARAM(lParam
), 
2043             y 
= GET_Y_LPARAM(lParam
); 
2044         HTREEITEM htItem 
= GetItemFromPoint(GetHwnd(), x
, y
); 
2048             case WM_RBUTTONDOWN
: 
2049                 // if the item we are about to right click on 
2050                 // is not already select, remove the entire 
2051                 // previous selection 
2052                 if (!::IsItemSelected(GetHwnd(), htItem
)) 
2057                 // select item and set the focus to the 
2058                 // newly selected item 
2059                 ::SelectItem(GetHwnd(), htItem
); 
2060                 ::SetFocus(GetHwnd(), htItem
); 
2063 #if !wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE 
2064             case WM_LBUTTONDOWN
: 
2065                 if ( htItem 
&& isMultiple 
) 
2067                     if ( wParam 
& MK_CONTROL 
) 
2071                         // toggle selected state 
2072                         ToggleItemSelection(GetHwnd(), htItem
); 
2074                         ::SetFocus(GetHwnd(), htItem
); 
2076                         // reset on any click without Shift 
2081                     else if ( wParam 
& MK_SHIFT 
) 
2083                         // this selects all items between the starting one and 
2086                         if ( !m_htSelStart 
) 
2088                             // take the focused item 
2089                             m_htSelStart 
= (WXHTREEITEM
) 
2090                                 TreeView_GetSelection(GetHwnd()); 
2093                         SelectRange(GetHwnd(), HITEM(m_htSelStart
), htItem
, 
2094                                     !(wParam 
& MK_CONTROL
)); 
2096                         ::SetFocus(GetHwnd(), htItem
); 
2100                     else // normal click 
2102                         // avoid doing anything if we click on the only 
2103                         // currently selected item 
2104                         wxArrayTreeItemIds selections
; 
2105                         size_t count 
= GetSelections(selections
); 
2108                                     HITEM(selections
[0]) != htItem 
) 
2110                             // clear the previously selected items 
2113                             // prevent the click from starting in-place editing 
2114                             // which should only happen if we click on the 
2115                             // already selected item (and nothing else is 
2117                             TreeView_SelectItem(GetHwnd(), 0); 
2120                         // reset on any click without Shift 
2125 #endif // wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE 
2130                     m_dragImage
->Move(wxPoint(x
, y
)); 
2133                         // highlight the item as target (hiding drag image is 
2134                         // necessary - otherwise the display will be corrupted) 
2135                         m_dragImage
->Hide(); 
2136                         TreeView_SelectDropTarget(GetHwnd(), htItem
); 
2137                         m_dragImage
->Show(); 
2146                     m_dragImage
->EndDrag(); 
2150                     // generate the drag end event 
2151                     wxTreeEvent 
event(wxEVT_COMMAND_TREE_END_DRAG
, m_windowId
); 
2153                     event
.m_item 
= (WXHTREEITEM
)htItem
; 
2154                     event
.m_pointDrag 
= wxPoint(x
, y
); 
2155                     event
.SetEventObject(this); 
2157                     (void)GetEventHandler()->ProcessEvent(event
); 
2159                     // if we don't do it, the tree seems to think that 2 items 
2160                     // are selected simultaneously which is quite weird 
2161                     TreeView_SelectDropTarget(GetHwnd(), 0); 
2166 #if !wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE 
2167     else if ( (nMsg 
== WM_SETFOCUS 
|| nMsg 
== WM_KILLFOCUS
) && isMultiple 
) 
2169         // the tree control greys out the selected item when it loses focus and 
2170         // paints it as selected again when it regains it, but it won't do it 
2171         // for the other items itself - help it 
2172         wxArrayTreeItemIds selections
; 
2173         size_t count 
= GetSelections(selections
); 
2175         for ( size_t n 
= 0; n 
< count
; n
++ ) 
2177             // TreeView_GetItemRect() will return false if item is not visible, 
2178             // which may happen perfectly well 
2179             if ( TreeView_GetItemRect(GetHwnd(), HITEM(selections
[n
]), 
2182                 ::InvalidateRect(GetHwnd(), &rect
, false); 
2186     else if ( nMsg 
== WM_KEYDOWN 
&& isMultiple 
) 
2188         bool bCtrl 
= wxIsCtrlDown(), 
2189              bShift 
= wxIsShiftDown(); 
2191         // we handle.arrows and space, but not page up/down and home/end: the 
2192         // latter should be easy, but not the former 
2194         HTREEITEM htSel 
= (HTREEITEM
)TreeView_GetSelection(GetHwnd()); 
2195         if ( !m_htSelStart 
) 
2197             m_htSelStart 
= (WXHTREEITEM
)htSel
; 
2200         if ( wParam 
== VK_SPACE 
) 
2204                 ToggleItemSelection(GetHwnd(), htSel
); 
2210                 ::SelectItem(GetHwnd(), htSel
); 
2215         else if ( wParam 
== VK_UP 
|| wParam 
== VK_DOWN 
) 
2217             if ( !bCtrl 
&& !bShift 
) 
2219                 // no modifiers, just clear selection and then let the default 
2220                 // processing to take place 
2225                 (void)wxControl::MSWWindowProc(nMsg
, wParam
, lParam
); 
2227                 HTREEITEM htNext 
= (HTREEITEM
)(wParam 
== VK_UP
 
2228                                     ? TreeView_GetPrevVisible(GetHwnd(), htSel
) 
2229                                     : TreeView_GetNextVisible(GetHwnd(), htSel
)); 
2233                     // at the top/bottom 
2239                     SelectRange(GetHwnd(), HITEM(m_htSelStart
), htNext
); 
2243                     // without changing selection 
2244                     ::SetFocus(GetHwnd(), htNext
); 
2251 #endif // !wxUSE_CHECKBOXES_IN_MULTI_SEL_TREE 
2252     else if ( nMsg 
== WM_CHAR 
) 
2254         // don't let the control process Space and Return keys because it 
2255         // doesn't do anything useful with them anyhow but always beeps 
2256         // annoyingly when it receives them and there is no way to turn it off 
2257         // simply if you just process TREEITEM_ACTIVATED event to which Space 
2258         // and Enter presses are mapped in your code 
2259         if ( wParam 
== VK_SPACE 
|| wParam 
== VK_RETURN 
) 
2266         rc 
= wxControl::MSWWindowProc(nMsg
, wParam
, lParam
); 
2271 // process WM_NOTIFY Windows message 
2272 bool wxTreeCtrl::MSWOnNotify(int idCtrl
, WXLPARAM lParam
, WXLPARAM 
*result
) 
2274     wxTreeEvent 
event(wxEVT_NULL
, m_windowId
); 
2275     wxEventType eventType 
= wxEVT_NULL
; 
2276     NMHDR 
*hdr 
= (NMHDR 
*)lParam
; 
2278     switch ( hdr
->code 
) 
2281             eventType 
= wxEVT_COMMAND_TREE_BEGIN_DRAG
; 
2284         case TVN_BEGINRDRAG
: 
2286                 if ( eventType 
== wxEVT_NULL 
) 
2287                     eventType 
= wxEVT_COMMAND_TREE_BEGIN_RDRAG
; 
2288                 //else: left drag, already set above 
2290                 NM_TREEVIEW 
*tv 
= (NM_TREEVIEW 
*)lParam
; 
2292                 event
.m_item 
= (WXHTREEITEM
) tv
->itemNew
.hItem
; 
2293                 event
.m_pointDrag 
= wxPoint(tv
->ptDrag
.x
, tv
->ptDrag
.y
); 
2295                 // don't allow dragging by default: the user code must 
2296                 // explicitly say that it wants to allow it to avoid breaking 
2302         case TVN_BEGINLABELEDIT
: 
2304                 eventType 
= wxEVT_COMMAND_TREE_BEGIN_LABEL_EDIT
; 
2305                 TV_DISPINFO 
*info 
= (TV_DISPINFO 
*)lParam
; 
2307                 event
.m_item 
= (WXHTREEITEM
) info
->item
.hItem
; 
2308                 event
.m_label 
= info
->item
.pszText
; 
2309                 event
.m_editCancelled 
= false; 
2313         case TVN_DELETEITEM
: 
2315                 eventType 
= wxEVT_COMMAND_TREE_DELETE_ITEM
; 
2316                 NM_TREEVIEW 
*tv 
= (NM_TREEVIEW 
*)lParam
; 
2318                 event
.m_item 
= (WXHTREEITEM
)tv
->itemOld
.hItem
; 
2322                     delete (wxTreeItemAttr 
*)m_attrs
. 
2323                                 Delete((long)tv
->itemOld
.hItem
); 
2328         case TVN_ENDLABELEDIT
: 
2330                 eventType 
= wxEVT_COMMAND_TREE_END_LABEL_EDIT
; 
2331                 TV_DISPINFO 
*info 
= (TV_DISPINFO 
*)lParam
; 
2333                 event
.m_item 
= (WXHTREEITEM
)info
->item
.hItem
; 
2334                 event
.m_label 
= info
->item
.pszText
; 
2335                 if (info
->item
.pszText 
== NULL
) 
2337                     event
.m_editCancelled 
= true; 
2341                     event
.m_editCancelled 
= false; 
2346         case TVN_GETDISPINFO
: 
2347             eventType 
= wxEVT_COMMAND_TREE_GET_INFO
; 
2350         case TVN_SETDISPINFO
: 
2352                 if ( eventType 
== wxEVT_NULL 
) 
2353                     eventType 
= wxEVT_COMMAND_TREE_SET_INFO
; 
2354                 //else: get, already set above 
2356                 TV_DISPINFO 
*info 
= (TV_DISPINFO 
*)lParam
; 
2358                 event
.m_item 
= (WXHTREEITEM
) info
->item
.hItem
; 
2362         case TVN_ITEMEXPANDING
: 
2363         case TVN_ITEMEXPANDED
: 
2365                 NM_TREEVIEW
* tv 
= (NM_TREEVIEW
*)lParam
; 
2368                 switch ( tv
->action 
) 
2371                         wxLogDebug(wxT("unexpected code %d in TVN_ITEMEXPAND message"), tv
->action
); 
2379                         what 
= IDX_COLLAPSE
; 
2383                 int how 
= hdr
->code 
== TVN_ITEMEXPANDING 
? IDX_DOING
 
2386                 eventType 
= gs_expandEvents
[what
][how
]; 
2388                 event
.m_item 
= (WXHTREEITEM
) tv
->itemNew
.hItem
; 
2394                 eventType 
= wxEVT_COMMAND_TREE_KEY_DOWN
; 
2395                 TV_KEYDOWN 
*info 
= (TV_KEYDOWN 
*)lParam
; 
2397                 // fabricate the lParam and wParam parameters sufficiently 
2398                 // similar to the ones from a "real" WM_KEYDOWN so that 
2399                 // CreateKeyEvent() works correctly 
2401 //                    (::GetKeyState(VK_MENU) & 0x100 ? KF_ALTDOWN : 0) << 16; 
2402                      // Returns different negative values on WinME and WinNT, 
2403                      // so simply test for negative value. 
2404                      (::GetKeyState(VK_MENU
) < 0 ? KF_ALTDOWN 
: 0) << 16; 
2406                 WXWPARAM wParam 
= info
->wVKey
; 
2408                 int keyCode 
= wxCharCodeMSWToWX(info
->wVKey
); 
2411                     // wxCharCodeMSWToWX() returns 0 to indicate that this is a 
2416                 event
.m_evtKey 
= CreateKeyEvent(wxEVT_KEY_DOWN
, 
2421                 // a separate event for Space/Return 
2422                 if ( !wxIsCtrlDown() && !wxIsShiftDown() && 
2423                      ((info
->wVKey 
== VK_SPACE
) || (info
->wVKey 
== VK_RETURN
)) ) 
2425                     wxTreeEvent 
event2(wxEVT_COMMAND_TREE_ITEM_ACTIVATED
, 
2427                     event2
.SetEventObject(this); 
2428                     if ( !(GetWindowStyle() & wxTR_MULTIPLE
) ) 
2430                         event2
.m_item 
= GetSelection(); 
2432                     //else: don't know how to get it 
2434                     (void)GetEventHandler()->ProcessEvent(event2
); 
2439         // NB: MSLU is broken and sends TVN_SELCHANGEDA instead of  
2440         //     TVN_SELCHANGEDW in Unicode mode under Win98. Therefore 
2441         //     we have to handle both messages: 
2442         case TVN_SELCHANGEDA
: 
2443         case TVN_SELCHANGEDW
: 
2444             eventType 
= wxEVT_COMMAND_TREE_SEL_CHANGED
; 
2447         case TVN_SELCHANGINGA
: 
2448         case TVN_SELCHANGINGW
: 
2450                 if ( eventType 
== wxEVT_NULL 
) 
2451                     eventType 
= wxEVT_COMMAND_TREE_SEL_CHANGING
; 
2452                 //else: already set above 
2454                 if (hdr
->code 
== TVN_SELCHANGINGW 
||  
2455                     hdr
->code 
== TVN_SELCHANGEDW
) 
2457                     NM_TREEVIEWW
* tv 
= (NM_TREEVIEWW 
*)lParam
; 
2458                     event
.m_item 
= (WXHTREEITEM
) tv
->itemNew
.hItem
; 
2459                     event
.m_itemOld 
= (WXHTREEITEM
) tv
->itemOld
.hItem
; 
2463                     NM_TREEVIEWA
* tv 
= (NM_TREEVIEWA 
*)lParam
; 
2464                     event
.m_item 
= (WXHTREEITEM
) tv
->itemNew
.hItem
; 
2465                     event
.m_itemOld 
= (WXHTREEITEM
) tv
->itemOld
.hItem
; 
2470 #if defined(_WIN32_IE) && _WIN32_IE >= 0x300 && !wxUSE_COMCTL32_SAFELY && !( defined(__GNUWIN32__) && !wxCHECK_W32API_VERSION( 1, 0 ) ) 
2473                 LPNMTVCUSTOMDRAW lptvcd 
= (LPNMTVCUSTOMDRAW
)lParam
; 
2474                 NMCUSTOMDRAW
& nmcd 
= lptvcd
->nmcd
; 
2475                 switch ( nmcd
.dwDrawStage 
) 
2478                         // if we've got any items with non standard attributes, 
2479                         // notify us before painting each item 
2480                         *result 
= m_hasAnyAttr 
? CDRF_NOTIFYITEMDRAW
 
2484                     case CDDS_ITEMPREPAINT
: 
2486                             wxTreeItemAttr 
*attr 
= 
2487                                 (wxTreeItemAttr 
*)m_attrs
.Get(nmcd
.dwItemSpec
); 
2491                                 // nothing to do for this item 
2492                                 *result 
= CDRF_DODEFAULT
; 
2497                             if ( attr
->HasFont() ) 
2499                                 hFont 
= GetHfontOf(attr
->GetFont()); 
2507                             if ( attr
->HasTextColour() ) 
2509                                 colText 
= attr
->GetTextColour(); 
2513                                 colText 
= GetForegroundColour(); 
2516                             // selection colours should override ours 
2517                             if ( nmcd
.uItemState 
& CDIS_SELECTED 
) 
2520                                     ::GetSysColor(COLOR_HIGHLIGHT
); 
2522                                     ::GetSysColor(COLOR_HIGHLIGHTTEXT
); 
2527                                 if ( attr
->HasBackgroundColour() ) 
2529                                     colBack 
= attr
->GetBackgroundColour(); 
2533                                     colBack 
= GetBackgroundColour(); 
2536                                 lptvcd
->clrText 
= wxColourToRGB(colText
); 
2537                                 lptvcd
->clrTextBk 
= wxColourToRGB(colBack
); 
2540                             // note that if we wanted to set colours for 
2541                             // individual columns (subitems), we would have 
2542                             // returned CDRF_NOTIFYSUBITEMREDRAW from here 
2545                                 ::SelectObject(nmcd
.hdc
, hFont
); 
2547                                 *result 
= CDRF_NEWFONT
; 
2551                                 *result 
= CDRF_DODEFAULT
; 
2557                         *result 
= CDRF_DODEFAULT
; 
2561             // we always process it 
2563 #endif // _WIN32_IE >= 0x300 
2568                 TV_HITTESTINFO tvhti
; 
2569                 ::GetCursorPos(&tvhti
.pt
); 
2570                 ::ScreenToClient(GetHwnd(), &tvhti
.pt
); 
2571                 if ( TreeView_HitTest(GetHwnd(), &tvhti
) ) 
2573                     if ( tvhti
.flags 
& TVHT_ONITEM 
) 
2575                         event
.m_item 
= (WXHTREEITEM
) tvhti
.hItem
; 
2576                         eventType 
= (int)hdr
->code 
== NM_DBLCLK
 
2577                                     ? wxEVT_COMMAND_TREE_ITEM_ACTIVATED
 
2578                                     : wxEVT_COMMAND_TREE_ITEM_RIGHT_CLICK
; 
2580                         event
.m_pointDrag
.x 
= tvhti
.pt
.x
; 
2581                         event
.m_pointDrag
.y 
= tvhti
.pt
.y
; 
2590             return wxControl::MSWOnNotify(idCtrl
, lParam
, result
); 
2593     event
.SetEventObject(this); 
2594     event
.SetEventType(eventType
); 
2596     bool processed 
= GetEventHandler()->ProcessEvent(event
); 
2599     switch ( hdr
->code 
) 
2602             // we translate NM_DBLCLK into ACTIVATED event, so don't interpret 
2603             // the return code of this event handler as the return value for 
2604             // NM_DBLCLK - otherwise, double clicking the item to toggle its 
2605             // expanded status would never work 
2610         case TVN_BEGINRDRAG
: 
2611             if ( event
.IsAllowed() ) 
2613                 // normally this is impossible because the m_dragImage is 
2614                 // deleted once the drag operation is over 
2615                 wxASSERT_MSG( !m_dragImage
, _T("starting to drag once again?") ); 
2617                 m_dragImage 
= new wxDragImage(*this, event
.m_item
); 
2618                 m_dragImage
->BeginDrag(wxPoint(0, 0), this); 
2619                 m_dragImage
->Show(); 
2623         case TVN_DELETEITEM
: 
2625                 // NB: we might process this message using wxWindows event 
2626                 //     tables, but due to overhead of wxWin event system we 
2627                 //     prefer to do it here ourself (otherwise deleting a tree 
2628                 //     with many items is just too slow) 
2629                 NM_TREEVIEW
* tv 
= (NM_TREEVIEW 
*)lParam
; 
2631                 wxTreeItemId item 
= event
.m_item
; 
2632                 if ( HasIndirectData(item
) ) 
2634                     wxTreeItemIndirectData 
*data 
= (wxTreeItemIndirectData 
*) 
2636                     delete data
; // can't be NULL here 
2640                     wxTreeItemData 
*data 
= (wxTreeItemData 
*)tv
->itemOld
.lParam
; 
2641                     delete data
; // may be NULL, ok 
2644                 processed 
= true; // Make sure we don't get called twice 
2648         case TVN_BEGINLABELEDIT
: 
2649             // return true to cancel label editing 
2650             *result 
= !event
.IsAllowed(); 
2651             // set ES_WANTRETURN ( like we do in BeginLabelEdit ) 
2652             if(event
.IsAllowed()) 
2654                 HWND hText 
= TreeView_GetEditControl(GetHwnd()); 
2657                     // MBN: if m_textCtrl already has an HWND, it is a stale  
2658                     // pointer from a previous edit (because the user 
2659                     // didn't modify the label before dismissing the control, 
2660                     // and TVN_ENDLABELEDIT was not sent), so delete it 
2661                     if(m_textCtrl 
&& m_textCtrl
->GetHWND() != 0) 
2664                         m_textCtrl 
= new wxTextCtrl(); 
2665                     m_textCtrl
->SetParent(this); 
2666                     m_textCtrl
->SetHWND((WXHWND
)hText
); 
2667                     m_textCtrl
->SubclassWin((WXHWND
)hText
); 
2669                     // set wxTE_PROCESS_ENTER style for the text control to 
2670                     // force it to process the Enter presses itself, otherwise 
2671                     // they could be stolen from it by the dialog 
2673                     m_textCtrl
->SetWindowStyle(m_textCtrl
->GetWindowStyle() 
2674                                                | wxTE_PROCESS_ENTER
); 
2679         case TVN_ENDLABELEDIT
: 
2680             // return true to set the label to the new string: note that we 
2681             // also must pretend that we did process the message or it is going 
2682             // to be passed to DefWindowProc() which will happily return false 
2683             // cancelling the label change 
2684             *result 
= event
.IsAllowed(); 
2687             // ensure that we don't have the text ctrl which is going to be 
2692         case TVN_SELCHANGING
: 
2693         case TVN_ITEMEXPANDING
: 
2694             // return true to prevent the action from happening 
2695             *result 
= !event
.IsAllowed(); 
2698         case TVN_ITEMEXPANDED
: 
2699             // the item is not refreshed properly after expansion when it has 
2700             // an image depending on the expanded/collapsed state - bug in 
2701             // comctl32.dll or our code? 
2703                 NM_TREEVIEW
* tv 
= (NM_TREEVIEW 
*)lParam
; 
2704                 wxTreeItemId id 
= (WXHTREEITEM
)tv
->itemNew
.hItem
; 
2706                 int image 
= GetItemImage(id
, wxTreeItemIcon_Expanded
); 
2714         case TVN_GETDISPINFO
: 
2715             // NB: so far the user can't set the image himself anyhow, so do it 
2716             //     anyway - but this may change later 
2717             //if ( /* !processed && */ 1 ) 
2719                 wxTreeItemId item 
= event
.m_item
; 
2720                 TV_DISPINFO 
*info 
= (TV_DISPINFO 
*)lParam
; 
2721                 if ( info
->item
.mask 
& TVIF_IMAGE 
) 
2724                         DoGetItemImageFromData
 
2727                          IsExpanded(item
) ? wxTreeItemIcon_Expanded
 
2728                                           : wxTreeItemIcon_Normal
 
2731                 if ( info
->item
.mask 
& TVIF_SELECTEDIMAGE 
) 
2733                     info
->item
.iSelectedImage 
= 
2734                         DoGetItemImageFromData
 
2737                          IsExpanded(item
) ? wxTreeItemIcon_SelectedExpanded
 
2738                                           : wxTreeItemIcon_Selected
 
2745             // for the other messages the return value is ignored and there is 
2746             // nothing special to do 
2753 #endif // wxUSE_TREECTRL