1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     generic tree control implementation 
   4 // Author:      Robert Roebling 
   6 // Modified:    22/10/98 - almost total rewrite, simpler interface (VZ) 
   8 // Copyright:   (c) 1998 Robert Roebling and Julian Smart 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================= 
  14 // ============================================================================= 
  16 // ----------------------------------------------------------------------------- 
  18 // ----------------------------------------------------------------------------- 
  20 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA) 
  21   #pragma implementation "treectlg.h" 
  24 // For compilers that support precompilation, includes "wx.h". 
  25 #include "wx/wxprec.h" 
  33 #include "wx/treebase.h" 
  34 #include "wx/generic/treectlg.h" 
  36 #include "wx/textctrl.h" 
  37 #include "wx/imaglist.h" 
  38 #include "wx/settings.h" 
  39 #include "wx/dcclient.h" 
  41 #include "wx/renderer.h" 
  44     #include "wx/mac/private.h" 
  47 // ----------------------------------------------------------------------------- 
  49 // ----------------------------------------------------------------------------- 
  51 class WXDLLEXPORT wxGenericTreeItem
; 
  53 WX_DEFINE_EXPORTED_ARRAY_PTR(wxGenericTreeItem 
*, wxArrayGenericTreeItems
); 
  55 // ---------------------------------------------------------------------------- 
  57 // ---------------------------------------------------------------------------- 
  59 static const int NO_IMAGE 
= -1; 
  61 static const int PIXELS_PER_UNIT 
= 10; 
  63 // ----------------------------------------------------------------------------- 
  65 // ----------------------------------------------------------------------------- 
  67 // timer used for enabling in-place edit 
  68 class WXDLLEXPORT wxTreeRenameTimer
: public wxTimer
 
  71     // start editing the current item after half a second (if the mouse hasn't 
  72     // been clicked/moved) 
  75     wxTreeRenameTimer( wxGenericTreeCtrl 
*owner 
); 
  77     virtual void Notify(); 
  80     wxGenericTreeCtrl 
*m_owner
; 
  82     DECLARE_NO_COPY_CLASS(wxTreeRenameTimer
) 
  85 // control used for in-place edit 
  86 class WXDLLEXPORT wxTreeTextCtrl
: public wxTextCtrl
 
  89     wxTreeTextCtrl(wxGenericTreeCtrl 
*owner
, wxGenericTreeItem 
*item
); 
  94         m_owner
->OnRenameCancelled(m_itemEdited
); 
  96     const wxGenericTreeItem
* item() const { return m_itemEdited
; } 
  99     void OnChar( wxKeyEvent 
&event 
); 
 100     void OnKeyUp( wxKeyEvent 
&event 
); 
 101     void OnKillFocus( wxFocusEvent 
&event 
); 
 103     bool AcceptChanges(); 
 107     wxGenericTreeCtrl  
*m_owner
; 
 108     wxGenericTreeItem  
*m_itemEdited
; 
 109     wxString            m_startValue
; 
 111     bool                m_aboutToFinish
; 
 113     DECLARE_EVENT_TABLE() 
 114     DECLARE_NO_COPY_CLASS(wxTreeTextCtrl
) 
 117 // timer used to clear wxGenericTreeCtrl::m_findPrefix if no key was pressed 
 118 // for a sufficiently long time 
 119 class WXDLLEXPORT wxTreeFindTimer 
: public wxTimer
 
 122     // reset the current prefix after half a second of inactivity 
 123     enum { DELAY 
= 500 }; 
 125     wxTreeFindTimer( wxGenericTreeCtrl 
*owner 
) { m_owner 
= owner
; } 
 127     virtual void Notify() { m_owner
->m_findPrefix
.clear(); } 
 130     wxGenericTreeCtrl 
*m_owner
; 
 132     DECLARE_NO_COPY_CLASS(wxTreeFindTimer
) 
 136 class WXDLLEXPORT wxGenericTreeItem
 
 140     wxGenericTreeItem() { m_data 
= NULL
; } 
 141     wxGenericTreeItem( wxGenericTreeItem 
*parent
, 
 142                        const wxString
& text
, 
 145                        wxTreeItemData 
*data 
); 
 147     ~wxGenericTreeItem(); 
 150     wxArrayGenericTreeItems
& GetChildren() { return m_children
; } 
 152     const wxString
& GetText() const { return m_text
; } 
 153     int GetImage(wxTreeItemIcon which 
= wxTreeItemIcon_Normal
) const 
 154         { return m_images
[which
]; } 
 155     wxTreeItemData 
*GetData() const { return m_data
; } 
 157     // returns the current image for the item (depending on its 
 158     // selected/expanded/whatever state) 
 159     int GetCurrentImage() const; 
 161     void SetText( const wxString 
&text 
); 
 162     void SetImage(int image
, wxTreeItemIcon which
) { m_images
[which
] = image
; } 
 163     void SetData(wxTreeItemData 
*data
) { m_data 
= data
; } 
 165     void SetHasPlus(bool has 
= true) { m_hasPlus 
= has
; } 
 167     void SetBold(bool bold
) { m_isBold 
= bold
; } 
 169     int GetX() const { return m_x
; } 
 170     int GetY() const { return m_y
; } 
 172     void SetX(int x
) { m_x 
= x
; } 
 173     void SetY(int y
) { m_y 
= y
; } 
 175     int  GetHeight() const { return m_height
; } 
 176     int  GetWidth()  const { return m_width
; } 
 178     void SetHeight(int h
) { m_height 
= h
; } 
 179     void SetWidth(int w
) { m_width 
= w
; } 
 181     wxGenericTreeItem 
*GetParent() const { return m_parent
; } 
 184         // deletes all children notifying the treectrl about it if !NULL 
 186     void DeleteChildren(wxGenericTreeCtrl 
*tree 
= NULL
); 
 188     // get count of all children (and grand children if 'recursively') 
 189     size_t GetChildrenCount(bool recursively 
= true) const; 
 191     void Insert(wxGenericTreeItem 
*child
, size_t index
) 
 192     { m_children
.Insert(child
, index
); } 
 194     void GetSize( int &x
, int &y
, const wxGenericTreeCtrl
* ); 
 196         // return the item at given position (or NULL if no item), onButton is 
 197         // true if the point belongs to the item's button, otherwise it lies 
 198         // on the item's label 
 199     wxGenericTreeItem 
*HitTest( const wxPoint
& point
, 
 200                                 const wxGenericTreeCtrl 
*, 
 204     void Expand() { m_isCollapsed 
= false; } 
 205     void Collapse() { m_isCollapsed 
= true; } 
 207     void SetHilight( bool set 
= true ) { m_hasHilight 
= set
; } 
 210     bool HasChildren() const { return !m_children
.IsEmpty(); } 
 211     bool IsSelected()  const { return m_hasHilight 
!= 0; } 
 212     bool IsExpanded()  const { return !m_isCollapsed
; } 
 213     bool HasPlus()     const { return m_hasPlus 
|| HasChildren(); } 
 214     bool IsBold()      const { return m_isBold 
!= 0; } 
 217         // get them - may be NULL 
 218     wxTreeItemAttr 
*GetAttributes() const { return m_attr
; } 
 219         // get them ensuring that the pointer is not NULL 
 220     wxTreeItemAttr
& Attr() 
 224             m_attr 
= new wxTreeItemAttr
; 
 230     void SetAttributes(wxTreeItemAttr 
*attr
) 
 232         if ( m_ownsAttr 
) delete m_attr
; 
 236         // set them and delete when done 
 237     void AssignAttributes(wxTreeItemAttr 
*attr
) 
 244     // since there can be very many of these, we save size by chosing 
 245     // the smallest representation for the elements and by ordering 
 246     // the members to avoid padding. 
 247     wxString            m_text
;         // label to be rendered for item 
 249     wxTreeItemData     
*m_data
;         // user-provided data 
 251     wxArrayGenericTreeItems m_children
; // list of children 
 252     wxGenericTreeItem  
*m_parent
;       // parent of this item 
 254     wxTreeItemAttr     
*m_attr
;         // attributes??? 
 256     // tree ctrl images for the normal, selected, expanded and 
 257     // expanded+selected states 
 258     int                 m_images
[wxTreeItemIcon_Max
]; 
 260     wxCoord             m_x
;            // (virtual) offset from top 
 261     wxCoord             m_y
;            // (virtual) offset from left 
 262     int                 m_width
;        // width of this item 
 263     int                 m_height
;       // height of this item 
 265     // use bitfields to save size 
 266     int                 m_isCollapsed 
:1; 
 267     int                 m_hasHilight  
:1; // same as focused 
 268     int                 m_hasPlus     
:1; // used for item which doesn't have 
 269                                           // children but has a [+] button 
 270     int                 m_isBold      
:1; // render the label in bold font 
 271     int                 m_ownsAttr    
:1; // delete attribute when done 
 273     DECLARE_NO_COPY_CLASS(wxGenericTreeItem
) 
 276 // ============================================================================= 
 278 // ============================================================================= 
 280 // ---------------------------------------------------------------------------- 
 282 // ---------------------------------------------------------------------------- 
 284 // translate the key or mouse event flags to the type of selection we're 
 286 static void EventFlagsToSelType(long style
, 
 290                                 bool &extended_select
, 
 291                                 bool &unselect_others
) 
 293     is_multiple 
= (style 
& wxTR_MULTIPLE
) != 0; 
 294     extended_select 
= shiftDown 
&& is_multiple
; 
 295     unselect_others 
= !(extended_select 
|| (ctrlDown 
&& is_multiple
)); 
 298 // check if the given item is under another one 
 299 static bool IsDescendantOf(const wxGenericTreeItem 
*parent
, const wxGenericTreeItem 
*item
) 
 303         if ( item 
== parent 
) 
 305             // item is a descendant of parent 
 309         item 
= item
->GetParent(); 
 315 // ----------------------------------------------------------------------------- 
 316 // wxTreeRenameTimer (internal) 
 317 // ----------------------------------------------------------------------------- 
 319 wxTreeRenameTimer::wxTreeRenameTimer( wxGenericTreeCtrl 
*owner 
) 
 324 void wxTreeRenameTimer::Notify() 
 326     m_owner
->OnRenameTimer(); 
 329 //----------------------------------------------------------------------------- 
 330 // wxTreeTextCtrl (internal) 
 331 //----------------------------------------------------------------------------- 
 333 BEGIN_EVENT_TABLE(wxTreeTextCtrl
,wxTextCtrl
) 
 334     EVT_CHAR           (wxTreeTextCtrl::OnChar
) 
 335     EVT_KEY_UP         (wxTreeTextCtrl::OnKeyUp
) 
 336     EVT_KILL_FOCUS     (wxTreeTextCtrl::OnKillFocus
) 
 339 wxTreeTextCtrl::wxTreeTextCtrl(wxGenericTreeCtrl 
*owner
, 
 340                                wxGenericTreeItem 
*item
) 
 341               : m_itemEdited(item
), m_startValue(item
->GetText()) 
 345     m_aboutToFinish 
= false; 
 347     int w 
= m_itemEdited
->GetWidth(), 
 348         h 
= m_itemEdited
->GetHeight(); 
 351     m_owner
->CalcScrolledPosition(item
->GetX(), item
->GetY(), &x
, &y
); 
 356     int image 
= item
->GetCurrentImage(); 
 357     if ( image 
!= NO_IMAGE 
) 
 359         if ( m_owner
->m_imageListNormal 
) 
 361             m_owner
->m_imageListNormal
->GetSize( image
, image_w
, image_h 
); 
 366             wxFAIL_MSG(_T("you must create an image list to use images!")); 
 370     // FIXME: what are all these hardcoded 4, 8 and 11s really? 
 374     wxSize bs 
= DoGetBestSize() ; 
 375     // edit control height 
 378         int diff 
= h 
- ( bs
.y 
- 8 ) ; 
 384     (void)Create(m_owner
, wxID_ANY
, m_startValue
, 
 385                  wxPoint(x 
- 4, y 
- 4), wxSize(w 
+ 11, h 
+ 8)); 
 388 bool wxTreeTextCtrl::AcceptChanges() 
 390     const wxString value 
= GetValue(); 
 392     if ( value 
== m_startValue 
) 
 394         // nothing changed, always accept 
 395         // when an item remains unchanged, the owner 
 396         // needs to be notified that the user decided 
 397         // not to change the tree item label, and that 
 398         // the edit has been cancelled 
 400         m_owner
->OnRenameCancelled(m_itemEdited
); 
 404     if ( !m_owner
->OnRenameAccept(m_itemEdited
, value
) ) 
 406         // vetoed by the user 
 410     // accepted, do rename the item 
 411     m_owner
->SetItemText(m_itemEdited
, value
); 
 416 void wxTreeTextCtrl::Finish() 
 420         m_owner
->ResetTextControl(); 
 422         wxPendingDelete
.Append(this); 
 426         m_owner
->SetFocusIgnoringChildren(); 
 430 void wxTreeTextCtrl::OnChar( wxKeyEvent 
&event 
) 
 432     switch ( event
.m_keyCode 
) 
 435             m_aboutToFinish 
= true; 
 436             // Notify the owner about the changes 
 438             // Even if vetoed, close the control (consistent with MSW) 
 451 void wxTreeTextCtrl::OnKeyUp( wxKeyEvent 
&event 
) 
 455         // auto-grow the textctrl: 
 456         wxSize parentSize 
= m_owner
->GetSize(); 
 457         wxPoint myPos 
= GetPosition(); 
 458         wxSize mySize 
= GetSize(); 
 460         GetTextExtent(GetValue() + _T("M"), &sx
, &sy
); 
 461         if (myPos
.x 
+ sx 
> parentSize
.x
) 
 462             sx 
= parentSize
.x 
- myPos
.x
; 
 465         SetSize(sx
, wxDefaultCoord
); 
 471 void wxTreeTextCtrl::OnKillFocus( wxFocusEvent 
&event 
) 
 473     if ( !m_finished 
&& !m_aboutToFinish 
) 
 475         // We must finish regardless of success, otherwise we'll get 
 479         if ( !AcceptChanges() ) 
 480             m_owner
->OnRenameCancelled( m_itemEdited 
); 
 483     // We must let the native text control handle focus, too, otherwise 
 484     // it could have problems with the cursor (e.g., in wxGTK). 
 488 // ----------------------------------------------------------------------------- 
 490 // ----------------------------------------------------------------------------- 
 492 wxGenericTreeItem::wxGenericTreeItem(wxGenericTreeItem 
*parent
, 
 493                                      const wxString
& text
, 
 494                                      int image
, int selImage
, 
 495                                      wxTreeItemData 
*data
) 
 498     m_images
[wxTreeItemIcon_Normal
] = image
; 
 499     m_images
[wxTreeItemIcon_Selected
] = selImage
; 
 500     m_images
[wxTreeItemIcon_Expanded
] = NO_IMAGE
; 
 501     m_images
[wxTreeItemIcon_SelectedExpanded
] = NO_IMAGE
; 
 506     m_isCollapsed 
= true; 
 507     m_hasHilight 
= false; 
 513     m_attr 
= (wxTreeItemAttr 
*)NULL
; 
 516     // We don't know the height here yet. 
 521 wxGenericTreeItem::~wxGenericTreeItem() 
 525     if (m_ownsAttr
) delete m_attr
; 
 527     wxASSERT_MSG( m_children
.IsEmpty(), 
 528                   wxT("please call DeleteChildren() before deleting the item") ); 
 531 void wxGenericTreeItem::DeleteChildren(wxGenericTreeCtrl 
*tree
) 
 533     size_t count 
= m_children
.Count(); 
 534     for ( size_t n 
= 0; n 
< count
; n
++ ) 
 536         wxGenericTreeItem 
*child 
= m_children
[n
]; 
 538             tree
->SendDeleteEvent(child
); 
 540         child
->DeleteChildren(tree
); 
 541         if (child 
== tree
->m_select_me
) 
 542             tree
->m_select_me 
= NULL
; 
 549 void wxGenericTreeItem::SetText( const wxString 
&text 
) 
 554 size_t wxGenericTreeItem::GetChildrenCount(bool recursively
) const 
 556     size_t count 
= m_children
.Count(); 
 560     size_t total 
= count
; 
 561     for (size_t n 
= 0; n 
< count
; ++n
) 
 563         total 
+= m_children
[n
]->GetChildrenCount(); 
 569 void wxGenericTreeItem::GetSize( int &x
, int &y
, 
 570                                  const wxGenericTreeCtrl 
*theButton 
) 
 572     int bottomY
=m_y
+theButton
->GetLineHeight(this); 
 573     if ( y 
< bottomY 
) y 
= bottomY
; 
 574     int width 
= m_x 
+  m_width
; 
 575     if ( x 
< width 
) x 
= width
; 
 579         size_t count 
= m_children
.Count(); 
 580         for ( size_t n 
= 0; n 
< count
; ++n 
) 
 582             m_children
[n
]->GetSize( x
, y
, theButton 
); 
 587 wxGenericTreeItem 
*wxGenericTreeItem::HitTest(const wxPoint
& point
, 
 588                                               const wxGenericTreeCtrl 
*theCtrl
, 
 592     // for a hidden root node, don't evaluate it, but do evaluate children 
 593     if ( !(level 
== 0 && theCtrl
->HasFlag(wxTR_HIDE_ROOT
)) ) 
 596         int h 
= theCtrl
->GetLineHeight(this); 
 597         if ((point
.y 
> m_y
) && (point
.y 
< m_y 
+ h
)) 
 599             int y_mid 
= m_y 
+ h
/2; 
 600             if (point
.y 
< y_mid 
) 
 601                 flags 
|= wxTREE_HITTEST_ONITEMUPPERPART
; 
 603                 flags 
|= wxTREE_HITTEST_ONITEMLOWERPART
; 
 605             int xCross 
= m_x 
- theCtrl
->GetSpacing(); 
 607             // according to the drawing code the triangels are drawn 
 608             // at -4 , -4  from the position up to +10/+10 max 
 609             if ((point
.x 
> xCross
-4) && (point
.x 
< xCross
+10) && 
 610                 (point
.y 
> y_mid
-4) && (point
.y 
< y_mid
+10) && 
 611                 HasPlus() && theCtrl
->HasButtons() ) 
 613             // 5 is the size of the plus sign 
 614             if ((point
.x 
> xCross
-6) && (point
.x 
< xCross
+6) && 
 615                 (point
.y 
> y_mid
-6) && (point
.y 
< y_mid
+6) && 
 616                 HasPlus() && theCtrl
->HasButtons() ) 
 619                 flags 
|= wxTREE_HITTEST_ONITEMBUTTON
; 
 623             if ((point
.x 
>= m_x
) && (point
.x 
<= m_x
+m_width
)) 
 628                 // assuming every image (normal and selected) has the same size! 
 629                 if ( (GetImage() != NO_IMAGE
) && theCtrl
->m_imageListNormal 
) 
 630                     theCtrl
->m_imageListNormal
->GetSize(GetImage(), 
 633                 if ((image_w 
!= -1) && (point
.x 
<= m_x 
+ image_w 
+ 1)) 
 634                     flags 
|= wxTREE_HITTEST_ONITEMICON
; 
 636                     flags 
|= wxTREE_HITTEST_ONITEMLABEL
; 
 642                 flags 
|= wxTREE_HITTEST_ONITEMINDENT
; 
 643             if (point
.x 
> m_x
+m_width
) 
 644                 flags 
|= wxTREE_HITTEST_ONITEMRIGHT
; 
 649         // if children are expanded, fall through to evaluate them 
 650         if (m_isCollapsed
) return (wxGenericTreeItem
*) NULL
; 
 654     size_t count 
= m_children
.Count(); 
 655     for ( size_t n 
= 0; n 
< count
; n
++ ) 
 657         wxGenericTreeItem 
*res 
= m_children
[n
]->HitTest( point
, 
 665     return (wxGenericTreeItem
*) NULL
; 
 668 int wxGenericTreeItem::GetCurrentImage() const 
 670     int image 
= NO_IMAGE
; 
 675             image 
= GetImage(wxTreeItemIcon_SelectedExpanded
); 
 678         if ( image 
== NO_IMAGE 
) 
 680             // we usually fall back to the normal item, but try just the 
 681             // expanded one (and not selected) first in this case 
 682             image 
= GetImage(wxTreeItemIcon_Expanded
); 
 688             image 
= GetImage(wxTreeItemIcon_Selected
); 
 691     // maybe it doesn't have the specific image we want, 
 692     // try the default one instead 
 693     if ( image 
== NO_IMAGE 
) image 
= GetImage(); 
 698 // ----------------------------------------------------------------------------- 
 699 // wxGenericTreeCtrl implementation 
 700 // ----------------------------------------------------------------------------- 
 702 IMPLEMENT_DYNAMIC_CLASS(wxGenericTreeCtrl
, wxScrolledWindow
) 
 704 BEGIN_EVENT_TABLE(wxGenericTreeCtrl
,wxScrolledWindow
) 
 705     EVT_PAINT          (wxGenericTreeCtrl::OnPaint
) 
 706     EVT_MOUSE_EVENTS   (wxGenericTreeCtrl::OnMouse
) 
 707     EVT_CHAR           (wxGenericTreeCtrl::OnChar
) 
 708     EVT_SET_FOCUS      (wxGenericTreeCtrl::OnSetFocus
) 
 709     EVT_KILL_FOCUS     (wxGenericTreeCtrl::OnKillFocus
) 
 710     EVT_TREE_ITEM_GETTOOLTIP(wxID_ANY
, wxGenericTreeCtrl::OnGetToolTip
) 
 713 #if !defined(__WXMSW__) || defined(__WXUNIVERSAL__) 
 715  * wxTreeCtrl has to be a real class or we have problems with 
 716  * the run-time information. 
 719 IMPLEMENT_DYNAMIC_CLASS(wxTreeCtrl
, wxGenericTreeCtrl
) 
 722 // ----------------------------------------------------------------------------- 
 723 // construction/destruction 
 724 // ----------------------------------------------------------------------------- 
 726 void wxGenericTreeCtrl::Init() 
 728     m_current 
= m_key_current 
= m_anchor 
= m_select_me 
= (wxGenericTreeItem 
*) NULL
; 
 736     m_hilightBrush 
= new wxBrush
 
 738                             wxSystemSettings::GetColour
 
 740                                 wxSYS_COLOUR_HIGHLIGHT
 
 745     m_hilightUnfocusedBrush 
= new wxBrush
 
 747                                  wxSystemSettings::GetColour
 
 749                                      wxSYS_COLOUR_BTNSHADOW
 
 754     m_imageListNormal 
= m_imageListButtons 
= 
 755     m_imageListState 
= (wxImageList 
*) NULL
; 
 756     m_ownsImageListNormal 
= m_ownsImageListButtons 
= 
 757     m_ownsImageListState 
= false; 
 760     m_isDragging 
= false; 
 761     m_dropTarget 
= m_oldSelection 
= NULL
; 
 765     m_renameTimer 
= NULL
; 
 770     m_dropEffectAboveItem 
= false; 
 772     m_lastOnSame 
= false; 
 774 #ifdef __WXMAC_CARBON__ 
 775     m_normalFont
.MacCreateThemeFont( kThemeViewsFont 
) ; 
 777     m_normalFont 
= wxSystemSettings::GetFont( wxSYS_DEFAULT_GUI_FONT 
); 
 779     m_boldFont 
= wxFont(m_normalFont
.GetPointSize(), 
 780                         m_normalFont
.GetFamily(), 
 781                         m_normalFont
.GetStyle(), 
 783                         m_normalFont
.GetUnderlined(), 
 784                         m_normalFont
.GetFaceName(), 
 785                         m_normalFont
.GetEncoding()); 
 788 bool wxGenericTreeCtrl::Create(wxWindow 
*parent
, 
 793                                const wxValidator
& wxVALIDATOR_PARAM(validator
), 
 794                                const wxString
& name 
) 
 798     wxGetOsVersion( &major
, &minor 
); 
 800     style 
&= ~wxTR_LINES_AT_ROOT
; 
 801     style 
|= wxTR_NO_LINES
; 
 803         style 
|= wxTR_ROW_LINES
; 
 806     wxScrolledWindow::Create( parent
, id
, pos
, size
, 
 807                               style
|wxHSCROLL
|wxVSCROLL
, name 
); 
 809     // If the tree display has no buttons, but does have 
 810     // connecting lines, we can use a narrower layout. 
 811     // It may not be a good idea to force this... 
 812     if (!HasButtons() && !HasFlag(wxTR_NO_LINES
)) 
 819     SetValidator( validator 
); 
 822     wxVisualAttributes attr 
= GetDefaultAttributes(); 
 823     SetOwnForegroundColour( attr
.colFg 
); 
 824     SetOwnBackgroundColour( attr
.colBg 
); 
 826         SetOwnFont(attr
.font
); 
 828 //  m_dottedPen = wxPen( "grey", 0, wxDOT );  too slow under XFree86 
 829     m_dottedPen 
= wxPen( wxT("grey"), 0, 0 ); 
 836 wxGenericTreeCtrl::~wxGenericTreeCtrl() 
 838     delete m_hilightBrush
; 
 839     delete m_hilightUnfocusedBrush
; 
 843     delete m_renameTimer
; 
 846     if (m_ownsImageListNormal
) 
 847         delete m_imageListNormal
; 
 848     if (m_ownsImageListState
) 
 849         delete m_imageListState
; 
 850     if (m_ownsImageListButtons
) 
 851         delete m_imageListButtons
; 
 854 // ----------------------------------------------------------------------------- 
 856 // ----------------------------------------------------------------------------- 
 858 size_t wxGenericTreeCtrl::GetCount() const 
 866     size_t count 
= m_anchor
->GetChildrenCount(); 
 867     if ( !HasFlag(wxTR_HIDE_ROOT
) ) 
 869         // take the root itself into account 
 876 void wxGenericTreeCtrl::SetIndent(unsigned int indent
) 
 878     m_indent 
= (unsigned short) indent
; 
 882 void wxGenericTreeCtrl::SetSpacing(unsigned int spacing
) 
 884     m_spacing 
= (unsigned short) spacing
; 
 889 wxGenericTreeCtrl::GetChildrenCount(const wxTreeItemId
& item
, 
 890                                     bool recursively
) const 
 892     wxCHECK_MSG( item
.IsOk(), 0u, wxT("invalid tree item") ); 
 894     return ((wxGenericTreeItem
*) item
.m_pItem
)->GetChildrenCount(recursively
); 
 897 void wxGenericTreeCtrl::SetWindowStyle(const long styles
) 
 899     if (!HasFlag(wxTR_HIDE_ROOT
) && (styles 
& wxTR_HIDE_ROOT
)) 
 901         // if we will hide the root, make sure children are visible 
 902         m_anchor
->SetHasPlus(); 
 904         CalculatePositions(); 
 907     // right now, just sets the styles.  Eventually, we may 
 908     // want to update the inherited styles, but right now 
 909     // none of the parents has updatable styles 
 910     m_windowStyle 
= styles
; 
 914 // ----------------------------------------------------------------------------- 
 915 // functions to work with tree items 
 916 // ----------------------------------------------------------------------------- 
 918 wxString 
wxGenericTreeCtrl::GetItemText(const wxTreeItemId
& item
) const 
 920     wxCHECK_MSG( item
.IsOk(), wxEmptyString
, wxT("invalid tree item") ); 
 922     return ((wxGenericTreeItem
*) item
.m_pItem
)->GetText(); 
 925 int wxGenericTreeCtrl::GetItemImage(const wxTreeItemId
& item
, 
 926                              wxTreeItemIcon which
) const 
 928     wxCHECK_MSG( item
.IsOk(), -1, wxT("invalid tree item") ); 
 930     return ((wxGenericTreeItem
*) item
.m_pItem
)->GetImage(which
); 
 933 wxTreeItemData 
*wxGenericTreeCtrl::GetItemData(const wxTreeItemId
& item
) const 
 935     wxCHECK_MSG( item
.IsOk(), NULL
, wxT("invalid tree item") ); 
 937     return ((wxGenericTreeItem
*) item
.m_pItem
)->GetData(); 
 940 wxColour 
wxGenericTreeCtrl::GetItemTextColour(const wxTreeItemId
& item
) const 
 942     wxCHECK_MSG( item
.IsOk(), wxNullColour
, wxT("invalid tree item") ); 
 944     wxGenericTreeItem 
*pItem 
= (wxGenericTreeItem
*) item
.m_pItem
; 
 945     return pItem
->Attr().GetTextColour(); 
 948 wxColour 
wxGenericTreeCtrl::GetItemBackgroundColour(const wxTreeItemId
& item
) const 
 950     wxCHECK_MSG( item
.IsOk(), wxNullColour
, wxT("invalid tree item") ); 
 952     wxGenericTreeItem 
*pItem 
= (wxGenericTreeItem
*) item
.m_pItem
; 
 953     return pItem
->Attr().GetBackgroundColour(); 
 956 wxFont 
wxGenericTreeCtrl::GetItemFont(const wxTreeItemId
& item
) const 
 958     wxCHECK_MSG( item
.IsOk(), wxNullFont
, wxT("invalid tree item") ); 
 960     wxGenericTreeItem 
*pItem 
= (wxGenericTreeItem
*) item
.m_pItem
; 
 961     return pItem
->Attr().GetFont(); 
 964 void wxGenericTreeCtrl::SetItemText(const wxTreeItemId
& item
, const wxString
& text
) 
 966     wxCHECK_RET( item
.IsOk(), wxT("invalid tree item") ); 
 969     wxGenericTreeItem 
*pItem 
= (wxGenericTreeItem
*) item
.m_pItem
; 
 970     pItem
->SetText(text
); 
 971     CalculateSize(pItem
, dc
); 
 975 void wxGenericTreeCtrl::SetItemImage(const wxTreeItemId
& item
, 
 977                               wxTreeItemIcon which
) 
 979     wxCHECK_RET( item
.IsOk(), wxT("invalid tree item") ); 
 981     wxGenericTreeItem 
*pItem 
= (wxGenericTreeItem
*) item
.m_pItem
; 
 982     pItem
->SetImage(image
, which
); 
 985     CalculateSize(pItem
, dc
); 
 989 void wxGenericTreeCtrl::SetItemData(const wxTreeItemId
& item
, wxTreeItemData 
*data
) 
 991     wxCHECK_RET( item
.IsOk(), wxT("invalid tree item") ); 
 996     ((wxGenericTreeItem
*) item
.m_pItem
)->SetData(data
); 
 999 void wxGenericTreeCtrl::SetItemHasChildren(const wxTreeItemId
& item
, bool has
) 
1001     wxCHECK_RET( item
.IsOk(), wxT("invalid tree item") ); 
1003     wxGenericTreeItem 
*pItem 
= (wxGenericTreeItem
*) item
.m_pItem
; 
1004     pItem
->SetHasPlus(has
); 
1008 void wxGenericTreeCtrl::SetItemBold(const wxTreeItemId
& item
, bool bold
) 
1010     wxCHECK_RET( item
.IsOk(), wxT("invalid tree item") ); 
1012     // avoid redrawing the tree if no real change 
1013     wxGenericTreeItem 
*pItem 
= (wxGenericTreeItem
*) item
.m_pItem
; 
1014     if ( pItem
->IsBold() != bold 
) 
1016         pItem
->SetBold(bold
); 
1021 void wxGenericTreeCtrl::SetItemDropHighlight(const wxTreeItemId
& item
, 
1024     wxCHECK_RET( item
.IsOk(), wxT("invalid tree item") ); 
1030         bg 
= wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHT
); 
1031         fg 
= wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHTTEXT
); 
1034     wxGenericTreeItem 
*pItem 
= (wxGenericTreeItem
*) item
.m_pItem
; 
1035     pItem
->Attr().SetTextColour(fg
); 
1036     pItem
->Attr().SetBackgroundColour(bg
); 
1040 void wxGenericTreeCtrl::SetItemTextColour(const wxTreeItemId
& item
, 
1041                                    const wxColour
& col
) 
1043     wxCHECK_RET( item
.IsOk(), wxT("invalid tree item") ); 
1045     wxGenericTreeItem 
*pItem 
= (wxGenericTreeItem
*) item
.m_pItem
; 
1046     pItem
->Attr().SetTextColour(col
); 
1050 void wxGenericTreeCtrl::SetItemBackgroundColour(const wxTreeItemId
& item
, 
1051                                          const wxColour
& col
) 
1053     wxCHECK_RET( item
.IsOk(), wxT("invalid tree item") ); 
1055     wxGenericTreeItem 
*pItem 
= (wxGenericTreeItem
*) item
.m_pItem
; 
1056     pItem
->Attr().SetBackgroundColour(col
); 
1060 void wxGenericTreeCtrl::SetItemFont(const wxTreeItemId
& item
, const wxFont
& font
) 
1062     wxCHECK_RET( item
.IsOk(), wxT("invalid tree item") ); 
1064     wxGenericTreeItem 
*pItem 
= (wxGenericTreeItem
*) item
.m_pItem
; 
1065     pItem
->Attr().SetFont(font
); 
1069 bool wxGenericTreeCtrl::SetFont( const wxFont 
&font 
) 
1071     wxScrolledWindow::SetFont(font
); 
1073     m_normalFont 
= font 
; 
1074     m_boldFont 
= wxFont(m_normalFont
.GetPointSize(), 
1075                         m_normalFont
.GetFamily(), 
1076                         m_normalFont
.GetStyle(), 
1078                         m_normalFont
.GetUnderlined(), 
1079                         m_normalFont
.GetFaceName(), 
1080                         m_normalFont
.GetEncoding()); 
1086 // ----------------------------------------------------------------------------- 
1087 // item status inquiries 
1088 // ----------------------------------------------------------------------------- 
1090 bool wxGenericTreeCtrl::IsVisible(const wxTreeItemId
& item
) const 
1092     wxCHECK_MSG( item
.IsOk(), false, wxT("invalid tree item") ); 
1094     // An item is only visible if it's not a descendant of a collapsed item 
1095     wxGenericTreeItem 
*pItem 
= (wxGenericTreeItem
*) item
.m_pItem
; 
1096     wxGenericTreeItem
* parent 
= pItem
->GetParent(); 
1099         if (!parent
->IsExpanded()) 
1101         parent 
= parent
->GetParent(); 
1105     GetViewStart(& startX
, & startY
); 
1107     wxSize clientSize 
= GetClientSize(); 
1110     if (!GetBoundingRect(item
, rect
)) 
1112     if (rect
.GetWidth() == 0 || rect
.GetHeight() == 0) 
1114     if (rect
.GetBottom() < 0 || rect
.GetTop() > clientSize
.y
) 
1116     if (rect
.GetRight() < 0 || rect
.GetLeft() > clientSize
.x
) 
1122 bool wxGenericTreeCtrl::ItemHasChildren(const wxTreeItemId
& item
) const 
1124     wxCHECK_MSG( item
.IsOk(), false, wxT("invalid tree item") ); 
1126     // consider that the item does have children if it has the "+" button: it 
1127     // might not have them (if it had never been expanded yet) but then it 
1128     // could have them as well and it's better to err on this side rather than 
1129     // disabling some operations which are restricted to the items with 
1130     // children for an item which does have them 
1131     return ((wxGenericTreeItem
*) item
.m_pItem
)->HasPlus(); 
1134 bool wxGenericTreeCtrl::IsExpanded(const wxTreeItemId
& item
) const 
1136     wxCHECK_MSG( item
.IsOk(), false, wxT("invalid tree item") ); 
1138     return ((wxGenericTreeItem
*) item
.m_pItem
)->IsExpanded(); 
1141 bool wxGenericTreeCtrl::IsSelected(const wxTreeItemId
& item
) const 
1143     wxCHECK_MSG( item
.IsOk(), false, wxT("invalid tree item") ); 
1145     return ((wxGenericTreeItem
*) item
.m_pItem
)->IsSelected(); 
1148 bool wxGenericTreeCtrl::IsBold(const wxTreeItemId
& item
) const 
1150     wxCHECK_MSG( item
.IsOk(), false, wxT("invalid tree item") ); 
1152     return ((wxGenericTreeItem
*) item
.m_pItem
)->IsBold(); 
1155 // ----------------------------------------------------------------------------- 
1157 // ----------------------------------------------------------------------------- 
1159 wxTreeItemId 
wxGenericTreeCtrl::GetItemParent(const wxTreeItemId
& item
) const 
1161     wxCHECK_MSG( item
.IsOk(), wxTreeItemId(), wxT("invalid tree item") ); 
1163     return ((wxGenericTreeItem
*) item
.m_pItem
)->GetParent(); 
1166 wxTreeItemId 
wxGenericTreeCtrl::GetFirstChild(const wxTreeItemId
& item
, 
1167                                               wxTreeItemIdValue
& cookie
) const 
1169     wxCHECK_MSG( item
.IsOk(), wxTreeItemId(), wxT("invalid tree item") ); 
1172     return GetNextChild(item
, cookie
); 
1175 wxTreeItemId 
wxGenericTreeCtrl::GetNextChild(const wxTreeItemId
& item
, 
1176                                              wxTreeItemIdValue
& cookie
) const 
1178     wxCHECK_MSG( item
.IsOk(), wxTreeItemId(), wxT("invalid tree item") ); 
1180     wxArrayGenericTreeItems
& children 
= ((wxGenericTreeItem
*) item
.m_pItem
)->GetChildren(); 
1182     // it's ok to cast cookie to size_t, we never have indices big enough to 
1183     // overflow "void *" 
1184     size_t *pIndex 
= (size_t *)&cookie
; 
1185     if ( *pIndex 
< children
.Count() ) 
1187         return children
.Item((*pIndex
)++); 
1191         // there are no more of them 
1192         return wxTreeItemId(); 
1196 #if WXWIN_COMPATIBILITY_2_4 
1198 wxTreeItemId 
wxGenericTreeCtrl::GetFirstChild(const wxTreeItemId
& item
, 
1201     wxCHECK_MSG( item
.IsOk(), wxTreeItemId(), wxT("invalid tree item") ); 
1204     return GetNextChild(item
, cookie
); 
1207 wxTreeItemId 
wxGenericTreeCtrl::GetNextChild(const wxTreeItemId
& item
, 
1210     wxCHECK_MSG( item
.IsOk(), wxTreeItemId(), wxT("invalid tree item") ); 
1212     wxArrayGenericTreeItems
& children 
= ((wxGenericTreeItem
*) item
.m_pItem
)->GetChildren(); 
1213     if ( (size_t)cookie 
< children
.Count() ) 
1215         return children
.Item((size_t)cookie
++); 
1219         // there are no more of them 
1220         return wxTreeItemId(); 
1224 #endif // WXWIN_COMPATIBILITY_2_4 
1226 wxTreeItemId 
wxGenericTreeCtrl::GetLastChild(const wxTreeItemId
& item
) const 
1228     wxCHECK_MSG( item
.IsOk(), wxTreeItemId(), wxT("invalid tree item") ); 
1230     wxArrayGenericTreeItems
& children 
= ((wxGenericTreeItem
*) item
.m_pItem
)->GetChildren(); 
1231     return (children
.IsEmpty() ? wxTreeItemId() : wxTreeItemId(children
.Last())); 
1234 wxTreeItemId 
wxGenericTreeCtrl::GetNextSibling(const wxTreeItemId
& item
) const 
1236     wxCHECK_MSG( item
.IsOk(), wxTreeItemId(), wxT("invalid tree item") ); 
1238     wxGenericTreeItem 
*i 
= (wxGenericTreeItem
*) item
.m_pItem
; 
1239     wxGenericTreeItem 
*parent 
= i
->GetParent(); 
1240     if ( parent 
== NULL 
) 
1242         // root item doesn't have any siblings 
1243         return wxTreeItemId(); 
1246     wxArrayGenericTreeItems
& siblings 
= parent
->GetChildren(); 
1247     int index 
= siblings
.Index(i
); 
1248     wxASSERT( index 
!= wxNOT_FOUND 
); // I'm not a child of my parent? 
1250     size_t n 
= (size_t)(index 
+ 1); 
1251     return n 
== siblings
.Count() ? wxTreeItemId() : wxTreeItemId(siblings
[n
]); 
1254 wxTreeItemId 
wxGenericTreeCtrl::GetPrevSibling(const wxTreeItemId
& item
) const 
1256     wxCHECK_MSG( item
.IsOk(), wxTreeItemId(), wxT("invalid tree item") ); 
1258     wxGenericTreeItem 
*i 
= (wxGenericTreeItem
*) item
.m_pItem
; 
1259     wxGenericTreeItem 
*parent 
= i
->GetParent(); 
1260     if ( parent 
== NULL 
) 
1262         // root item doesn't have any siblings 
1263         return wxTreeItemId(); 
1266     wxArrayGenericTreeItems
& siblings 
= parent
->GetChildren(); 
1267     int index 
= siblings
.Index(i
); 
1268     wxASSERT( index 
!= wxNOT_FOUND 
); // I'm not a child of my parent? 
1270     return index 
== 0 ? wxTreeItemId() 
1271                       : wxTreeItemId(siblings
[(size_t)(index 
- 1)]); 
1274 // Only for internal use right now, but should probably be public 
1275 wxTreeItemId 
wxGenericTreeCtrl::GetNext(const wxTreeItemId
& item
) const 
1277     wxCHECK_MSG( item
.IsOk(), wxTreeItemId(), wxT("invalid tree item") ); 
1279     wxGenericTreeItem 
*i 
= (wxGenericTreeItem
*) item
.m_pItem
; 
1281     // First see if there are any children. 
1282     wxArrayGenericTreeItems
& children 
= i
->GetChildren(); 
1283     if (children
.GetCount() > 0) 
1285          return children
.Item(0); 
1289          // Try a sibling of this or ancestor instead 
1290          wxTreeItemId p 
= item
; 
1291          wxTreeItemId toFind
; 
1294               toFind 
= GetNextSibling(p
); 
1295               p 
= GetItemParent(p
); 
1296          } while (p
.IsOk() && !toFind
.IsOk()); 
1301 wxTreeItemId 
wxGenericTreeCtrl::GetFirstVisibleItem() const 
1303     wxTreeItemId id 
= GetRootItem(); 
1312     } while (id
.IsOk()); 
1314     return wxTreeItemId(); 
1317 wxTreeItemId 
wxGenericTreeCtrl::GetNextVisible(const wxTreeItemId
& item
) const 
1319     wxCHECK_MSG( item
.IsOk(), wxTreeItemId(), wxT("invalid tree item") ); 
1321     wxTreeItemId id 
= item
; 
1324         while (id 
= GetNext(id
), id
.IsOk()) 
1330     return wxTreeItemId(); 
1333 wxTreeItemId 
wxGenericTreeCtrl::GetPrevVisible(const wxTreeItemId
& item
) const 
1335     wxCHECK_MSG( item
.IsOk(), wxTreeItemId(), wxT("invalid tree item") ); 
1337     wxFAIL_MSG(wxT("not implemented")); 
1339     return wxTreeItemId(); 
1342 // called by wxTextTreeCtrl when it marks itself for deletion 
1343 void wxGenericTreeCtrl::ResetTextControl() 
1348 // find the first item starting with the given prefix after the given item 
1349 wxTreeItemId 
wxGenericTreeCtrl::FindItem(const wxTreeItemId
& idParent
, 
1350                                          const wxString
& prefixOrig
) const 
1352     // match is case insensitive as this is more convenient to the user: having 
1353     // to press Shift-letter to go to the item starting with a capital letter 
1354     // would be too bothersome 
1355     wxString prefix 
= prefixOrig
.Lower(); 
1357     // determine the starting point: we shouldn't take the current item (this 
1358     // allows to switch between two items starting with the same letter just by 
1359     // pressing it) but we shouldn't jump to the next one if the user is 
1360     // continuing to type as otherwise he might easily skip the item he wanted 
1361     wxTreeItemId id 
= idParent
; 
1362     if ( prefix
.length() == 1 ) 
1367     // look for the item starting with the given prefix after it 
1368     while ( id
.IsOk() && !GetItemText(id
).Lower().StartsWith(prefix
) ) 
1373     // if we haven't found anything... 
1376         // ... wrap to the beginning 
1378         if ( HasFlag(wxTR_HIDE_ROOT
) ) 
1380             // can't select virtual root 
1384         // and try all the items (stop when we get to the one we started from) 
1385         while ( id 
!= idParent 
&& !GetItemText(id
).Lower().StartsWith(prefix
) ) 
1394 // ----------------------------------------------------------------------------- 
1396 // ----------------------------------------------------------------------------- 
1398 wxTreeItemId 
wxGenericTreeCtrl::DoInsertItem(const wxTreeItemId
& parentId
, 
1400                                       const wxString
& text
, 
1401                                       int image
, int selImage
, 
1402                                       wxTreeItemData 
*data
) 
1404     wxGenericTreeItem 
*parent 
= (wxGenericTreeItem
*) parentId
.m_pItem
; 
1407         // should we give a warning here? 
1408         return AddRoot(text
, image
, selImage
, data
); 
1411     m_dirty 
= true;     // do this first so stuff below doesn't cause flicker 
1413     wxGenericTreeItem 
*item 
= 
1414         new wxGenericTreeItem( parent
, text
, image
, selImage
, data 
); 
1418         data
->m_pItem 
= item
; 
1421     parent
->Insert( item
, previous 
); 
1426 wxTreeItemId 
wxGenericTreeCtrl::AddRoot(const wxString
& text
, 
1427                                  int image
, int selImage
, 
1428                                  wxTreeItemData 
*data
) 
1430     wxCHECK_MSG( !m_anchor
, wxTreeItemId(), wxT("tree can have only one root") ); 
1432     m_dirty 
= true;     // do this first so stuff below doesn't cause flicker 
1434     m_anchor 
= new wxGenericTreeItem((wxGenericTreeItem 
*)NULL
, text
, 
1435                                    image
, selImage
, data
); 
1438         data
->m_pItem 
= m_anchor
; 
1441     if (HasFlag(wxTR_HIDE_ROOT
)) 
1443         // if root is hidden, make sure we can navigate 
1445         m_anchor
->SetHasPlus(); 
1447         CalculatePositions(); 
1450     if (!HasFlag(wxTR_MULTIPLE
)) 
1452         m_current 
= m_key_current 
= m_anchor
; 
1453         m_current
->SetHilight( true ); 
1459 wxTreeItemId 
wxGenericTreeCtrl::PrependItem(const wxTreeItemId
& parent
, 
1460                                      const wxString
& text
, 
1461                                      int image
, int selImage
, 
1462                                      wxTreeItemData 
*data
) 
1464     return DoInsertItem(parent
, 0u, text
, image
, selImage
, data
); 
1467 wxTreeItemId 
wxGenericTreeCtrl::InsertItem(const wxTreeItemId
& parentId
, 
1468                                     const wxTreeItemId
& idPrevious
, 
1469                                     const wxString
& text
, 
1470                                     int image
, int selImage
, 
1471                                     wxTreeItemData 
*data
) 
1473     wxGenericTreeItem 
*parent 
= (wxGenericTreeItem
*) parentId
.m_pItem
; 
1476         // should we give a warning here? 
1477         return AddRoot(text
, image
, selImage
, data
); 
1481     if (idPrevious
.IsOk()) 
1483         index 
= parent
->GetChildren().Index((wxGenericTreeItem
*) idPrevious
.m_pItem
); 
1484         wxASSERT_MSG( index 
!= wxNOT_FOUND
, 
1485                       wxT("previous item in wxGenericTreeCtrl::InsertItem() is not a sibling") ); 
1488     return DoInsertItem(parentId
, (size_t)++index
, text
, image
, selImage
, data
); 
1491 wxTreeItemId 
wxGenericTreeCtrl::InsertItem(const wxTreeItemId
& parentId
, 
1493                                     const wxString
& text
, 
1494                                     int image
, int selImage
, 
1495                                     wxTreeItemData 
*data
) 
1497     wxGenericTreeItem 
*parent 
= (wxGenericTreeItem
*) parentId
.m_pItem
; 
1500         // should we give a warning here? 
1501         return AddRoot(text
, image
, selImage
, data
); 
1504     return DoInsertItem(parentId
, before
, text
, image
, selImage
, data
); 
1507 wxTreeItemId 
wxGenericTreeCtrl::AppendItem(const wxTreeItemId
& parentId
, 
1508                                     const wxString
& text
, 
1509                                     int image
, int selImage
, 
1510                                     wxTreeItemData 
*data
) 
1512     wxGenericTreeItem 
*parent 
= (wxGenericTreeItem
*) parentId
.m_pItem
; 
1515         // should we give a warning here? 
1516         return AddRoot(text
, image
, selImage
, data
); 
1519     return DoInsertItem( parent
, parent
->GetChildren().Count(), text
, 
1520                          image
, selImage
, data
); 
1523 void wxGenericTreeCtrl::SendDeleteEvent(wxGenericTreeItem 
*item
) 
1525     wxTreeEvent 
event( wxEVT_COMMAND_TREE_DELETE_ITEM
, GetId() ); 
1526     event
.m_item 
= item
; 
1527     event
.SetEventObject( this ); 
1528     ProcessEvent( event 
); 
1531 // Don't leave edit or selection on a child which is about to disappear 
1532 void wxGenericTreeCtrl::ChildrenClosing(wxGenericTreeItem
* item
) 
1534     if (m_textCtrl 
!= NULL 
&& item 
!= m_textCtrl
->item() && IsDescendantOf(item
, m_textCtrl
->item())) { 
1535         m_textCtrl
->StopEditing(); 
1537     if (item 
!= m_key_current 
&& IsDescendantOf(item
, m_key_current
)) { 
1538         m_key_current 
= NULL
; 
1540     if (IsDescendantOf(item
, m_select_me
)) { 
1543     if (item 
!= m_current 
&& IsDescendantOf(item
, m_current
)) { 
1544         m_current
->SetHilight( false ); 
1550 void wxGenericTreeCtrl::DeleteChildren(const wxTreeItemId
& itemId
) 
1552     m_dirty 
= true;     // do this first so stuff below doesn't cause flicker 
1554     wxGenericTreeItem 
*item 
= (wxGenericTreeItem
*) itemId
.m_pItem
; 
1555     ChildrenClosing(item
); 
1556     item
->DeleteChildren(this); 
1559 void wxGenericTreeCtrl::Delete(const wxTreeItemId
& itemId
) 
1561     m_dirty 
= true;     // do this first so stuff below doesn't cause flicker 
1563     wxGenericTreeItem 
*item 
= (wxGenericTreeItem
*) itemId
.m_pItem
; 
1565     if (m_textCtrl 
!= NULL 
&& IsDescendantOf(item
, m_textCtrl
->item())) 
1567         // can't delete the item being edited, cancel editing it first 
1568         m_textCtrl
->StopEditing(); 
1571     wxGenericTreeItem 
*parent 
= item
->GetParent(); 
1573     // don't keep stale pointers around! 
1574     if ( IsDescendantOf(item
, m_key_current
) ) 
1576         // Don't silently change the selection: 
1577         // do it properly in idle time, so event 
1578         // handlers get called. 
1580         // m_key_current = parent; 
1581         m_key_current 
= NULL
; 
1584     // m_select_me records whether we need to select 
1585     // a different item, in idle time. 
1586     if ( m_select_me 
&& IsDescendantOf(item
, m_select_me
) ) 
1588         m_select_me 
= parent
; 
1591     if ( IsDescendantOf(item
, m_current
) ) 
1593         // Don't silently change the selection: 
1594         // do it properly in idle time, so event 
1595         // handlers get called. 
1597         // m_current = parent; 
1599         m_select_me 
= parent
; 
1602     // remove the item from the tree 
1605         parent
->GetChildren().Remove( item 
);  // remove by value 
1607     else // deleting the root 
1609         // nothing will be left in the tree 
1613     // and delete all of its children and the item itself now 
1614     item
->DeleteChildren(this); 
1615     SendDeleteEvent(item
); 
1617     if (item 
== m_select_me
) 
1623 void wxGenericTreeCtrl::DeleteAllItems() 
1631 void wxGenericTreeCtrl::Expand(const wxTreeItemId
& itemId
) 
1633     wxGenericTreeItem 
*item 
= (wxGenericTreeItem
*) itemId
.m_pItem
; 
1635     wxCHECK_RET( item
, _T("invalid item in wxGenericTreeCtrl::Expand") ); 
1636     wxCHECK_RET( !HasFlag(wxTR_HIDE_ROOT
) || itemId 
!= GetRootItem(), 
1637                  _T("can't expand hidden root") ); 
1639     if ( !item
->HasPlus() ) 
1642     if ( item
->IsExpanded() ) 
1645     wxTreeEvent 
event( wxEVT_COMMAND_TREE_ITEM_EXPANDING
, GetId() ); 
1646     event
.m_item 
= item
; 
1647     event
.SetEventObject( this ); 
1649     if ( ProcessEvent( event 
) && !event
.IsAllowed() ) 
1651         // cancelled by program 
1656     CalculatePositions(); 
1658     RefreshSubtree(item
); 
1660     event
.SetEventType(wxEVT_COMMAND_TREE_ITEM_EXPANDED
); 
1661     ProcessEvent( event 
); 
1664 void wxGenericTreeCtrl::ExpandAll(const wxTreeItemId
& item
) 
1666     if ( !HasFlag(wxTR_HIDE_ROOT
) || item 
!= GetRootItem()) 
1669         if ( !IsExpanded(item
) ) 
1673     wxTreeItemIdValue cookie
; 
1674     wxTreeItemId child 
= GetFirstChild(item
, cookie
); 
1675     while ( child
.IsOk() ) 
1679         child 
= GetNextChild(item
, cookie
); 
1683 void wxGenericTreeCtrl::Collapse(const wxTreeItemId
& itemId
) 
1685     wxCHECK_RET( !HasFlag(wxTR_HIDE_ROOT
) || itemId 
!= GetRootItem(), 
1686                  _T("can't collapse hidden root") ); 
1688     wxGenericTreeItem 
*item 
= (wxGenericTreeItem
*) itemId
.m_pItem
; 
1690     if ( !item
->IsExpanded() ) 
1693     wxTreeEvent 
event( wxEVT_COMMAND_TREE_ITEM_COLLAPSING
, GetId() ); 
1694     event
.m_item 
= item
; 
1695     event
.SetEventObject( this ); 
1696     if ( ProcessEvent( event 
) && !event
.IsAllowed() ) 
1698         // cancelled by program 
1702     ChildrenClosing(item
); 
1705 #if 0  // TODO why should items be collapsed recursively? 
1706     wxArrayGenericTreeItems
& children 
= item
->GetChildren(); 
1707     size_t count 
= children
.Count(); 
1708     for ( size_t n 
= 0; n 
< count
; n
++ ) 
1710         Collapse(children
[n
]); 
1714     CalculatePositions(); 
1716     RefreshSubtree(item
); 
1718     event
.SetEventType(wxEVT_COMMAND_TREE_ITEM_COLLAPSED
); 
1719     ProcessEvent( event 
); 
1722 void wxGenericTreeCtrl::CollapseAndReset(const wxTreeItemId
& item
) 
1725     DeleteChildren(item
); 
1728 void wxGenericTreeCtrl::Toggle(const wxTreeItemId
& itemId
) 
1730     wxGenericTreeItem 
*item 
= (wxGenericTreeItem
*) itemId
.m_pItem
; 
1732     if (item
->IsExpanded()) 
1738 void wxGenericTreeCtrl::Unselect() 
1742         m_current
->SetHilight( false ); 
1743         RefreshLine( m_current 
); 
1750 void wxGenericTreeCtrl::UnselectAllChildren(wxGenericTreeItem 
*item
) 
1752     if (item
->IsSelected()) 
1754         item
->SetHilight(false); 
1758     if (item
->HasChildren()) 
1760         wxArrayGenericTreeItems
& children 
= item
->GetChildren(); 
1761         size_t count 
= children
.Count(); 
1762         for ( size_t n 
= 0; n 
< count
; ++n 
) 
1764             UnselectAllChildren(children
[n
]); 
1769 void wxGenericTreeCtrl::UnselectAll() 
1771     wxTreeItemId rootItem 
= GetRootItem(); 
1773     // the tree might not have the root item at all 
1776         UnselectAllChildren((wxGenericTreeItem
*) rootItem
.m_pItem
); 
1780 // Recursive function ! 
1781 // To stop we must have crt_item<last_item 
1783 // Tag all next children, when no more children, 
1784 // Move to parent (not to tag) 
1785 // Keep going... if we found last_item, we stop. 
1786 bool wxGenericTreeCtrl::TagNextChildren(wxGenericTreeItem 
*crt_item
, wxGenericTreeItem 
*last_item
, bool select
) 
1788     wxGenericTreeItem 
*parent 
= crt_item
->GetParent(); 
1790     if (parent 
== NULL
) // This is root item 
1791         return TagAllChildrenUntilLast(crt_item
, last_item
, select
); 
1793     wxArrayGenericTreeItems
& children 
= parent
->GetChildren(); 
1794     int index 
= children
.Index(crt_item
); 
1795     wxASSERT( index 
!= wxNOT_FOUND 
); // I'm not a child of my parent? 
1797     size_t count 
= children
.Count(); 
1798     for (size_t n
=(size_t)(index
+1); n
<count
; ++n
) 
1800         if (TagAllChildrenUntilLast(children
[n
], last_item
, select
)) return true; 
1803     return TagNextChildren(parent
, last_item
, select
); 
1806 bool wxGenericTreeCtrl::TagAllChildrenUntilLast(wxGenericTreeItem 
*crt_item
, wxGenericTreeItem 
*last_item
, bool select
) 
1808     crt_item
->SetHilight(select
); 
1809     RefreshLine(crt_item
); 
1811     if (crt_item
==last_item
) 
1814     if (crt_item
->HasChildren()) 
1816         wxArrayGenericTreeItems
& children 
= crt_item
->GetChildren(); 
1817         size_t count 
= children
.Count(); 
1818         for ( size_t n 
= 0; n 
< count
; ++n 
) 
1820             if (TagAllChildrenUntilLast(children
[n
], last_item
, select
)) 
1828 void wxGenericTreeCtrl::SelectItemRange(wxGenericTreeItem 
*item1
, wxGenericTreeItem 
*item2
) 
1832     // item2 is not necessary after item1 
1833     // choice first' and 'last' between item1 and item2 
1834     wxGenericTreeItem 
*first
= (item1
->GetY()<item2
->GetY()) ? item1 
: item2
; 
1835     wxGenericTreeItem 
*last 
= (item1
->GetY()<item2
->GetY()) ? item2 
: item1
; 
1837     bool select 
= m_current
->IsSelected(); 
1839     if ( TagAllChildrenUntilLast(first
,last
,select
) ) 
1842     TagNextChildren(first
,last
,select
); 
1845 void wxGenericTreeCtrl::DoSelectItem(const wxTreeItemId
& itemId
, 
1846                                      bool unselect_others
, 
1847                                      bool extended_select
) 
1849     wxCHECK_RET( itemId
.IsOk(), wxT("invalid tree item") ); 
1853     bool is_single
=!(GetWindowStyleFlag() & wxTR_MULTIPLE
); 
1854     wxGenericTreeItem 
*item 
= (wxGenericTreeItem
*) itemId
.m_pItem
; 
1856     //wxCHECK_RET( ( (!unselect_others) && is_single), 
1857     //           wxT("this is a single selection tree") ); 
1859     // to keep going anyhow !!! 
1862         if (item
->IsSelected()) 
1863             return; // nothing to do 
1864         unselect_others 
= true; 
1865         extended_select 
= false; 
1867     else if ( unselect_others 
&& item
->IsSelected() ) 
1869         // selection change if there is more than one item currently selected 
1870         wxArrayTreeItemIds selected_items
; 
1871         if ( GetSelections(selected_items
) == 1 ) 
1875     wxTreeEvent 
event( wxEVT_COMMAND_TREE_SEL_CHANGING
, GetId() ); 
1876     event
.m_item 
= item
; 
1877     event
.m_itemOld 
= m_current
; 
1878     event
.SetEventObject( this ); 
1879     // TODO : Here we don't send any selection mode yet ! 
1881     if ( GetEventHandler()->ProcessEvent( event 
) && !event
.IsAllowed() ) 
1884     wxTreeItemId parent 
= GetItemParent( itemId 
); 
1885     while (parent
.IsOk()) 
1887         if (!IsExpanded(parent
)) 
1890         parent 
= GetItemParent( parent 
); 
1893     EnsureVisible( itemId 
); 
1896     if (unselect_others
) 
1898         if (is_single
) Unselect(); // to speed up thing 
1903     if (extended_select
) 
1907             m_current 
= m_key_current 
= (wxGenericTreeItem
*) GetRootItem().m_pItem
; 
1910         // don't change the mark (m_current) 
1911         SelectItemRange(m_current
, item
); 
1915         bool select 
= true; // the default 
1917         // Check if we need to toggle hilight (ctrl mode) 
1918         if (!unselect_others
) 
1919             select
=!item
->IsSelected(); 
1921         m_current 
= m_key_current 
= item
; 
1922         m_current
->SetHilight(select
); 
1923         RefreshLine( m_current 
); 
1926     event
.SetEventType(wxEVT_COMMAND_TREE_SEL_CHANGED
); 
1927     GetEventHandler()->ProcessEvent( event 
); 
1930 void wxGenericTreeCtrl::SelectItem(const wxTreeItemId
& itemId
, bool select
) 
1934         DoSelectItem(itemId
); 
1938         wxGenericTreeItem 
*item 
= (wxGenericTreeItem
*) itemId
.m_pItem
; 
1939         wxCHECK_RET( item
, wxT("SelectItem(): invalid tree item") ); 
1941         item
->SetHilight(false); 
1946 void wxGenericTreeCtrl::FillArray(wxGenericTreeItem 
*item
, 
1947                                   wxArrayTreeItemIds 
&array
) const 
1949     if ( item
->IsSelected() ) 
1950         array
.Add(wxTreeItemId(item
)); 
1952     if ( item
->HasChildren() ) 
1954         wxArrayGenericTreeItems
& children 
= item
->GetChildren(); 
1955         size_t count 
= children
.GetCount(); 
1956         for ( size_t n 
= 0; n 
< count
; ++n 
) 
1957             FillArray(children
[n
], array
); 
1961 size_t wxGenericTreeCtrl::GetSelections(wxArrayTreeItemIds 
&array
) const 
1964     wxTreeItemId idRoot 
= GetRootItem(); 
1965     if ( idRoot
.IsOk() ) 
1967         FillArray((wxGenericTreeItem
*) idRoot
.m_pItem
, array
); 
1969     //else: the tree is empty, so no selections 
1971     return array
.Count(); 
1974 void wxGenericTreeCtrl::EnsureVisible(const wxTreeItemId
& item
) 
1976     wxCHECK_RET( item
.IsOk(), wxT("invalid tree item") ); 
1978     if (!item
.IsOk()) return; 
1980     wxGenericTreeItem 
*gitem 
= (wxGenericTreeItem
*) item
.m_pItem
; 
1982     // first expand all parent branches 
1983     wxGenericTreeItem 
*parent 
= gitem
->GetParent(); 
1985     if ( HasFlag(wxTR_HIDE_ROOT
) ) 
1987         while ( parent 
&& parent 
!= m_anchor 
) 
1990             parent 
= parent
->GetParent(); 
1998             parent 
= parent
->GetParent(); 
2002     //if (parent) CalculatePositions(); 
2007 void wxGenericTreeCtrl::ScrollTo(const wxTreeItemId 
&item
) 
2009     if (!item
.IsOk()) return; 
2011     // We have to call this here because the label in 
2012     // question might just have been added and no screen 
2013     // update taken place. 
2015 #if defined( __WXMSW__ ) || defined(__WXMAC__) 
2020     wxGenericTreeItem 
*gitem 
= (wxGenericTreeItem
*) item
.m_pItem
; 
2022     // now scroll to the item 
2023     int item_y 
= gitem
->GetY(); 
2027     GetViewStart( &start_x
, &start_y 
); 
2028     start_y 
*= PIXELS_PER_UNIT
; 
2032     GetClientSize( &client_w
, &client_h 
); 
2034     if (item_y 
< start_y
+3) 
2039         m_anchor
->GetSize( x
, y
, this ); 
2040         y 
+= PIXELS_PER_UNIT
+2; // one more scrollbar unit + 2 pixels 
2041         x 
+= PIXELS_PER_UNIT
+2; // one more scrollbar unit + 2 pixels 
2042         int x_pos 
= GetScrollPos( wxHORIZONTAL 
); 
2043         // Item should appear at top 
2044         SetScrollbars( PIXELS_PER_UNIT
, PIXELS_PER_UNIT
, x
/PIXELS_PER_UNIT
, y
/PIXELS_PER_UNIT
, x_pos
, item_y
/PIXELS_PER_UNIT 
); 
2046     else if (item_y
+GetLineHeight(gitem
) > start_y
+client_h
) 
2051         m_anchor
->GetSize( x
, y
, this ); 
2052         y 
+= PIXELS_PER_UNIT
+2; // one more scrollbar unit + 2 pixels 
2053         x 
+= PIXELS_PER_UNIT
+2; // one more scrollbar unit + 2 pixels 
2054         item_y 
+= PIXELS_PER_UNIT
+2; 
2055         int x_pos 
= GetScrollPos( wxHORIZONTAL 
); 
2056         // Item should appear at bottom 
2057         SetScrollbars( PIXELS_PER_UNIT
, PIXELS_PER_UNIT
, x
/PIXELS_PER_UNIT
, y
/PIXELS_PER_UNIT
, x_pos
, (item_y
+GetLineHeight(gitem
)-client_h
)/PIXELS_PER_UNIT 
); 
2061 // FIXME: tree sorting functions are not reentrant and not MT-safe! 
2062 static wxGenericTreeCtrl 
*s_treeBeingSorted 
= NULL
; 
2064 static int LINKAGEMODE 
tree_ctrl_compare_func(wxGenericTreeItem 
**item1
, 
2065                                   wxGenericTreeItem 
**item2
) 
2067     wxCHECK_MSG( s_treeBeingSorted
, 0, wxT("bug in wxGenericTreeCtrl::SortChildren()") ); 
2069     return s_treeBeingSorted
->OnCompareItems(*item1
, *item2
); 
2072 int wxGenericTreeCtrl::OnCompareItems(const wxTreeItemId
& item1
, 
2073                                const wxTreeItemId
& item2
) 
2075     return wxStrcmp(GetItemText(item1
), GetItemText(item2
)); 
2078 void wxGenericTreeCtrl::SortChildren(const wxTreeItemId
& itemId
) 
2080     wxCHECK_RET( itemId
.IsOk(), wxT("invalid tree item") ); 
2082     wxGenericTreeItem 
*item 
= (wxGenericTreeItem
*) itemId
.m_pItem
; 
2084     wxCHECK_RET( !s_treeBeingSorted
, 
2085                  wxT("wxGenericTreeCtrl::SortChildren is not reentrant") ); 
2087     wxArrayGenericTreeItems
& children 
= item
->GetChildren(); 
2088     if ( children
.Count() > 1 ) 
2092         s_treeBeingSorted 
= this; 
2093         children
.Sort(tree_ctrl_compare_func
); 
2094         s_treeBeingSorted 
= NULL
; 
2096     //else: don't make the tree dirty as nothing changed 
2099 wxImageList 
*wxGenericTreeCtrl::GetImageList() const 
2101     return m_imageListNormal
; 
2104 wxImageList 
*wxGenericTreeCtrl::GetButtonsImageList() const 
2106     return m_imageListButtons
; 
2109 wxImageList 
*wxGenericTreeCtrl::GetStateImageList() const 
2111     return m_imageListState
; 
2114 void wxGenericTreeCtrl::CalculateLineHeight() 
2116     wxClientDC 
dc(this); 
2117     m_lineHeight 
= (int)(dc
.GetCharHeight() + 4); 
2119     if ( m_imageListNormal 
) 
2121         // Calculate a m_lineHeight value from the normal Image sizes. 
2122         // May be toggle off. Then wxGenericTreeCtrl will spread when 
2123         // necessary (which might look ugly). 
2124         int n 
= m_imageListNormal
->GetImageCount(); 
2125         for (int i 
= 0; i 
< n 
; i
++) 
2127             int width 
= 0, height 
= 0; 
2128             m_imageListNormal
->GetSize(i
, width
, height
); 
2129             if (height 
> m_lineHeight
) m_lineHeight 
= height
; 
2133     if (m_imageListButtons
) 
2135         // Calculate a m_lineHeight value from the Button image sizes. 
2136         // May be toggle off. Then wxGenericTreeCtrl will spread when 
2137         // necessary (which might look ugly). 
2138         int n 
= m_imageListButtons
->GetImageCount(); 
2139         for (int i 
= 0; i 
< n 
; i
++) 
2141             int width 
= 0, height 
= 0; 
2142             m_imageListButtons
->GetSize(i
, width
, height
); 
2143             if (height 
> m_lineHeight
) m_lineHeight 
= height
; 
2147     if (m_lineHeight 
< 30) 
2148         m_lineHeight 
+= 2;                 // at least 2 pixels 
2150         m_lineHeight 
+= m_lineHeight
/10;   // otherwise 10% extra spacing 
2153 void wxGenericTreeCtrl::SetImageList(wxImageList 
*imageList
) 
2155     if (m_ownsImageListNormal
) delete m_imageListNormal
; 
2156     m_imageListNormal 
= imageList
; 
2157     m_ownsImageListNormal 
= false; 
2159     // Don't do any drawing if we're setting the list to NULL, 
2160     // since we may be in the process of deleting the tree control. 
2162         CalculateLineHeight(); 
2165 void wxGenericTreeCtrl::SetStateImageList(wxImageList 
*imageList
) 
2167     if (m_ownsImageListState
) delete m_imageListState
; 
2168     m_imageListState 
= imageList
; 
2169     m_ownsImageListState 
= false; 
2172 void wxGenericTreeCtrl::SetButtonsImageList(wxImageList 
*imageList
) 
2174     if (m_ownsImageListButtons
) delete m_imageListButtons
; 
2175     m_imageListButtons 
= imageList
; 
2176     m_ownsImageListButtons 
= false; 
2178     CalculateLineHeight(); 
2181 void wxGenericTreeCtrl::AssignImageList(wxImageList 
*imageList
) 
2183     SetImageList(imageList
); 
2184     m_ownsImageListNormal 
= true; 
2187 void wxGenericTreeCtrl::AssignStateImageList(wxImageList 
*imageList
) 
2189     SetStateImageList(imageList
); 
2190     m_ownsImageListState 
= true; 
2193 void wxGenericTreeCtrl::AssignButtonsImageList(wxImageList 
*imageList
) 
2195     SetButtonsImageList(imageList
); 
2196     m_ownsImageListButtons 
= true; 
2199 // ----------------------------------------------------------------------------- 
2201 // ----------------------------------------------------------------------------- 
2203 void wxGenericTreeCtrl::AdjustMyScrollbars() 
2208         m_anchor
->GetSize( x
, y
, this ); 
2209         y 
+= PIXELS_PER_UNIT
+2; // one more scrollbar unit + 2 pixels 
2210         x 
+= PIXELS_PER_UNIT
+2; // one more scrollbar unit + 2 pixels 
2211         int x_pos 
= GetScrollPos( wxHORIZONTAL 
); 
2212         int y_pos 
= GetScrollPos( wxVERTICAL 
); 
2213         SetScrollbars( PIXELS_PER_UNIT
, PIXELS_PER_UNIT
, x
/PIXELS_PER_UNIT
, y
/PIXELS_PER_UNIT
, x_pos
, y_pos 
); 
2217         SetScrollbars( 0, 0, 0, 0 ); 
2221 int wxGenericTreeCtrl::GetLineHeight(wxGenericTreeItem 
*item
) const 
2223     if (GetWindowStyleFlag() & wxTR_HAS_VARIABLE_ROW_HEIGHT
) 
2224         return item
->GetHeight(); 
2226         return m_lineHeight
; 
2229 void wxGenericTreeCtrl::PaintItem(wxGenericTreeItem 
*item
, wxDC
& dc
) 
2231     // TODO implement "state" icon on items 
2233     wxTreeItemAttr 
*attr 
= item
->GetAttributes(); 
2234     if ( attr 
&& attr
->HasFont() ) 
2235         dc
.SetFont(attr
->GetFont()); 
2236     else if (item
->IsBold()) 
2237         dc
.SetFont(m_boldFont
); 
2239     long text_w 
= 0, text_h 
= 0; 
2240     dc
.GetTextExtent( item
->GetText(), &text_w
, &text_h 
); 
2242     int image_h 
= 0, image_w 
= 0; 
2243     int image 
= item
->GetCurrentImage(); 
2244     if ( image 
!= NO_IMAGE 
) 
2246         if ( m_imageListNormal 
) 
2248             m_imageListNormal
->GetSize( image
, image_w
, image_h 
); 
2257     int total_h 
= GetLineHeight(item
); 
2259     if ( item
->IsSelected() ) 
2261 // under mac selections are only a rectangle in case they don't have the focus 
2265             dc
.SetBrush( *wxTRANSPARENT_BRUSH 
) ; 
2266             dc
.SetPen( wxPen( wxSystemSettings::GetColour( wxSYS_COLOUR_HIGHLIGHT 
) , 1 , wxSOLID 
) ) ; 
2270             dc
.SetBrush( *m_hilightBrush 
) ; 
2273         dc
.SetBrush(*(m_hasFocus 
? m_hilightBrush 
: m_hilightUnfocusedBrush
)); 
2279         if ( attr 
&& attr
->HasBackgroundColour() ) 
2280             colBg 
= attr
->GetBackgroundColour(); 
2282             colBg 
= m_backgroundColour
; 
2283         dc
.SetBrush(wxBrush(colBg
, wxSOLID
)); 
2286     int offset 
= HasFlag(wxTR_ROW_LINES
) ? 1 : 0; 
2288     if ( HasFlag(wxTR_FULL_ROW_HIGHLIGHT
) ) 
2292         DoGetPosition(&x
, &y
); 
2294         dc
.DrawRectangle(x
, item
->GetY()+offset
, w
, total_h
-offset
); 
2298         if ( item
->IsSelected() && image 
!= NO_IMAGE 
) 
2300             // If it's selected, and there's an image, then we should 
2301             // take care to leave the area under the image painted in the 
2302             // background colour. 
2303             dc
.DrawRectangle( item
->GetX() + image_w 
- 2, item
->GetY()+offset
, 
2304                               item
->GetWidth() - image_w 
+ 2, total_h
-offset 
); 
2308             dc
.DrawRectangle( item
->GetX()-2, item
->GetY()+offset
, 
2309                               item
->GetWidth()+2, total_h
-offset 
); 
2313     if ( image 
!= NO_IMAGE 
) 
2315         dc
.SetClippingRegion( item
->GetX(), item
->GetY(), image_w
-2, total_h 
); 
2316         m_imageListNormal
->Draw( image
, dc
, 
2318                                  item
->GetY() +((total_h 
> image_h
)?((total_h
-image_h
)/2):0), 
2319                                  wxIMAGELIST_DRAW_TRANSPARENT 
); 
2320         dc
.DestroyClippingRegion(); 
2323     dc
.SetBackgroundMode(wxTRANSPARENT
); 
2324     int extraH 
= (total_h 
> text_h
) ? (total_h 
- text_h
)/2 : 0; 
2325     dc
.DrawText( item
->GetText(), 
2326                  (wxCoord
)(image_w 
+ item
->GetX()), 
2327                  (wxCoord
)(item
->GetY() + extraH
)); 
2329     // restore normal font 
2330     dc
.SetFont( m_normalFont 
); 
2333 // Now y stands for the top of the item, whereas it used to stand for middle ! 
2334 void wxGenericTreeCtrl::PaintLevel( wxGenericTreeItem 
*item
, wxDC 
&dc
, int level
, int &y 
) 
2336     int x 
= level
*m_indent
; 
2337     if (!HasFlag(wxTR_HIDE_ROOT
)) 
2341     else if (level 
== 0) 
2343         // always expand hidden root 
2345         wxArrayGenericTreeItems
& children 
= item
->GetChildren(); 
2346         int count 
= children
.Count(); 
2352                 PaintLevel(children
[n
], dc
, 1, y
); 
2353             } while (++n 
< count
); 
2355             if (!HasFlag(wxTR_NO_LINES
) && HasFlag(wxTR_LINES_AT_ROOT
) && count 
> 0) 
2357                 // draw line down to last child 
2358                 origY 
+= GetLineHeight(children
[0])>>1; 
2359                 oldY 
+= GetLineHeight(children
[n
-1])>>1; 
2360                 dc
.DrawLine(3, origY
, 3, oldY
); 
2366     item
->SetX(x
+m_spacing
); 
2369     int h 
= GetLineHeight(item
); 
2371     int y_mid 
= y_top 
+ (h
>>1); 
2374     int exposed_x 
= dc
.LogicalToDeviceX(0); 
2375     int exposed_y 
= dc
.LogicalToDeviceY(y_top
); 
2377     if (IsExposed(exposed_x
, exposed_y
, 10000, h
))  // 10000 = very much 
2381             // don't draw rect outline if we already have the 
2382             // background color under Mac 
2383             (item
->IsSelected() && m_hasFocus
) ? wxBLACK_PEN 
: 
2384 #endif // !__WXMAC__ 
2388         if ( item
->IsSelected() ) 
2390             colText 
= wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHTTEXT
); 
2394             wxTreeItemAttr 
*attr 
= item
->GetAttributes(); 
2395             if (attr 
&& attr
->HasTextColour()) 
2396                 colText 
= attr
->GetTextColour(); 
2398                 colText 
= GetForegroundColour(); 
2402         dc
.SetTextForeground(colText
); 
2406         PaintItem(item
, dc
); 
2408         if (HasFlag(wxTR_ROW_LINES
)) 
2410             // if the background colour is white, choose a 
2411             // contrasting color for the lines 
2412             dc
.SetPen(*((GetBackgroundColour() == *wxWHITE
) 
2413                          ? wxMEDIUM_GREY_PEN 
: wxWHITE_PEN
)); 
2414             dc
.DrawLine(0, y_top
, 10000, y_top
); 
2415             dc
.DrawLine(0, y
, 10000, y
); 
2418         // restore DC objects 
2419         dc
.SetBrush(*wxWHITE_BRUSH
); 
2420         dc
.SetPen(m_dottedPen
); 
2421         dc
.SetTextForeground(*wxBLACK
); 
2423         if ( !HasFlag(wxTR_NO_LINES
) ) 
2425             // draw the horizontal line here 
2427             if (x 
> (signed)m_indent
) 
2428                 x_start 
-= m_indent
; 
2429             else if (HasFlag(wxTR_LINES_AT_ROOT
)) 
2431             dc
.DrawLine(x_start
, y_mid
, x 
+ m_spacing
, y_mid
); 
2434         // should the item show a button? 
2435         if ( item
->HasPlus() && HasButtons() ) 
2437             if ( m_imageListButtons 
) 
2439                 // draw the image button here 
2442                 int image 
= item
->IsExpanded() ? wxTreeItemIcon_Expanded
 
2443                                                : wxTreeItemIcon_Normal
; 
2444                 if ( item
->IsSelected() ) 
2445                     image 
+= wxTreeItemIcon_Selected 
- wxTreeItemIcon_Normal
; 
2447                 m_imageListButtons
->GetSize(image
, image_w
, image_h
); 
2448                 int xx 
= x 
- image_w
/2; 
2449                 int yy 
= y_mid 
- image_h
/2; 
2451                 wxDCClipper 
clip(dc
, xx
, yy
, image_w
, image_h
); 
2452                 m_imageListButtons
->Draw(image
, dc
, xx
, yy
, 
2453                                          wxIMAGELIST_DRAW_TRANSPARENT
); 
2455             else // no custom buttons 
2457                 static const int wImage 
= 9; 
2458                 static const int hImage 
= 9; 
2461                 if (item
->IsExpanded()) 
2462                     flag 
|= wxCONTROL_EXPANDED
; 
2463                 if (item 
== m_underMouse
) 
2464                     flag 
|= wxCONTROL_CURRENT
; 
2466                 wxRendererNative::Get().DrawTreeItemButton
 
2470                                             wxRect(x 
- wImage
/2, 
2479     if (item
->IsExpanded()) 
2481         wxArrayGenericTreeItems
& children 
= item
->GetChildren(); 
2482         int count 
= children
.Count(); 
2489                 PaintLevel(children
[n
], dc
, level
, y
); 
2490             } while (++n 
< count
); 
2492             if (!HasFlag(wxTR_NO_LINES
) && count 
> 0) 
2494                 // draw line down to last child 
2495                 oldY 
+= GetLineHeight(children
[n
-1])>>1; 
2496                 if (HasButtons()) y_mid 
+= 5; 
2498                 // Only draw the portion of the line that is visible, in case it is huge 
2499                 wxCoord xOrigin
=0, yOrigin
=0, width
, height
; 
2500                 dc
.GetDeviceOrigin(&xOrigin
, &yOrigin
); 
2501                 yOrigin 
= abs(yOrigin
); 
2502                 GetClientSize(&width
, &height
); 
2504                 // Move end points to the begining/end of the view? 
2505                 if (y_mid 
< yOrigin
) 
2507                 if (oldY 
> yOrigin 
+ height
) 
2508                     oldY 
= yOrigin 
+ height
; 
2510                 // after the adjustments if y_mid is larger than oldY then the line 
2511                 // isn't visible at all so don't draw anything 
2513                     dc
.DrawLine(x
, y_mid
, x
, oldY
); 
2519 void wxGenericTreeCtrl::DrawDropEffect(wxGenericTreeItem 
*item
) 
2523         if ( item
->HasPlus() ) 
2525             // it's a folder, indicate it by a border 
2530             // draw a line under the drop target because the item will be 
2532             DrawLine(item
, !m_dropEffectAboveItem 
); 
2535         SetCursor(wxCURSOR_BULLSEYE
); 
2540         SetCursor(wxCURSOR_NO_ENTRY
); 
2544 void wxGenericTreeCtrl::DrawBorder(const wxTreeItemId 
&item
) 
2546     wxCHECK_RET( item
.IsOk(), _T("invalid item in wxGenericTreeCtrl::DrawLine") ); 
2548     wxGenericTreeItem 
*i 
= (wxGenericTreeItem
*) item
.m_pItem
; 
2550     wxClientDC 
dc(this); 
2552     dc
.SetLogicalFunction(wxINVERT
); 
2553     dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
2555     int w 
= i
->GetWidth() + 2; 
2556     int h 
= GetLineHeight(i
) + 2; 
2558     dc
.DrawRectangle( i
->GetX() - 1, i
->GetY() - 1, w
, h
); 
2561 void wxGenericTreeCtrl::DrawLine(const wxTreeItemId 
&item
, bool below
) 
2563     wxCHECK_RET( item
.IsOk(), _T("invalid item in wxGenericTreeCtrl::DrawLine") ); 
2565     wxGenericTreeItem 
*i 
= (wxGenericTreeItem
*) item
.m_pItem
; 
2567     wxClientDC 
dc(this); 
2569     dc
.SetLogicalFunction(wxINVERT
); 
2575         y 
+= GetLineHeight(i
) - 1; 
2578     dc
.DrawLine( x
, y
, x 
+ i
->GetWidth(), y
); 
2581 // ----------------------------------------------------------------------------- 
2582 // wxWidgets callbacks 
2583 // ----------------------------------------------------------------------------- 
2585 void wxGenericTreeCtrl::OnPaint( wxPaintEvent 
&WXUNUSED(event
) ) 
2593     dc
.SetFont( m_normalFont 
); 
2594     dc
.SetPen( m_dottedPen 
); 
2596     // this is now done dynamically 
2597     //if(GetImageList() == NULL) 
2598     // m_lineHeight = (int)(dc.GetCharHeight() + 4); 
2601     PaintLevel( m_anchor
, dc
, 0, y 
); 
2604 void wxGenericTreeCtrl::OnSetFocus( wxFocusEvent 
&event 
) 
2613 void wxGenericTreeCtrl::OnKillFocus( wxFocusEvent 
&event 
) 
2622 void wxGenericTreeCtrl::OnChar( wxKeyEvent 
&event 
) 
2624     wxTreeEvent 
te( wxEVT_COMMAND_TREE_KEY_DOWN
, GetId() ); 
2625     te
.m_evtKey 
= event
; 
2626     te
.SetEventObject( this ); 
2627     if ( GetEventHandler()->ProcessEvent( te 
) ) 
2629         // intercepted by the user code 
2633     if ( (m_current 
== 0) || (m_key_current 
== 0) ) 
2639     // how should the selection work for this event? 
2640     bool is_multiple
, extended_select
, unselect_others
; 
2641     EventFlagsToSelType(GetWindowStyleFlag(), 
2644                         is_multiple
, extended_select
, unselect_others
); 
2648     // * : Expand all/Collapse all 
2649     // ' ' | return : activate 
2650     // up    : go up (not last children!) 
2652     // left  : go to parent 
2653     // right : open if parent and go next 
2654     // home  : go to root 
2655     // end   : go to last item without opening parents 
2656     // alnum : start or continue searching for the item with this prefix 
2657     int keyCode 
= event
.GetKeyCode(); 
2662             if (m_current
->HasPlus() && !IsExpanded(m_current
)) 
2670             if ( !IsExpanded(m_current
) ) 
2673                 ExpandAll(m_current
); 
2676             //else: fall through to Collapse() it 
2680             if (IsExpanded(m_current
)) 
2682                 Collapse(m_current
); 
2688                 wxTreeEvent 
event( wxEVT_COMMAND_TREE_ITEM_MENU
, GetId() ); 
2689                 event
.m_item 
= m_current
; 
2690                 event
.SetEventObject( this ); 
2691                 GetEventHandler()->ProcessEvent( event 
); 
2696             if ( !event
.HasModifiers() ) 
2698                 wxTreeEvent 
event( wxEVT_COMMAND_TREE_ITEM_ACTIVATED
, GetId() ); 
2699                 event
.m_item 
= m_current
; 
2700                 event
.SetEventObject( this ); 
2701                 GetEventHandler()->ProcessEvent( event 
); 
2704             // in any case, also generate the normal key event for this key, 
2705             // even if we generated the ACTIVATED event above: this is what 
2706             // wxMSW does and it makes sense because you might not want to 
2707             // process ACTIVATED event at all and handle Space and Return 
2708             // directly (and differently) which would be impossible otherwise 
2712             // up goes to the previous sibling or to the last 
2713             // of its children if it's expanded 
2716                 wxTreeItemId prev 
= GetPrevSibling( m_key_current 
); 
2719                     prev 
= GetItemParent( m_key_current 
); 
2720                     if ((prev 
== GetRootItem()) && HasFlag(wxTR_HIDE_ROOT
)) 
2722                         break;  // don't go to root if it is hidden 
2726                         wxTreeItemIdValue cookie
; 
2727                         wxTreeItemId current 
= m_key_current
; 
2728                         // TODO: Huh?  If we get here, we'd better be the first child of our parent.  How else could it be? 
2729                         if (current 
== GetFirstChild( prev
, cookie 
)) 
2731                             // otherwise we return to where we came from 
2732                             DoSelectItem( prev
, unselect_others
, extended_select 
); 
2733                             m_key_current
= (wxGenericTreeItem
*) prev
.m_pItem
; 
2740                     while ( IsExpanded(prev
) && HasChildren(prev
) ) 
2742                         wxTreeItemId child 
= GetLastChild(prev
); 
2749                     DoSelectItem( prev
, unselect_others
, extended_select 
); 
2750                     m_key_current
=(wxGenericTreeItem
*) prev
.m_pItem
; 
2755             // left arrow goes to the parent 
2758                 wxTreeItemId prev 
= GetItemParent( m_current 
); 
2759                 if ((prev 
== GetRootItem()) && HasFlag(wxTR_HIDE_ROOT
)) 
2761                     // don't go to root if it is hidden 
2762                     prev 
= GetPrevSibling( m_current 
); 
2766                     DoSelectItem( prev
, unselect_others
, extended_select 
); 
2772             // this works the same as the down arrow except that we 
2773             // also expand the item if it wasn't expanded yet 
2779                 if (IsExpanded(m_key_current
) && HasChildren(m_key_current
)) 
2781                     wxTreeItemIdValue cookie
; 
2782                     wxTreeItemId child 
= GetFirstChild( m_key_current
, cookie 
); 
2783                     DoSelectItem( child
, unselect_others
, extended_select 
); 
2784                     m_key_current
=(wxGenericTreeItem
*) child
.m_pItem
; 
2788                     wxTreeItemId next 
= GetNextSibling( m_key_current 
); 
2791                         wxTreeItemId current 
= m_key_current
; 
2792                         while (current
.IsOk() && !next
) 
2794                             current 
= GetItemParent( current 
); 
2795                             if (current
) next 
= GetNextSibling( current 
); 
2800                         DoSelectItem( next
, unselect_others
, extended_select 
); 
2801                         m_key_current
=(wxGenericTreeItem
*) next
.m_pItem
; 
2807             // <End> selects the last visible tree item 
2810                 wxTreeItemId last 
= GetRootItem(); 
2812                 while ( last
.IsOk() && IsExpanded(last
) ) 
2814                     wxTreeItemId lastChild 
= GetLastChild(last
); 
2816                     // it may happen if the item was expanded but then all of 
2817                     // its children have been deleted - so IsExpanded() returned 
2818                     // true, but GetLastChild() returned invalid item 
2827                     DoSelectItem( last
, unselect_others
, extended_select 
); 
2832             // <Home> selects the root item 
2835                 wxTreeItemId prev 
= GetRootItem(); 
2839                 if ( HasFlag(wxTR_HIDE_ROOT
) ) 
2841                     wxTreeItemIdValue cookie
; 
2842                     prev 
= GetFirstChild(prev
, cookie
); 
2847                 DoSelectItem( prev
, unselect_others
, extended_select 
); 
2852             // do not use wxIsalnum() here 
2853             if ( !event
.HasModifiers() && 
2854                  ((keyCode 
>= '0' && keyCode 
<= '9') || 
2855                   (keyCode 
>= 'a' && keyCode 
<= 'z') || 
2856                   (keyCode 
>= 'A' && keyCode 
<= 'Z' ))) 
2858                 // find the next item starting with the given prefix 
2859                 wxChar ch 
= (wxChar
)keyCode
; 
2861                 wxTreeItemId id 
= FindItem(m_current
, m_findPrefix 
+ ch
); 
2872                 // also start the timer to reset the current prefix if the user 
2873                 // doesn't press any more alnum keys soon -- we wouldn't want 
2874                 // to use this prefix for a new item search 
2877                     m_findTimer 
= new wxTreeFindTimer(this); 
2880                 m_findTimer
->Start(wxTreeFindTimer::DELAY
, wxTIMER_ONE_SHOT
); 
2889 wxTreeItemId 
wxGenericTreeCtrl::HitTest(const wxPoint
& point
, int& flags
) 
2891     // JACS: removed wxYieldIfNeeded() because it can cause the window 
2892     // to be deleted from under us if a close window event is pending 
2897     if (point
.x
<0) flags 
|= wxTREE_HITTEST_TOLEFT
; 
2898     if (point
.x
>w
) flags 
|= wxTREE_HITTEST_TORIGHT
; 
2899     if (point
.y
<0) flags 
|= wxTREE_HITTEST_ABOVE
; 
2900     if (point
.y
>h
) flags 
|= wxTREE_HITTEST_BELOW
; 
2901     if (flags
) return wxTreeItemId(); 
2903     if (m_anchor 
== NULL
) 
2905         flags 
= wxTREE_HITTEST_NOWHERE
; 
2906         return wxTreeItemId(); 
2909     wxGenericTreeItem 
*hit 
=  m_anchor
->HitTest(CalcUnscrolledPosition(point
), 
2913         flags 
= wxTREE_HITTEST_NOWHERE
; 
2914         return wxTreeItemId(); 
2919 // get the bounding rectangle of the item (or of its label only) 
2920 bool wxGenericTreeCtrl::GetBoundingRect(const wxTreeItemId
& item
, 
2922                                         bool WXUNUSED(textOnly
)) const 
2924     wxCHECK_MSG( item
.IsOk(), false, _T("invalid item in wxGenericTreeCtrl::GetBoundingRect") ); 
2926     wxGenericTreeItem 
*i 
= (wxGenericTreeItem
*) item
.m_pItem
; 
2929     GetViewStart(& startX
, & startY
); 
2931     rect
.x 
= i
->GetX() - startX
*PIXELS_PER_UNIT
; 
2932     rect
.y 
= i
->GetY() - startY
*PIXELS_PER_UNIT
; 
2933     rect
.width 
= i
->GetWidth(); 
2934     //rect.height = i->GetHeight(); 
2935     rect
.height 
= GetLineHeight(i
); 
2940 void wxGenericTreeCtrl::Edit( const wxTreeItemId
& item 
) 
2942     wxCHECK_RET( item
.IsOk(), _T("can't edit an invalid item") ); 
2944     wxGenericTreeItem 
*itemEdit 
= (wxGenericTreeItem 
*)item
.m_pItem
; 
2946     wxTreeEvent 
te( wxEVT_COMMAND_TREE_BEGIN_LABEL_EDIT
, GetId() ); 
2947     te
.m_item 
= itemEdit
; 
2948     te
.SetEventObject( this ); 
2949     if ( GetEventHandler()->ProcessEvent( te 
) && !te
.IsAllowed() ) 
2955     // We have to call this here because the label in 
2956     // question might just have been added and no screen 
2957     // update taken place. 
2959 #if defined( __WXMSW__ ) || defined(__WXMAC__) 
2965     m_textCtrl 
= new wxTreeTextCtrl(this, itemEdit
); 
2967     m_textCtrl
->SetFocus(); 
2970 // returns a pointer to the text edit control if the item is being 
2971 // edited, NULL otherwise (it's assumed that no more than one item may 
2972 // be edited simultaneously) 
2973 wxTextCtrl
* wxGenericTreeCtrl::GetEditControl() const 
2978 bool wxGenericTreeCtrl::OnRenameAccept(wxGenericTreeItem 
*item
, 
2979                                        const wxString
& value
) 
2981     wxTreeEvent 
le( wxEVT_COMMAND_TREE_END_LABEL_EDIT
, GetId() ); 
2983     le
.SetEventObject( this ); 
2985     le
.m_editCancelled 
= false; 
2987     return !GetEventHandler()->ProcessEvent( le 
) || le
.IsAllowed(); 
2990 void wxGenericTreeCtrl::OnRenameCancelled(wxGenericTreeItem 
*item
) 
2992     // let owner know that the edit was cancelled 
2993     wxTreeEvent 
le( wxEVT_COMMAND_TREE_END_LABEL_EDIT
, GetId() ); 
2995     le
.SetEventObject( this ); 
2996     le
.m_label 
= wxEmptyString
; 
2997     le
.m_editCancelled 
= true; 
2999     GetEventHandler()->ProcessEvent( le 
); 
3002 void wxGenericTreeCtrl::OnRenameTimer() 
3007 void wxGenericTreeCtrl::OnMouse( wxMouseEvent 
&event 
) 
3009     if ( !m_anchor 
) return; 
3011     wxPoint pt 
= CalcUnscrolledPosition(event
.GetPosition()); 
3013     // Is the mouse over a tree item button? 
3015     wxGenericTreeItem 
*thisItem 
= m_anchor
->HitTest(pt
, this, flags
, 0); 
3016     wxGenericTreeItem 
*underMouse 
= thisItem
; 
3018     bool underMouseChanged 
= (underMouse 
!= m_underMouse
) ; 
3019 #endif // wxUSE_TOOLTIPS 
3022         (flags 
& wxTREE_HITTEST_ONITEMBUTTON
) && 
3023         (!event
.LeftIsDown()) && 
3025         (!m_renameTimer 
|| !m_renameTimer
->IsRunning())) 
3033     if (underMouse 
!= m_underMouse
) 
3037             // unhighlight old item 
3038             wxGenericTreeItem 
*tmp 
= m_underMouse
; 
3039             m_underMouse 
= NULL
; 
3043          m_underMouse 
= underMouse
; 
3045             RefreshLine( m_underMouse 
); 
3049     // Determines what item we are hovering over and need a tooltip for 
3050     wxTreeItemId hoverItem 
= thisItem
; 
3052     // We do not want a tooltip if we are dragging, or if the rename timer is running 
3053     if (underMouseChanged 
&& hoverItem
.IsOk() && !m_isDragging 
&& (!m_renameTimer 
|| !m_renameTimer
->IsRunning())) 
3055         // Ask the tree control what tooltip (if any) should be shown 
3056         wxTreeEvent 
hevent(wxEVT_COMMAND_TREE_ITEM_GETTOOLTIP
, GetId()); 
3057         hevent
.m_item 
= hoverItem
; 
3058         hevent
.SetEventObject(this); 
3060         if ( GetEventHandler()->ProcessEvent(hevent
) && hevent
.IsAllowed() ) 
3062             SetToolTip(hevent
.m_label
); 
3067     // we process left mouse up event (enables in-place edit), right down 
3068     // (pass to the user code), left dbl click (activate item) and 
3069     // dragging/moving events for items drag-and-drop 
3070     if ( !(event
.LeftDown() || 
3072            event
.RightDown() || 
3073            event
.LeftDClick() || 
3075            ((event
.Moving() || event
.RightUp()) && m_isDragging
)) ) 
3084     wxGenericTreeItem 
*item 
= m_anchor
->HitTest(pt
, this, flags
, 0); 
3086     if ( event
.Dragging() && !m_isDragging 
) 
3088         if (m_dragCount 
== 0) 
3093         if (m_dragCount 
!= 3) 
3095             // wait until user drags a bit further... 
3099         wxEventType command 
= event
.RightIsDown() 
3100                               ? wxEVT_COMMAND_TREE_BEGIN_RDRAG
 
3101                               : wxEVT_COMMAND_TREE_BEGIN_DRAG
; 
3103         wxTreeEvent 
nevent( command
, GetId() ); 
3104         nevent
.m_item 
= m_current
; 
3105         nevent
.SetEventObject(this); 
3106         nevent
.SetPoint(pt
); 
3108         // by default the dragging is not supported, the user code must 
3109         // explicitly allow the event for it to take place 
3112         if ( GetEventHandler()->ProcessEvent(nevent
) && nevent
.IsAllowed() ) 
3114             // we're going to drag this item 
3115             m_isDragging 
= true; 
3117             // remember the old cursor because we will change it while 
3119             m_oldCursor 
= m_cursor
; 
3121             // in a single selection control, hide the selection temporarily 
3122             if ( !(GetWindowStyleFlag() & wxTR_MULTIPLE
) ) 
3124                 m_oldSelection 
= (wxGenericTreeItem
*) GetSelection().m_pItem
; 
3126                 if ( m_oldSelection 
) 
3128                     m_oldSelection
->SetHilight(false); 
3129                     RefreshLine(m_oldSelection
); 
3136     else if ( event
.Dragging() ) 
3138         if ( item 
!= m_dropTarget 
) 
3140             // unhighlight the previous drop target 
3141             DrawDropEffect(m_dropTarget
); 
3143             m_dropTarget 
= item
; 
3145             // highlight the current drop target if any 
3146             DrawDropEffect(m_dropTarget
); 
3148 #if defined(__WXMSW__) || defined(__WXMAC__) || defined(__WXGTK20__) 
3155     else if ( (event
.LeftUp() || event
.RightUp()) && m_isDragging 
) 
3157         // erase the highlighting 
3158         DrawDropEffect(m_dropTarget
); 
3160         if ( m_oldSelection 
) 
3162             m_oldSelection
->SetHilight(true); 
3163             RefreshLine(m_oldSelection
); 
3164             m_oldSelection 
= (wxGenericTreeItem 
*)NULL
; 
3167         // generate the drag end event 
3168         wxTreeEvent 
event(wxEVT_COMMAND_TREE_END_DRAG
, GetId()); 
3170         event
.m_item 
= item
; 
3171         event
.m_pointDrag 
= pt
; 
3172         event
.SetEventObject(this); 
3174         (void)GetEventHandler()->ProcessEvent(event
); 
3176         m_isDragging 
= false; 
3177         m_dropTarget 
= (wxGenericTreeItem 
*)NULL
; 
3181         SetCursor(m_oldCursor
); 
3183 #if defined( __WXMSW__ ) || defined(__WXMAC__) 
3191         // If we got to this point, we are not dragging or moving the mouse. 
3192         // Because the code in carbon/toplevel.cpp will only set focus to the tree 
3193         // if we skip for EVT_LEFT_DOWN, we MUST skip this event here for focus to work. 
3194         // We skip even if we didn't hit an item because we still should 
3195         // restore focus to the tree control even if we didn't exactly hit an item. 
3196         if ( event
.LeftDown() ) 
3201         // here we process only the messages which happen on tree items 
3205         if (item 
== NULL
) return;  /* we hit the blank area */ 
3207         if ( event
.RightDown() ) 
3209             // If the item is already selected, do not update the selection. 
3210             // Multi-selections should not be cleared if a selected item is clicked. 
3211             if (!IsSelected(item
)) 
3213                 DoSelectItem(item
, true, false); 
3216             wxTreeEvent 
nevent(wxEVT_COMMAND_TREE_ITEM_RIGHT_CLICK
, GetId()); 
3217             nevent
.m_item 
= item
; 
3218             nevent
.m_pointDrag 
= CalcScrolledPosition(pt
); 
3219             nevent
.SetEventObject(this); 
3220             event
.Skip(!GetEventHandler()->ProcessEvent(nevent
)); 
3222         else if ( event
.LeftUp() ) 
3224             // this facilitates multiple-item drag-and-drop 
3226             if (item 
&& HasFlag(wxTR_MULTIPLE
)) 
3228                 wxArrayTreeItemIds selections
; 
3229                 size_t count 
= GetSelections(selections
); 
3235                     DoSelectItem(item
, true, false); 
3241                 if ( (item 
== m_current
) && 
3242                      (flags 
& wxTREE_HITTEST_ONITEMLABEL
) && 
3243                      HasFlag(wxTR_EDIT_LABELS
) ) 
3245                     if ( m_renameTimer 
) 
3247                         if ( m_renameTimer
->IsRunning() ) 
3248                             m_renameTimer
->Stop(); 
3252                         m_renameTimer 
= new wxTreeRenameTimer( this ); 
3255                     m_renameTimer
->Start( wxTreeRenameTimer::DELAY
, true ); 
3258                 m_lastOnSame 
= false; 
3261         else // !RightDown() && !LeftUp() ==> LeftDown() || LeftDClick() 
3263             if ( event
.LeftDown() ) 
3265                 m_lastOnSame 
= item 
== m_current
; 
3268             if ( flags 
& wxTREE_HITTEST_ONITEMBUTTON 
) 
3270                 // only toggle the item for a single click, double click on 
3271                 // the button doesn't do anything (it toggles the item twice) 
3272                 if ( event
.LeftDown() ) 
3277                 // don't select the item if the button was clicked 
3282             // clear the previously selected items, if the 
3283             // user clicked outside of the present selection. 
3284             // otherwise, perform the deselection on mouse-up. 
3285             // this allows multiple drag and drop to work. 
3286             // but if Cmd is down, toggle selection of the clicked item 
3287             if (!IsSelected(item
) || event
.CmdDown()) 
3289                 // how should the selection work for this event? 
3290                 bool is_multiple
, extended_select
, unselect_others
; 
3291                 EventFlagsToSelType(GetWindowStyleFlag(), 
3294                                     is_multiple
, extended_select
, unselect_others
); 
3296                 DoSelectItem(item
, unselect_others
, extended_select
); 
3300             // For some reason, Windows isn't recognizing a left double-click, 
3301             // so we need to simulate it here.  Allow 200 milliseconds for now. 
3302             if ( event
.LeftDClick() ) 
3304                 // double clicking should not start editing the item label 
3305                 if ( m_renameTimer 
) 
3306                     m_renameTimer
->Stop(); 
3308                 m_lastOnSame 
= false; 
3310                 // send activate event first 
3311                 wxTreeEvent 
nevent( wxEVT_COMMAND_TREE_ITEM_ACTIVATED
, GetId() ); 
3312                 nevent
.m_item 
= item
; 
3313                 nevent
.m_pointDrag 
= CalcScrolledPosition(pt
); 
3314                 nevent
.SetEventObject( this ); 
3315                 if ( !GetEventHandler()->ProcessEvent( nevent 
) ) 
3317                     // if the user code didn't process the activate event, 
3318                     // handle it ourselves by toggling the item when it is 
3320                     if ( item
->HasPlus() ) 
3330 void wxGenericTreeCtrl::OnInternalIdle() 
3332     wxWindow::OnInternalIdle(); 
3334     // Check if we need to select the root item 
3335     // because nothing else has been selected. 
3336     // Delaying it means that we can invoke event handlers 
3337     // as required, when a first item is selected. 
3338     if (!HasFlag(wxTR_MULTIPLE
) && !GetSelection().IsOk()) 
3341             SelectItem(m_select_me
); 
3342         else if (GetRootItem().IsOk()) 
3343             SelectItem(GetRootItem()); 
3346     /* after all changes have been done to the tree control, 
3347      * we actually redraw the tree when everything is over */ 
3349     if (!m_dirty
) return; 
3350     if (m_freezeCount
) return; 
3354     CalculatePositions(); 
3356     AdjustMyScrollbars(); 
3359 void wxGenericTreeCtrl::CalculateSize( wxGenericTreeItem 
*item
, wxDC 
&dc 
) 
3364     wxTreeItemAttr 
*attr 
= item
->GetAttributes(); 
3365     if ( attr 
&& attr
->HasFont() ) 
3366         dc
.SetFont(attr
->GetFont()); 
3367     else if ( item
->IsBold() ) 
3368         dc
.SetFont(m_boldFont
); 
3370         dc
.SetFont(m_normalFont
); 
3372     dc
.GetTextExtent( item
->GetText(), &text_w
, &text_h 
); 
3375     // restore normal font 
3376     dc
.SetFont( m_normalFont 
); 
3380     int image 
= item
->GetCurrentImage(); 
3381     if ( image 
!= NO_IMAGE 
) 
3383         if ( m_imageListNormal 
) 
3385             m_imageListNormal
->GetSize( image
, image_w
, image_h 
); 
3390     int total_h 
= (image_h 
> text_h
) ? image_h 
: text_h
; 
3393         total_h 
+= 2;            // at least 2 pixels 
3395         total_h 
+= total_h
/10;   // otherwise 10% extra spacing 
3397     item
->SetHeight(total_h
); 
3398     if (total_h
>m_lineHeight
) 
3399         m_lineHeight
=total_h
; 
3401     item
->SetWidth(image_w
+text_w
+2); 
3404 // ----------------------------------------------------------------------------- 
3405 // for developper : y is now the top of the level 
3406 // not the middle of it ! 
3407 void wxGenericTreeCtrl::CalculateLevel( wxGenericTreeItem 
*item
, wxDC 
&dc
, int level
, int &y 
) 
3409     int x 
= level
*m_indent
; 
3410     if (!HasFlag(wxTR_HIDE_ROOT
)) 
3414     else if (level 
== 0) 
3416         // a hidden root is not evaluated, but its 
3417         // children are always calculated 
3421     CalculateSize( item
, dc 
); 
3424     item
->SetX( x
+m_spacing 
); 
3426     y 
+= GetLineHeight(item
); 
3428     if ( !item
->IsExpanded() ) 
3430         // we don't need to calculate collapsed branches 
3435     wxArrayGenericTreeItems
& children 
= item
->GetChildren(); 
3436     size_t n
, count 
= children
.Count(); 
3438     for (n 
= 0; n 
< count
; ++n 
) 
3439         CalculateLevel( children
[n
], dc
, level
, y 
);  // recurse 
3442 void wxGenericTreeCtrl::CalculatePositions() 
3444     if ( !m_anchor 
) return; 
3446     wxClientDC 
dc(this); 
3449     dc
.SetFont( m_normalFont 
); 
3451     dc
.SetPen( m_dottedPen 
); 
3452     //if(GetImageList() == NULL) 
3453     // m_lineHeight = (int)(dc.GetCharHeight() + 4); 
3456     CalculateLevel( m_anchor
, dc
, 0, y 
); // start recursion 
3459 void wxGenericTreeCtrl::RefreshSubtree(wxGenericTreeItem 
*item
) 
3461     if (m_dirty
) return; 
3462     if (m_freezeCount
) return; 
3464     wxSize client 
= GetClientSize(); 
3467     CalcScrolledPosition(0, item
->GetY(), NULL
, &rect
.y
); 
3468     rect
.width 
= client
.x
; 
3469     rect
.height 
= client
.y
; 
3471     Refresh(true, &rect
); 
3473     AdjustMyScrollbars(); 
3476 void wxGenericTreeCtrl::RefreshLine( wxGenericTreeItem 
*item 
) 
3478     if (m_dirty
) return; 
3479     if (m_freezeCount
) return; 
3482     CalcScrolledPosition(0, item
->GetY(), NULL
, &rect
.y
); 
3483     rect
.width 
= GetClientSize().x
; 
3484     rect
.height 
= GetLineHeight(item
); //dc.GetCharHeight() + 6; 
3486     Refresh(true, &rect
); 
3489 void wxGenericTreeCtrl::RefreshSelected() 
3491     if (m_freezeCount
) return; 
3493     // TODO: this is awfully inefficient, we should keep the list of all 
3494     //       selected items internally, should be much faster 
3496         RefreshSelectedUnder(m_anchor
); 
3499 void wxGenericTreeCtrl::RefreshSelectedUnder(wxGenericTreeItem 
*item
) 
3501     if (m_freezeCount
) return; 
3503     if ( item
->IsSelected() ) 
3506     const wxArrayGenericTreeItems
& children 
= item
->GetChildren(); 
3507     size_t count 
= children
.GetCount(); 
3508     for ( size_t n 
= 0; n 
< count
; n
++ ) 
3510         RefreshSelectedUnder(children
[n
]); 
3514 void wxGenericTreeCtrl::Freeze() 
3519 void wxGenericTreeCtrl::Thaw() 
3521     wxCHECK_RET( m_freezeCount 
> 0, _T("thawing unfrozen tree control?") ); 
3523     if ( !--m_freezeCount 
) 
3529 // ---------------------------------------------------------------------------- 
3530 // changing colours: we need to refresh the tree control 
3531 // ---------------------------------------------------------------------------- 
3533 bool wxGenericTreeCtrl::SetBackgroundColour(const wxColour
& colour
) 
3535     if ( !wxWindow::SetBackgroundColour(colour
) ) 
3538     if (m_freezeCount
) return true; 
3545 bool wxGenericTreeCtrl::SetForegroundColour(const wxColour
& colour
) 
3547     if ( !wxWindow::SetForegroundColour(colour
) ) 
3550     if (m_freezeCount
) return true; 
3557 // Process the tooltip event, to speed up event processing. 
3558 // Doesn't actually get a tooltip. 
3559 void wxGenericTreeCtrl::OnGetToolTip( wxTreeEvent 
&event 
) 
3565 wxSize 
wxGenericTreeCtrl::DoGetBestSize() const 
3567     // something is better than nothing... 
3568     // 100x80 is what the MSW version will get from the default 
3569     // wxControl::DoGetBestSize 
3570     return wxSize(100,80); 
3574 // NOTE: If using the wxListBox visual attributes works everywhere then this can 
3575 // be removed, as well as the #else case below. 
3576 #define _USE_VISATTR 0 
3579 #include "wx/listbox.h" 
3585 wxGenericTreeCtrl::GetClassDefaultAttributes(wxWindowVariant variant
) 
3587 wxGenericTreeCtrl::GetClassDefaultAttributes(wxWindowVariant 
WXUNUSED(variant
)) 
3591     // Use the same color scheme as wxListBox 
3592     return wxListBox::GetClassDefaultAttributes(variant
); 
3594     wxVisualAttributes attr
; 
3595     attr
.colFg 
= wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT
); 
3596     attr
.colBg 
= wxSystemSettings::GetColour(wxSYS_COLOUR_LISTBOX
); 
3597     attr
.font  
= wxSystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT
); 
3602 #if WXWIN_COMPATIBILITY_2_4 
3604 int wxGenericTreeCtrl::GetItemSelectedImage(const wxTreeItemId
& item
) const 
3606     return GetItemImage(item
, wxTreeItemIcon_Selected
); 
3609 void wxGenericTreeCtrl::SetItemSelectedImage(const wxTreeItemId
& item
, int image
) 
3611     SetItemImage(item
, image
, wxTreeItemIcon_Selected
); 
3614 #endif // WXWIN_COMPATIBILITY_2_4 
3616 #if WXWIN_COMPATIBILITY_2_2 
3618 wxTreeItemId 
wxGenericTreeCtrl::GetParent(const wxTreeItemId
& item
) const 
3620     return GetItemParent( item 
); 
3623 #endif  // WXWIN_COMPATIBILITY_2_2 
3625 #endif // wxUSE_TREECTRL