1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        generic/listctrl.cpp 
   3 // Purpose:     generic implementation of wxListCtrl 
   4 // Author:      Robert Roebling 
   5 //              Vadim Zeitlin (virtual list control support) 
   7 // Copyright:   (c) 1998 Robert Roebling 
   8 // Licence:     wxWindows licence 
   9 ///////////////////////////////////////////////////////////////////////////// 
  14    1. we need to implement searching/sorting for virtual controls somehow 
  15   ?2. when changing selection the lines are refreshed twice 
  18 // ============================================================================ 
  20 // ============================================================================ 
  22 // ---------------------------------------------------------------------------- 
  24 // ---------------------------------------------------------------------------- 
  26 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA) 
  27     #pragma implementation "listctrl.h" 
  28     #pragma implementation "listctrlbase.h" 
  31 // For compilers that support precompilation, includes "wx.h". 
  32 #include "wx/wxprec.h" 
  43     #include "wx/dynarray.h" 
  45     #include "wx/dcscreen.h" 
  47     #include "wx/textctrl.h" 
  50 // under Win32 we always use the native version and also may use the generic 
  51 // one, however some things should be done only if we use only the generic 
  53 #if defined(__WIN32__) && !defined(__WXUNIVERSAL__) 
  54     #define HAVE_NATIVE_LISTCTRL 
  57 // if we have the native control, wx/listctrl.h declares it and not this one 
  58 #ifdef HAVE_NATIVE_LISTCTRL 
  59     #include "wx/generic/listctrl.h" 
  60 #else // !HAVE_NATIVE_LISTCTRL 
  61     #include "wx/listctrl.h" 
  63     // if we have a native version, its implementation file does all this 
  64     IMPLEMENT_DYNAMIC_CLASS(wxListItem
, wxObject
) 
  65     IMPLEMENT_DYNAMIC_CLASS(wxListView
, wxListCtrl
) 
  66     IMPLEMENT_DYNAMIC_CLASS(wxListEvent
, wxNotifyEvent
) 
  68     IMPLEMENT_DYNAMIC_CLASS(wxListCtrl
, wxGenericListCtrl
) 
  69 #endif // HAVE_NATIVE_LISTCTRL/!HAVE_NATIVE_LISTCTRL 
  71 #include "wx/selstore.h" 
  73 #include "wx/renderer.h" 
  75 // ---------------------------------------------------------------------------- 
  77 // ---------------------------------------------------------------------------- 
  79 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_BEGIN_DRAG
) 
  80 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_BEGIN_RDRAG
) 
  81 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_BEGIN_LABEL_EDIT
) 
  82 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_END_LABEL_EDIT
) 
  83 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_DELETE_ITEM
) 
  84 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_DELETE_ALL_ITEMS
) 
  85 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_GET_INFO
) 
  86 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_SET_INFO
) 
  87 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_ITEM_SELECTED
) 
  88 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_ITEM_DESELECTED
) 
  89 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_KEY_DOWN
) 
  90 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_INSERT_ITEM
) 
  91 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_COL_CLICK
) 
  92 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_COL_RIGHT_CLICK
) 
  93 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_COL_BEGIN_DRAG
) 
  94 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_COL_DRAGGING
) 
  95 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_COL_END_DRAG
) 
  96 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_ITEM_RIGHT_CLICK
) 
  97 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_ITEM_MIDDLE_CLICK
) 
  98 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_ITEM_ACTIVATED
) 
  99 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_ITEM_FOCUSED
) 
 100 DEFINE_EVENT_TYPE(wxEVT_COMMAND_LIST_CACHE_HINT
) 
 102 // ---------------------------------------------------------------------------- 
 104 // ---------------------------------------------------------------------------- 
 106 // // the height of the header window (FIXME: should depend on its font!) 
 107 // static const int HEADER_HEIGHT = 23; 
 109 static const int SCROLL_UNIT_X 
= 15; 
 111 // the spacing between the lines (in report mode) 
 112 static const int LINE_SPACING 
= 0; 
 114 // extra margins around the text label 
 115 static const int EXTRA_WIDTH 
= 3; 
 116 static const int EXTRA_HEIGHT 
= 4; 
 118 // margin between the window and the items 
 119 static const int EXTRA_BORDER_X 
= 2; 
 120 static const int EXTRA_BORDER_Y 
= 2; 
 122 // offset for the header window 
 123 static const int HEADER_OFFSET_X 
= 1; 
 124 static const int HEADER_OFFSET_Y 
= 1; 
 126 // margin between rows of icons in [small] icon view 
 127 static const int MARGIN_BETWEEN_ROWS 
= 6; 
 129 // when autosizing the columns, add some slack 
 130 static const int AUTOSIZE_COL_MARGIN 
= 10; 
 132 // default and minimal widths for the header columns 
 133 static const int WIDTH_COL_DEFAULT 
= 80; 
 134 static const int WIDTH_COL_MIN 
= 10; 
 136 // the space between the image and the text in the report mode 
 137 static const int IMAGE_MARGIN_IN_REPORT_MODE 
= 5; 
 139 // ============================================================================ 
 141 // ============================================================================ 
 143 //----------------------------------------------------------------------------- 
 144 //  wxListItemData (internal) 
 145 //----------------------------------------------------------------------------- 
 147 class WXDLLEXPORT wxListItemData
 
 150     wxListItemData(wxListMainWindow 
*owner
); 
 153     void SetItem( const wxListItem 
&info 
); 
 154     void SetImage( int image 
) { m_image 
= image
; } 
 155     void SetData( long data 
) { m_data 
= data
; } 
 156     void SetPosition( int x
, int y 
); 
 157     void SetSize( int width
, int height 
); 
 159     bool HasText() const { return !m_text
.empty(); } 
 160     const wxString
& GetText() const { return m_text
; } 
 161     void SetText(const wxString
& text
) { m_text 
= text
; } 
 163     // we can't use empty string for measuring the string width/height, so 
 164     // always return something 
 165     wxString 
GetTextForMeasuring() const 
 167         wxString s 
= GetText(); 
 174     bool IsHit( int x
, int y 
) const; 
 178     int GetWidth() const; 
 179     int GetHeight() const; 
 181     int GetImage() const { return m_image
; } 
 182     bool HasImage() const { return GetImage() != -1; } 
 184     void GetItem( wxListItem 
&info 
) const; 
 186     void SetAttr(wxListItemAttr 
*attr
) { m_attr 
= attr
; } 
 187     wxListItemAttr 
*GetAttr() const { return m_attr
; } 
 190     // the item image or -1 
 193     // user data associated with the item 
 196     // the item coordinates are not used in report mode, instead this pointer 
 197     // is NULL and the owner window is used to retrieve the item position and 
 201     // the list ctrl we are in 
 202     wxListMainWindow 
*m_owner
; 
 204     // custom attributes or NULL 
 205     wxListItemAttr 
*m_attr
; 
 208     // common part of all ctors 
 214 //----------------------------------------------------------------------------- 
 215 //  wxListHeaderData (internal) 
 216 //----------------------------------------------------------------------------- 
 218 class WXDLLEXPORT wxListHeaderData 
: public wxObject
 
 222     wxListHeaderData( const wxListItem 
&info 
); 
 223     void SetItem( const wxListItem 
&item 
); 
 224     void SetPosition( int x
, int y 
); 
 225     void SetWidth( int w 
); 
 226     void SetFormat( int format 
); 
 227     void SetHeight( int h 
); 
 228     bool HasImage() const; 
 230     bool HasText() const { return !m_text
.empty(); } 
 231     const wxString
& GetText() const { return m_text
; } 
 232     void SetText(const wxString
& text
) { m_text 
= text
; } 
 234     void GetItem( wxListItem 
&item 
); 
 236     bool IsHit( int x
, int y 
) const; 
 237     int GetImage() const; 
 238     int GetWidth() const; 
 239     int GetFormat() const; 
 255 //----------------------------------------------------------------------------- 
 256 //  wxListLineData (internal) 
 257 //----------------------------------------------------------------------------- 
 259 WX_DECLARE_LIST(wxListItemData
, wxListItemDataList
); 
 260 #include "wx/listimpl.cpp" 
 261 WX_DEFINE_LIST(wxListItemDataList
); 
 266     // the list of subitems: only may have more than one item in report mode 
 267     wxListItemDataList m_items
; 
 269     // this is not used in report view 
 281         // the part to be highlighted 
 282         wxRect m_rectHighlight
; 
 284         // extend all our rects to be centered inside theo ne of given width 
 285         void ExtendWidth(wxCoord w
) 
 287             wxASSERT_MSG( m_rectAll
.width 
<= w
, 
 288                             _T("width can only be increased") ); 
 291             m_rectLabel
.x 
= m_rectAll
.x 
+ (w 
- m_rectLabel
.width
)/2; 
 292             m_rectIcon
.x 
= m_rectAll
.x 
+ (w 
- m_rectIcon
.width
)/2; 
 293             m_rectHighlight
.x 
= m_rectAll
.x 
+ (w 
- m_rectHighlight
.width
)/2; 
 297     // is this item selected? [NB: not used in virtual mode] 
 300     // back pointer to the list ctrl 
 301     wxListMainWindow 
*m_owner
; 
 304     wxListLineData(wxListMainWindow 
*owner
); 
 308         WX_CLEAR_LIST(wxListItemDataList
, m_items
); 
 312     // are we in report mode? 
 313     inline bool InReportView() const; 
 315     // are we in virtual report mode? 
 316     inline bool IsVirtual() const; 
 318     // these 2 methods shouldn't be called for report view controls, in that 
 319     // case we determine our position/size ourselves 
 321     // calculate the size of the line 
 322     void CalculateSize( wxDC 
*dc
, int spacing 
); 
 324     // remember the position this line appears at 
 325     void SetPosition( int x
, int y
, int spacing 
); 
 329     void SetImage( int image 
) { SetImage(0, image
); } 
 330     int GetImage() const { return GetImage(0); } 
 331     bool HasImage() const { return GetImage() != -1; } 
 332     bool HasText() const { return !GetText(0).empty(); } 
 334     void SetItem( int index
, const wxListItem 
&info 
); 
 335     void GetItem( int index
, wxListItem 
&info 
); 
 337     wxString 
GetText(int index
) const; 
 338     void SetText( int index
, const wxString s 
); 
 340     wxListItemAttr 
*GetAttr() const; 
 341     void SetAttr(wxListItemAttr 
*attr
); 
 343     // return true if the highlighting really changed 
 344     bool Highlight( bool on 
); 
 346     void ReverseHighlight(); 
 348     bool IsHighlighted() const 
 350         wxASSERT_MSG( !IsVirtual(), _T("unexpected call to IsHighlighted") ); 
 352         return m_highlighted
; 
 355     // draw the line on the given DC in icon/list mode 
 356     void Draw( wxDC 
*dc 
); 
 358     // the same in report mode 
 359     void DrawInReportMode( wxDC 
*dc
, 
 361                            const wxRect
& rectHL
, 
 365     // set the line to contain num items (only can be > 1 in report mode) 
 366     void InitItems( int num 
); 
 368     // get the mode (i.e. style)  of the list control 
 369     inline int GetMode() const; 
 371     // prepare the DC for drawing with these item's attributes, return true if 
 372     // we need to draw the items background to highlight it, false otherwise 
 373     bool SetAttributes(wxDC 
*dc
, 
 374                        const wxListItemAttr 
*attr
, 
 377     // draw the text on the DC with the correct justification; also add an 
 378     // ellipsis if the text is too large to fit in the current width 
 379     void DrawTextFormatted(wxDC 
*dc
, const wxString 
&text
, int col
, int x
, int y
, int width
); 
 381     // these are only used by GetImage/SetImage above, we don't support images 
 382     // with subitems at the public API level yet 
 383     void SetImage( int index
, int image 
); 
 384     int GetImage( int index 
) const; 
 387 WX_DECLARE_EXPORTED_OBJARRAY(wxListLineData
, wxListLineDataArray
); 
 388 #include "wx/arrimpl.cpp" 
 389 WX_DEFINE_OBJARRAY(wxListLineDataArray
); 
 391 //----------------------------------------------------------------------------- 
 392 //  wxListHeaderWindow (internal) 
 393 //----------------------------------------------------------------------------- 
 395 class WXDLLEXPORT wxListHeaderWindow 
: public wxWindow
 
 398     wxListMainWindow  
*m_owner
; 
 399     wxCursor          
*m_currentCursor
; 
 400     wxCursor          
*m_resizeCursor
; 
 403     // column being resized or -1 
 406     // divider line position in logical (unscrolled) coords 
 409     // minimal position beyond which the divider line can't be dragged in 
 414     wxListHeaderWindow(); 
 416     wxListHeaderWindow( wxWindow 
*win
, 
 418                         wxListMainWindow 
*owner
, 
 419                         const wxPoint 
&pos 
= wxDefaultPosition
, 
 420                         const wxSize 
&size 
= wxDefaultSize
, 
 422                         const wxString 
&name 
= wxT("wxlistctrlcolumntitles") ); 
 424     virtual ~wxListHeaderWindow(); 
 427     void AdjustDC(wxDC
& dc
); 
 429     void OnPaint( wxPaintEvent 
&event 
); 
 430     void OnMouse( wxMouseEvent 
&event 
); 
 431     void OnSetFocus( wxFocusEvent 
&event 
); 
 437     // common part of all ctors 
 440     // generate and process the list event of the given type, return true if 
 441     // it wasn't vetoed, i.e. if we should proceed 
 442     bool SendListEvent(wxEventType type
, wxPoint pos
); 
 444     DECLARE_DYNAMIC_CLASS(wxListHeaderWindow
) 
 445     DECLARE_EVENT_TABLE() 
 448 //----------------------------------------------------------------------------- 
 449 // wxListRenameTimer (internal) 
 450 //----------------------------------------------------------------------------- 
 452 class WXDLLEXPORT wxListRenameTimer
: public wxTimer
 
 455     wxListMainWindow 
*m_owner
; 
 458     wxListRenameTimer( wxListMainWindow 
*owner 
); 
 462 //----------------------------------------------------------------------------- 
 463 //  wxListTextCtrl (internal) 
 464 //----------------------------------------------------------------------------- 
 466 class WXDLLEXPORT wxListTextCtrl
: public wxTextCtrl
 
 469     wxListTextCtrl(wxListMainWindow 
*owner
, size_t itemEdit
); 
 472     void OnChar( wxKeyEvent 
&event 
); 
 473     void OnKeyUp( wxKeyEvent 
&event 
); 
 474     void OnKillFocus( wxFocusEvent 
&event 
); 
 476     bool AcceptChanges(); 
 480     wxListMainWindow   
*m_owner
; 
 481     wxString            m_startValue
; 
 485     DECLARE_EVENT_TABLE() 
 488 //----------------------------------------------------------------------------- 
 489 //  wxListMainWindow (internal) 
 490 //----------------------------------------------------------------------------- 
 492 WX_DECLARE_LIST(wxListHeaderData
, wxListHeaderDataList
); 
 493 #include "wx/listimpl.cpp" 
 494 WX_DEFINE_LIST(wxListHeaderDataList
); 
 496 class wxListMainWindow 
: public wxScrolledWindow
 
 500     wxListMainWindow( wxWindow 
*parent
, 
 502                       const wxPoint
& pos 
= wxDefaultPosition
, 
 503                       const wxSize
& size 
= wxDefaultSize
, 
 505                       const wxString 
&name 
= _T("listctrlmainwindow") ); 
 507     virtual ~wxListMainWindow(); 
 509     bool HasFlag(int flag
) const { return m_parent
->HasFlag(flag
); } 
 511     // return true if this is a virtual list control 
 512     bool IsVirtual() const { return HasFlag(wxLC_VIRTUAL
); } 
 514     // return true if the control is in report mode 
 515     bool InReportView() const { return HasFlag(wxLC_REPORT
); } 
 517     // return true if we are in single selection mode, false if multi sel 
 518     bool IsSingleSel() const { return HasFlag(wxLC_SINGLE_SEL
); } 
 520     // do we have a header window? 
 521     bool HasHeader() const 
 522         { return HasFlag(wxLC_REPORT
) && !HasFlag(wxLC_NO_HEADER
); } 
 524     void HighlightAll( bool on 
); 
 526     // all these functions only do something if the line is currently visible 
 528     // change the line "selected" state, return TRUE if it really changed 
 529     bool HighlightLine( size_t line
, bool highlight 
= TRUE
); 
 531     // as HighlightLine() but do it for the range of lines: this is incredibly 
 532     // more efficient for virtual list controls! 
 534     // NB: unlike HighlightLine() this one does refresh the lines on screen 
 535     void HighlightLines( size_t lineFrom
, size_t lineTo
, bool on 
= TRUE 
); 
 537     // toggle the line state and refresh it 
 538     void ReverseHighlight( size_t line 
) 
 539         { HighlightLine(line
, !IsHighlighted(line
)); RefreshLine(line
); } 
 541     // return true if the line is highlighted 
 542     bool IsHighlighted(size_t line
) const; 
 544     // refresh one or several lines at once 
 545     void RefreshLine( size_t line 
); 
 546     void RefreshLines( size_t lineFrom
, size_t lineTo 
); 
 548     // refresh all selected items 
 549     void RefreshSelected(); 
 551     // refresh all lines below the given one: the difference with 
 552     // RefreshLines() is that the index here might not be a valid one (happens 
 553     // when the last line is deleted) 
 554     void RefreshAfter( size_t lineFrom 
); 
 556     // the methods which are forwarded to wxListLineData itself in list/icon 
 557     // modes but are here because the lines don't store their positions in the 
 560     // get the bound rect for the entire line 
 561     wxRect 
GetLineRect(size_t line
) const; 
 563     // get the bound rect of the label 
 564     wxRect 
GetLineLabelRect(size_t line
) const; 
 566     // get the bound rect of the items icon (only may be called if we do have 
 568     wxRect 
GetLineIconRect(size_t line
) const; 
 570     // get the rect to be highlighted when the item has focus 
 571     wxRect 
GetLineHighlightRect(size_t line
) const; 
 573     // get the size of the total line rect 
 574     wxSize 
GetLineSize(size_t line
) const 
 575         { return GetLineRect(line
).GetSize(); } 
 577     // return the hit code for the corresponding position (in this line) 
 578     long HitTestLine(size_t line
, int x
, int y
) const; 
 580     // bring the selected item into view, scrolling to it if necessary 
 581     void MoveToItem(size_t item
); 
 583     // bring the current item into view 
 584     void MoveToFocus() { MoveToItem(m_current
); } 
 586     // start editing the label of the given item 
 587     void EditLabel( long item 
); 
 589     // suspend/resume redrawing the control 
 595     void OnRenameTimer(); 
 596     bool OnRenameAccept(size_t itemEdit
, const wxString
& value
); 
 597     void OnRenameCancelled(size_t itemEdit
); 
 599     void OnMouse( wxMouseEvent 
&event 
); 
 601     // called to switch the selection from the current item to newCurrent, 
 602     void OnArrowChar( size_t newCurrent
, const wxKeyEvent
& event 
); 
 604     void OnChar( wxKeyEvent 
&event 
); 
 605     void OnKeyDown( wxKeyEvent 
&event 
); 
 606     void OnSetFocus( wxFocusEvent 
&event 
); 
 607     void OnKillFocus( wxFocusEvent 
&event 
); 
 608     void OnScroll(wxScrollWinEvent
& event
) ; 
 610     void OnPaint( wxPaintEvent 
&event 
); 
 612     void DrawImage( int index
, wxDC 
*dc
, int x
, int y 
); 
 613     void GetImageSize( int index
, int &width
, int &height 
) const; 
 614     int GetTextLength( const wxString 
&s 
) const; 
 616     void SetImageList( wxImageListType 
*imageList
, int which 
); 
 617     void SetItemSpacing( int spacing
, bool isSmall 
= FALSE 
); 
 618     int GetItemSpacing( bool isSmall 
= FALSE 
); 
 620     void SetColumn( int col
, wxListItem 
&item 
); 
 621     void SetColumnWidth( int col
, int width 
); 
 622     void GetColumn( int col
, wxListItem 
&item 
) const; 
 623     int GetColumnWidth( int col 
) const; 
 624     int GetColumnCount() const { return m_columns
.GetCount(); } 
 626     // returns the sum of the heights of all columns 
 627     int GetHeaderWidth() const; 
 629     int GetCountPerPage() const; 
 631     void SetItem( wxListItem 
&item 
); 
 632     void GetItem( wxListItem 
&item 
) const; 
 633     void SetItemState( long item
, long state
, long stateMask 
); 
 634     int GetItemState( long item
, long stateMask 
) const; 
 635     void GetItemRect( long index
, wxRect 
&rect 
) const; 
 636     wxRect 
GetViewRect() const; 
 637     bool GetItemPosition( long item
, wxPoint
& pos 
) const; 
 638     int GetSelectedItemCount() const; 
 640     wxString 
GetItemText(long item
) const 
 643         info
.m_itemId 
= item
; 
 648     void SetItemText(long item
, const wxString
& value
) 
 651         info
.m_mask 
= wxLIST_MASK_TEXT
; 
 652         info
.m_itemId 
= item
; 
 657     // set the scrollbars and update the positions of the items 
 658     void RecalculatePositions(bool noRefresh 
= FALSE
); 
 660     // refresh the window and the header 
 663     long GetNextItem( long item
, int geometry
, int state 
) const; 
 664     void DeleteItem( long index 
); 
 665     void DeleteAllItems(); 
 666     void DeleteColumn( int col 
); 
 667     void DeleteEverything(); 
 668     void EnsureVisible( long index 
); 
 669     long FindItem( long start
, const wxString
& str
, bool partial 
= FALSE 
); 
 670     long FindItem( long start
, long data
); 
 671     long HitTest( int x
, int y
, int &flags 
); 
 672     void InsertItem( wxListItem 
&item 
); 
 673     void InsertColumn( long col
, wxListItem 
&item 
); 
 674     void SortItems( wxListCtrlCompare fn
, long data 
); 
 676     size_t GetItemCount() const; 
 677     bool IsEmpty() const { return GetItemCount() == 0; } 
 678     void SetItemCount(long count
); 
 680     // change the current (== focused) item, send a notification event 
 681     void ChangeCurrent(size_t current
); 
 682     void ResetCurrent() { ChangeCurrent((size_t)-1); } 
 683     bool HasCurrent() const { return m_current 
!= (size_t)-1; } 
 685     // send out a wxListEvent 
 686     void SendNotify( size_t line
, 
 688                      wxPoint point 
= wxDefaultPosition 
); 
 690     // override base class virtual to reset m_lineHeight when the font changes 
 691     virtual bool SetFont(const wxFont
& font
) 
 693         if ( !wxScrolledWindow::SetFont(font
) ) 
 701     // these are for wxListLineData usage only 
 703     // get the backpointer to the list ctrl 
 704     wxGenericListCtrl 
*GetListCtrl() const 
 706         return wxStaticCast(GetParent(), wxGenericListCtrl
); 
 709     // get the height of all lines (assuming they all do have the same height) 
 710     wxCoord 
GetLineHeight() const; 
 712     // get the y position of the given line (only for report view) 
 713     wxCoord 
GetLineY(size_t line
) const; 
 715     // get the brush to use for the item highlighting 
 716     wxBrush 
*GetHighlightBrush() const 
 718         return m_hasFocus 
? m_highlightBrush 
: m_highlightUnfocusedBrush
; 
 722     // the array of all line objects for a non virtual list control (for the 
 723     // virtual list control we only ever use m_lines[0]) 
 724     wxListLineDataArray  m_lines
; 
 726     // the list of column objects 
 727     wxListHeaderDataList m_columns
; 
 729     // currently focused item or -1 
 732     // the number of lines per page 
 735     // this flag is set when something which should result in the window 
 736     // redrawing happens (i.e. an item was added or deleted, or its appearance 
 737     // changed) and OnPaint() doesn't redraw the window while it is set which 
 738     // allows to minimize the number of repaintings when a lot of items are 
 739     // being added. The real repainting occurs only after the next OnIdle() 
 743     wxColour            
*m_highlightColour
; 
 744     wxImageListType         
*m_small_image_list
; 
 745     wxImageListType         
*m_normal_image_list
; 
 747     int                  m_normal_spacing
; 
 751     wxTimer             
*m_renameTimer
; 
 756     // for double click logic 
 757     size_t m_lineLastClicked
, 
 758            m_lineBeforeLastClicked
; 
 761     // the total count of items in a virtual list control 
 764     // the object maintaining the items selection state, only used in virtual 
 766     wxSelectionStore m_selStore
; 
 768     // common part of all ctors 
 771     // get the line data for the given index 
 772     wxListLineData 
*GetLine(size_t n
) const 
 774         wxASSERT_MSG( n 
!= (size_t)-1, _T("invalid line index") ); 
 778             wxConstCast(this, wxListMainWindow
)->CacheLineData(n
); 
 786     // get a dummy line which can be used for geometry calculations and such: 
 787     // you must use GetLine() if you want to really draw the line 
 788     wxListLineData 
*GetDummyLine() const; 
 790     // cache the line data of the n-th line in m_lines[0] 
 791     void CacheLineData(size_t line
); 
 793     // get the range of visible lines 
 794     void GetVisibleLinesRange(size_t *from
, size_t *to
); 
 796     // force us to recalculate the range of visible lines 
 797     void ResetVisibleLinesRange() { m_lineFrom 
= (size_t)-1; } 
 799     // get the colour to be used for drawing the rules 
 800     wxColour 
GetRuleColour() const 
 805         return wxSystemSettings::GetColour(wxSYS_COLOUR_3DLIGHT
); 
 810     // initialize the current item if needed 
 811     void UpdateCurrent(); 
 813     // delete all items but don't refresh: called from dtor 
 814     void DoDeleteAllItems(); 
 816     // the height of one line using the current font 
 817     wxCoord m_lineHeight
; 
 819     // the total header width or 0 if not calculated yet 
 820     wxCoord m_headerWidth
; 
 822     // the first and last lines being shown on screen right now (inclusive), 
 823     // both may be -1 if they must be calculated so never access them directly: 
 824     // use GetVisibleLinesRange() above instead 
 828     // the brushes to use for item highlighting when we do/don't have focus 
 829     wxBrush 
*m_highlightBrush
, 
 830             *m_highlightUnfocusedBrush
; 
 832     // if this is > 0, the control is frozen and doesn't redraw itself 
 833     size_t m_freezeCount
; 
 835     DECLARE_DYNAMIC_CLASS(wxListMainWindow
) 
 836     DECLARE_EVENT_TABLE() 
 838     friend class wxGenericListCtrl
; 
 841 // ============================================================================ 
 843 // ============================================================================ 
 845 //----------------------------------------------------------------------------- 
 847 //----------------------------------------------------------------------------- 
 849 wxListItemData::~wxListItemData() 
 851     // in the virtual list control the attributes are managed by the main 
 852     // program, so don't delete them 
 853     if ( !m_owner
->IsVirtual() ) 
 861 void wxListItemData::Init() 
 869 wxListItemData::wxListItemData(wxListMainWindow 
*owner
) 
 875     if ( owner
->InReportView() ) 
 885 void wxListItemData::SetItem( const wxListItem 
&info 
) 
 887     if ( info
.m_mask 
& wxLIST_MASK_TEXT 
) 
 888         SetText(info
.m_text
); 
 889     if ( info
.m_mask 
& wxLIST_MASK_IMAGE 
) 
 890         m_image 
= info
.m_image
; 
 891     if ( info
.m_mask 
& wxLIST_MASK_DATA 
) 
 892         m_data 
= info
.m_data
; 
 894     if ( info
.HasAttributes() ) 
 897             *m_attr 
= *info
.GetAttributes(); 
 899             m_attr 
= new wxListItemAttr(*info
.GetAttributes()); 
 907         m_rect
->width 
= info
.m_width
; 
 911 void wxListItemData::SetPosition( int x
, int y 
) 
 913     wxCHECK_RET( m_rect
, _T("unexpected SetPosition() call") ); 
 919 void wxListItemData::SetSize( int width
, int height 
) 
 921     wxCHECK_RET( m_rect
, _T("unexpected SetSize() call") ); 
 924         m_rect
->width 
= width
; 
 926         m_rect
->height 
= height
; 
 929 bool wxListItemData::IsHit( int x
, int y 
) const 
 931     wxCHECK_MSG( m_rect
, FALSE
, _T("can't be called in this mode") ); 
 933     return wxRect(GetX(), GetY(), GetWidth(), GetHeight()).Inside(x
, y
); 
 936 int wxListItemData::GetX() const 
 938     wxCHECK_MSG( m_rect
, 0, _T("can't be called in this mode") ); 
 943 int wxListItemData::GetY() const 
 945     wxCHECK_MSG( m_rect
, 0, _T("can't be called in this mode") ); 
 950 int wxListItemData::GetWidth() const 
 952     wxCHECK_MSG( m_rect
, 0, _T("can't be called in this mode") ); 
 954     return m_rect
->width
; 
 957 int wxListItemData::GetHeight() const 
 959     wxCHECK_MSG( m_rect
, 0, _T("can't be called in this mode") ); 
 961     return m_rect
->height
; 
 964 void wxListItemData::GetItem( wxListItem 
&info 
) const 
 966     info
.m_text 
= m_text
; 
 967     info
.m_image 
= m_image
; 
 968     info
.m_data 
= m_data
; 
 972         if ( m_attr
->HasTextColour() ) 
 973             info
.SetTextColour(m_attr
->GetTextColour()); 
 974         if ( m_attr
->HasBackgroundColour() ) 
 975             info
.SetBackgroundColour(m_attr
->GetBackgroundColour()); 
 976         if ( m_attr
->HasFont() ) 
 977             info
.SetFont(m_attr
->GetFont()); 
 981 //----------------------------------------------------------------------------- 
 983 //----------------------------------------------------------------------------- 
 985 void wxListHeaderData::Init() 
 996 wxListHeaderData::wxListHeaderData() 
1001 wxListHeaderData::wxListHeaderData( const wxListItem 
&item 
) 
1008 void wxListHeaderData::SetItem( const wxListItem 
&item 
) 
1010     m_mask 
= item
.m_mask
; 
1012     if ( m_mask 
& wxLIST_MASK_TEXT 
) 
1013         m_text 
= item
.m_text
; 
1015     if ( m_mask 
& wxLIST_MASK_IMAGE 
) 
1016         m_image 
= item
.m_image
; 
1018     if ( m_mask 
& wxLIST_MASK_FORMAT 
) 
1019         m_format 
= item
.m_format
; 
1021     if ( m_mask 
& wxLIST_MASK_WIDTH 
) 
1022         SetWidth(item
.m_width
); 
1025 void wxListHeaderData::SetPosition( int x
, int y 
) 
1031 void wxListHeaderData::SetHeight( int h 
) 
1036 void wxListHeaderData::SetWidth( int w 
) 
1040         m_width 
= WIDTH_COL_DEFAULT
; 
1041     else if (m_width 
< WIDTH_COL_MIN
) 
1042         m_width 
= WIDTH_COL_MIN
; 
1045 void wxListHeaderData::SetFormat( int format 
) 
1050 bool wxListHeaderData::HasImage() const 
1052     return m_image 
!= -1; 
1055 bool wxListHeaderData::IsHit( int x
, int y 
) const 
1057     return ((x 
>= m_xpos
) && (x 
<= m_xpos
+m_width
) && (y 
>= m_ypos
) && (y 
<= m_ypos
+m_height
)); 
1060 void wxListHeaderData::GetItem( wxListItem
& item 
) 
1062     item
.m_mask 
= m_mask
; 
1063     item
.m_text 
= m_text
; 
1064     item
.m_image 
= m_image
; 
1065     item
.m_format 
= m_format
; 
1066     item
.m_width 
= m_width
; 
1069 int wxListHeaderData::GetImage() const 
1074 int wxListHeaderData::GetWidth() const 
1079 int wxListHeaderData::GetFormat() const 
1084 //----------------------------------------------------------------------------- 
1086 //----------------------------------------------------------------------------- 
1088 inline int wxListLineData::GetMode() const 
1090     return m_owner
->GetListCtrl()->GetWindowStyleFlag() & wxLC_MASK_TYPE
; 
1093 inline bool wxListLineData::InReportView() const 
1095     return m_owner
->HasFlag(wxLC_REPORT
); 
1098 inline bool wxListLineData::IsVirtual() const 
1100     return m_owner
->IsVirtual(); 
1103 wxListLineData::wxListLineData( wxListMainWindow 
*owner 
) 
1107     if ( InReportView() ) 
1113         m_gi 
= new GeometryInfo
; 
1116     m_highlighted 
= FALSE
; 
1118     InitItems( GetMode() == wxLC_REPORT 
? m_owner
->GetColumnCount() : 1 ); 
1121 void wxListLineData::CalculateSize( wxDC 
*dc
, int spacing 
) 
1123     wxListItemDataList::compatibility_iterator node 
= m_items
.GetFirst(); 
1124     wxCHECK_RET( node
, _T("no subitems at all??") ); 
1126     wxListItemData 
*item 
= node
->GetData(); 
1131     switch ( GetMode() ) 
1134         case wxLC_SMALL_ICON
: 
1135             m_gi
->m_rectAll
.width 
= spacing
; 
1137             s 
= item
->GetText(); 
1142                 m_gi
->m_rectLabel
.width 
= 
1143                 m_gi
->m_rectLabel
.height 
= 0; 
1147                 dc
->GetTextExtent( s
, &lw
, &lh 
); 
1151                 m_gi
->m_rectAll
.height 
= spacing 
+ lh
; 
1153                     m_gi
->m_rectAll
.width 
= lw
; 
1155                 m_gi
->m_rectLabel
.width 
= lw
; 
1156                 m_gi
->m_rectLabel
.height 
= lh
; 
1159             if (item
->HasImage()) 
1162                 m_owner
->GetImageSize( item
->GetImage(), w
, h 
); 
1163                 m_gi
->m_rectIcon
.width 
= w 
+ 8; 
1164                 m_gi
->m_rectIcon
.height 
= h 
+ 8; 
1166                 if ( m_gi
->m_rectIcon
.width 
> m_gi
->m_rectAll
.width 
) 
1167                     m_gi
->m_rectAll
.width 
= m_gi
->m_rectIcon
.width
; 
1168                 if ( m_gi
->m_rectIcon
.height 
+ lh 
> m_gi
->m_rectAll
.height 
- 4 ) 
1169                     m_gi
->m_rectAll
.height 
= m_gi
->m_rectIcon
.height 
+ lh 
+ 4; 
1172             if ( item
->HasText() ) 
1174                 m_gi
->m_rectHighlight
.width 
= m_gi
->m_rectLabel
.width
; 
1175                 m_gi
->m_rectHighlight
.height 
= m_gi
->m_rectLabel
.height
; 
1177             else // no text, highlight the icon 
1179                 m_gi
->m_rectHighlight
.width 
= m_gi
->m_rectIcon
.width
; 
1180                 m_gi
->m_rectHighlight
.height 
= m_gi
->m_rectIcon
.height
; 
1185             s 
= item
->GetTextForMeasuring(); 
1187             dc
->GetTextExtent( s
, &lw
, &lh 
); 
1191             m_gi
->m_rectLabel
.width 
= lw
; 
1192             m_gi
->m_rectLabel
.height 
= lh
; 
1194             m_gi
->m_rectAll
.width 
= lw
; 
1195             m_gi
->m_rectAll
.height 
= lh
; 
1197             if (item
->HasImage()) 
1200                 m_owner
->GetImageSize( item
->GetImage(), w
, h 
); 
1201                 m_gi
->m_rectIcon
.width 
= w
; 
1202                 m_gi
->m_rectIcon
.height 
= h
; 
1204                 m_gi
->m_rectAll
.width 
+= 4 + w
; 
1205                 if (h 
> m_gi
->m_rectAll
.height
) 
1206                     m_gi
->m_rectAll
.height 
= h
; 
1209             m_gi
->m_rectHighlight
.width 
= m_gi
->m_rectAll
.width
; 
1210             m_gi
->m_rectHighlight
.height 
= m_gi
->m_rectAll
.height
; 
1214             wxFAIL_MSG( _T("unexpected call to SetSize") ); 
1218             wxFAIL_MSG( _T("unknown mode") ); 
1222 void wxListLineData::SetPosition( int x
, int y
, int spacing 
) 
1224     wxListItemDataList::compatibility_iterator node 
= m_items
.GetFirst(); 
1225     wxCHECK_RET( node
, _T("no subitems at all??") ); 
1227     wxListItemData 
*item 
= node
->GetData(); 
1229     switch ( GetMode() ) 
1232         case wxLC_SMALL_ICON
: 
1233             m_gi
->m_rectAll
.x 
= x
; 
1234             m_gi
->m_rectAll
.y 
= y
; 
1236             if ( item
->HasImage() ) 
1238                 m_gi
->m_rectIcon
.x 
= m_gi
->m_rectAll
.x 
+ 4 + 
1239                     (m_gi
->m_rectAll
.width 
- m_gi
->m_rectIcon
.width
) / 2; 
1240                 m_gi
->m_rectIcon
.y 
= m_gi
->m_rectAll
.y 
+ 4; 
1243             if ( item
->HasText() ) 
1245                 if (m_gi
->m_rectAll
.width 
> spacing
) 
1246                     m_gi
->m_rectLabel
.x 
= m_gi
->m_rectAll
.x 
+ 2; 
1248                     m_gi
->m_rectLabel
.x 
= m_gi
->m_rectAll
.x 
+ 2 + (spacing
/2) - (m_gi
->m_rectLabel
.width
/2); 
1249                 m_gi
->m_rectLabel
.y 
= m_gi
->m_rectAll
.y 
+ m_gi
->m_rectAll
.height 
+ 2 - m_gi
->m_rectLabel
.height
; 
1250                 m_gi
->m_rectHighlight
.x 
= m_gi
->m_rectLabel
.x 
- 2; 
1251                 m_gi
->m_rectHighlight
.y 
= m_gi
->m_rectLabel
.y 
- 2; 
1253             else // no text, highlight the icon 
1255                 m_gi
->m_rectHighlight
.x 
= m_gi
->m_rectIcon
.x 
- 4; 
1256                 m_gi
->m_rectHighlight
.y 
= m_gi
->m_rectIcon
.y 
- 4; 
1261             m_gi
->m_rectAll
.x 
= x
; 
1262             m_gi
->m_rectAll
.y 
= y
; 
1264             m_gi
->m_rectHighlight
.x 
= m_gi
->m_rectAll
.x
; 
1265             m_gi
->m_rectHighlight
.y 
= m_gi
->m_rectAll
.y
; 
1266             m_gi
->m_rectLabel
.y 
= m_gi
->m_rectAll
.y 
+ 2; 
1268             if (item
->HasImage()) 
1270                 m_gi
->m_rectIcon
.x 
= m_gi
->m_rectAll
.x 
+ 2; 
1271                 m_gi
->m_rectIcon
.y 
= m_gi
->m_rectAll
.y 
+ 2; 
1272                 m_gi
->m_rectLabel
.x 
= m_gi
->m_rectAll
.x 
+ 6 + m_gi
->m_rectIcon
.width
; 
1276                 m_gi
->m_rectLabel
.x 
= m_gi
->m_rectAll
.x 
+ 2; 
1281             wxFAIL_MSG( _T("unexpected call to SetPosition") ); 
1285             wxFAIL_MSG( _T("unknown mode") ); 
1289 void wxListLineData::InitItems( int num 
) 
1291     for (int i 
= 0; i 
< num
; i
++) 
1292         m_items
.Append( new wxListItemData(m_owner
) ); 
1295 void wxListLineData::SetItem( int index
, const wxListItem 
&info 
) 
1297     wxListItemDataList::compatibility_iterator node 
= m_items
.Item( index 
); 
1298     wxCHECK_RET( node
, _T("invalid column index in SetItem") ); 
1300     wxListItemData 
*item 
= node
->GetData(); 
1301     item
->SetItem( info 
); 
1304 void wxListLineData::GetItem( int index
, wxListItem 
&info 
) 
1306     wxListItemDataList::compatibility_iterator node 
= m_items
.Item( index 
); 
1309         wxListItemData 
*item 
= node
->GetData(); 
1310         item
->GetItem( info 
); 
1314 wxString 
wxListLineData::GetText(int index
) const 
1318     wxListItemDataList::compatibility_iterator node 
= m_items
.Item( index 
); 
1321         wxListItemData 
*item 
= node
->GetData(); 
1322         s 
= item
->GetText(); 
1328 void wxListLineData::SetText( int index
, const wxString s 
) 
1330     wxListItemDataList::compatibility_iterator node 
= m_items
.Item( index 
); 
1333         wxListItemData 
*item 
= node
->GetData(); 
1338 void wxListLineData::SetImage( int index
, int image 
) 
1340     wxListItemDataList::compatibility_iterator node 
= m_items
.Item( index 
); 
1341     wxCHECK_RET( node
, _T("invalid column index in SetImage()") ); 
1343     wxListItemData 
*item 
= node
->GetData(); 
1344     item
->SetImage(image
); 
1347 int wxListLineData::GetImage( int index 
) const 
1349     wxListItemDataList::compatibility_iterator node 
= m_items
.Item( index 
); 
1350     wxCHECK_MSG( node
, -1, _T("invalid column index in GetImage()") ); 
1352     wxListItemData 
*item 
= node
->GetData(); 
1353     return item
->GetImage(); 
1356 wxListItemAttr 
*wxListLineData::GetAttr() const 
1358     wxListItemDataList::compatibility_iterator node 
= m_items
.GetFirst(); 
1359     wxCHECK_MSG( node
, NULL
, _T("invalid column index in GetAttr()") ); 
1361     wxListItemData 
*item 
= node
->GetData(); 
1362     return item
->GetAttr(); 
1365 void wxListLineData::SetAttr(wxListItemAttr 
*attr
) 
1367     wxListItemDataList::compatibility_iterator node 
= m_items
.GetFirst(); 
1368     wxCHECK_RET( node
, _T("invalid column index in SetAttr()") ); 
1370     wxListItemData 
*item 
= node
->GetData(); 
1371     item
->SetAttr(attr
); 
1374 bool wxListLineData::SetAttributes(wxDC 
*dc
, 
1375                                    const wxListItemAttr 
*attr
, 
1378     wxWindow 
*listctrl 
= m_owner
->GetParent(); 
1382     // don't use foreground colour for drawing highlighted items - this might 
1383     // make them completely invisible (and there is no way to do bit 
1384     // arithmetics on wxColour, unfortunately) 
1388         colText 
= wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHTTEXT
); 
1392         if ( attr 
&& attr
->HasTextColour() ) 
1394             colText 
= attr
->GetTextColour(); 
1398             colText 
= listctrl
->GetForegroundColour(); 
1402     dc
->SetTextForeground(colText
); 
1406     if ( attr 
&& attr
->HasFont() ) 
1408         font 
= attr
->GetFont(); 
1412         font 
= listctrl
->GetFont(); 
1418     bool hasBgCol 
= attr 
&& attr
->HasBackgroundColour(); 
1419     if ( highlighted 
|| hasBgCol 
) 
1423             dc
->SetBrush( *m_owner
->GetHighlightBrush() ); 
1427             dc
->SetBrush(wxBrush(attr
->GetBackgroundColour(), wxSOLID
)); 
1430         dc
->SetPen( *wxTRANSPARENT_PEN 
); 
1438 void wxListLineData::Draw( wxDC 
*dc 
) 
1440     wxListItemDataList::compatibility_iterator node 
= m_items
.GetFirst(); 
1441     wxCHECK_RET( node
, _T("no subitems at all??") ); 
1443     bool highlighted 
= IsHighlighted(); 
1445     wxListItemAttr 
*attr 
= GetAttr(); 
1447     if ( SetAttributes(dc
, attr
, highlighted
) ) 
1449         dc
->DrawRectangle( m_gi
->m_rectHighlight 
); 
1452     // just for debugging to better see where the items are 
1454     dc
->SetPen(*wxRED_PEN
); 
1455     dc
->SetBrush(*wxTRANSPARENT_BRUSH
); 
1456     dc
->DrawRectangle( m_gi
->m_rectAll 
); 
1457     dc
->SetPen(*wxGREEN_PEN
); 
1458     dc
->DrawRectangle( m_gi
->m_rectIcon 
); 
1461     wxListItemData 
*item 
= node
->GetData(); 
1462     if (item
->HasImage()) 
1464         // centre the image inside our rectangle, this looks nicer when items 
1465         // ae aligned in a row 
1466         const wxRect
& rectIcon 
= m_gi
->m_rectIcon
; 
1468         m_owner
->DrawImage(item
->GetImage(), dc
, rectIcon
.x
, rectIcon
.y
); 
1471     if (item
->HasText()) 
1473         const wxRect
& rectLabel 
= m_gi
->m_rectLabel
; 
1475         wxDCClipper 
clipper(*dc
, rectLabel
); 
1476         dc
->DrawText(item
->GetText(), rectLabel
.x
, rectLabel
.y
); 
1480 void wxListLineData::DrawInReportMode( wxDC 
*dc
, 
1482                                        const wxRect
& rectHL
, 
1485     // TODO: later we should support setting different attributes for 
1486     //       different columns - to do it, just add "col" argument to 
1487     //       GetAttr() and move these lines into the loop below 
1488     wxListItemAttr 
*attr 
= GetAttr(); 
1489     if ( SetAttributes(dc
, attr
, highlighted
) ) 
1491         dc
->DrawRectangle( rectHL 
); 
1494     wxCoord x 
= rect
.x 
+ HEADER_OFFSET_X
, 
1495             y 
= rect
.y 
+ (LINE_SPACING 
+ EXTRA_HEIGHT
) / 2; 
1498     for ( wxListItemDataList::compatibility_iterator node 
= m_items
.GetFirst(); 
1500           node 
= node
->GetNext(), col
++ ) 
1502         wxListItemData 
*item 
= node
->GetData(); 
1504         int width 
= m_owner
->GetColumnWidth(col
); 
1508         if ( item
->HasImage() ) 
1511             m_owner
->DrawImage( item
->GetImage(), dc
, xOld
, y 
); 
1512             m_owner
->GetImageSize( item
->GetImage(), ix
, iy 
); 
1514             ix 
+= IMAGE_MARGIN_IN_REPORT_MODE
; 
1520         wxDCClipper 
clipper(*dc
, xOld
, y
, width 
- 8, rect
.height
); 
1522         if ( item
->HasText() ) 
1524             DrawTextFormatted(dc
, item
->GetText(), col
, xOld
, y
, width 
- 8); 
1529 void wxListLineData::DrawTextFormatted(wxDC 
*dc
, 
1530                                        const wxString 
&text
, 
1536     wxString drawntext
, ellipsis
; 
1537     wxCoord w
, h
, base_w
; 
1540     // determine if the string can fit inside the current width 
1541     dc
->GetTextExtent(text
, &w
, &h
); 
1544         // it can, draw it using the items alignment 
1545         m_owner
->GetColumn(col
, item
); 
1546         switch ( item
.GetAlign() ) 
1549                 wxFAIL_MSG( _T("unknown list item format") ); 
1552             case wxLIST_FORMAT_LEFT
: 
1556             case wxLIST_FORMAT_RIGHT
: 
1560             case wxLIST_FORMAT_CENTER
: 
1561                 x 
+= (width 
- w
) / 2; 
1565         dc
->DrawText(text
, x
, y
); 
1567     else // otherwise, truncate and add an ellipsis if possible 
1569         // determine the base width 
1570         ellipsis 
= wxString(wxT("...")); 
1571         dc
->GetTextExtent(ellipsis
, &base_w
, &h
); 
1573         // continue until we have enough space or only one character left 
1575         size_t len 
= text
.Length(); 
1576         drawntext 
= text
.Left(len
); 
1579             dc
->GetTextExtent(drawntext
.Last(), &w_c
, &h_c
); 
1580             drawntext
.RemoveLast(); 
1583             if (w 
+ base_w 
<= width
) 
1587         // if still not enough space, remove ellipsis characters 
1588         while (ellipsis
.Length() > 0 && w 
+ base_w 
> width
) 
1590             ellipsis 
= ellipsis
.Left(ellipsis
.Length() - 1); 
1591             dc
->GetTextExtent(ellipsis
, &base_w
, &h
); 
1594         // now draw the text 
1595         dc
->DrawText(drawntext
, x
, y
); 
1596         dc
->DrawText(ellipsis
, x 
+ w
, y
); 
1600 bool wxListLineData::Highlight( bool on 
) 
1602     wxCHECK_MSG( !m_owner
->IsVirtual(), FALSE
, _T("unexpected call to Highlight") ); 
1604     if ( on 
== m_highlighted 
) 
1612 void wxListLineData::ReverseHighlight( void ) 
1614     Highlight(!IsHighlighted()); 
1617 //----------------------------------------------------------------------------- 
1618 //  wxListHeaderWindow 
1619 //----------------------------------------------------------------------------- 
1621 IMPLEMENT_DYNAMIC_CLASS(wxListHeaderWindow
,wxWindow
) 
1623 BEGIN_EVENT_TABLE(wxListHeaderWindow
,wxWindow
) 
1624     EVT_PAINT         (wxListHeaderWindow::OnPaint
) 
1625     EVT_MOUSE_EVENTS  (wxListHeaderWindow::OnMouse
) 
1626     EVT_SET_FOCUS     (wxListHeaderWindow::OnSetFocus
) 
1629 void wxListHeaderWindow::Init() 
1631     m_currentCursor 
= (wxCursor 
*) NULL
; 
1632     m_isDragging 
= FALSE
; 
1636 wxListHeaderWindow::wxListHeaderWindow() 
1640     m_owner 
= (wxListMainWindow 
*) NULL
; 
1641     m_resizeCursor 
= (wxCursor 
*) NULL
; 
1644 wxListHeaderWindow::wxListHeaderWindow( wxWindow 
*win
, 
1646                                         wxListMainWindow 
*owner
, 
1650                                         const wxString 
&name 
) 
1651                   : wxWindow( win
, id
, pos
, size
, style
, name 
) 
1656     m_resizeCursor 
= new wxCursor( wxCURSOR_SIZEWE 
); 
1658     SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_BTNFACE 
) ); 
1661 wxListHeaderWindow::~wxListHeaderWindow() 
1663     delete m_resizeCursor
; 
1666 #ifdef __WXUNIVERSAL__ 
1667 #include "wx/univ/renderer.h" 
1668 #include "wx/univ/theme.h" 
1671 // shift the DC origin to match the position of the main window horz 
1672 // scrollbar: this allows us to always use logical coords 
1673 void wxListHeaderWindow::AdjustDC(wxDC
& dc
) 
1676     m_owner
->GetScrollPixelsPerUnit( &xpix
, NULL 
); 
1679     m_owner
->GetViewStart( &x
, NULL 
); 
1681     // account for the horz scrollbar offset 
1682     dc
.SetDeviceOrigin( -x 
* xpix
, 0 ); 
1685 void wxListHeaderWindow::OnPaint( wxPaintEvent 
&WXUNUSED(event
) ) 
1687     wxPaintDC 
dc( this ); 
1694     dc
.SetFont( GetFont() ); 
1696     // width and height of the entire header window 
1698     GetClientSize( &w
, &h 
); 
1699     m_owner
->CalcUnscrolledPosition(w
, 0, &w
, NULL
); 
1701     dc
.SetBackgroundMode(wxTRANSPARENT
); 
1703     // do *not* use the listctrl colour for headers - one day we will have a 
1704     // function to set it separately 
1705     //dc.SetTextForeground( *wxBLACK ); 
1706     dc
.SetTextForeground(wxSystemSettings:: 
1707                             GetSystemColour( wxSYS_COLOUR_WINDOWTEXT 
)); 
1709     int x 
= HEADER_OFFSET_X
; 
1711     int numColumns 
= m_owner
->GetColumnCount(); 
1713     for ( int i 
= 0; i 
< numColumns 
&& x 
< w
; i
++ ) 
1715         m_owner
->GetColumn( i
, item 
); 
1716         int wCol 
= item
.m_width
; 
1718         // the width of the rect to draw: make it smaller to fit entirely 
1719         // inside the column rect 
1722         wxRendererNative::Get().DrawHeaderButton
 
1726                                     wxRect(x
, HEADER_OFFSET_Y
, cw
, h 
- 2), 
1727                                     m_parent
->IsEnabled() ? 0 
1728                                                           : wxCONTROL_DISABLED
 
1731         // see if we have enough space for the column label 
1733         // for this we need the width of the text 
1736         dc
.GetTextExtent(item
.GetText(), &wLabel
, &hLabel
); 
1737         wLabel 
+= 2*EXTRA_WIDTH
; 
1739         // and the width of the icon, if any 
1740         static const int MARGIN_BETWEEN_TEXT_AND_ICON 
= 2; 
1741         int ix 
= 0,     // init them just to suppress the compiler warnings 
1743         const int image 
= item
.m_image
; 
1744         wxImageListType 
*imageList
; 
1747             imageList 
= m_owner
->m_small_image_list
; 
1750                 imageList
->GetSize(image
, ix
, iy
); 
1751                 wLabel 
+= ix 
+ MARGIN_BETWEEN_TEXT_AND_ICON
; 
1759         // ignore alignment if there is not enough space anyhow 
1761         switch ( wLabel 
< cw 
? item
.GetAlign() : wxLIST_FORMAT_LEFT 
) 
1764                 wxFAIL_MSG( _T("unknown list item format") ); 
1767             case wxLIST_FORMAT_LEFT
: 
1771             case wxLIST_FORMAT_RIGHT
: 
1772                 xAligned 
= x 
+ cw 
- wLabel
; 
1775             case wxLIST_FORMAT_CENTER
: 
1776                 xAligned 
= x 
+ (cw 
- wLabel
) / 2; 
1781         // if we have an image, draw it on the right of the label 
1788                         xAligned 
+ wLabel 
- ix 
- MARGIN_BETWEEN_TEXT_AND_ICON
, 
1789                         HEADER_OFFSET_Y 
+ (h 
- 4 - iy
)/2, 
1790                         wxIMAGELIST_DRAW_TRANSPARENT
 
1793             cw 
-= ix 
+ MARGIN_BETWEEN_TEXT_AND_ICON
; 
1796         // draw the text clipping it so that it doesn't overwrite the column 
1798         wxDCClipper 
clipper(dc
, x
, HEADER_OFFSET_Y
, cw
, h 
- 4 ); 
1800         dc
.DrawText( item
.GetText(), 
1801                      xAligned 
+ EXTRA_WIDTH
, h 
/ 2 - hLabel 
/ 2 ); //HEADER_OFFSET_Y + EXTRA_HEIGHT ); 
1809 void wxListHeaderWindow::DrawCurrent() 
1811     int x1 
= m_currentX
; 
1813     m_owner
->ClientToScreen( &x1
, &y1 
); 
1815     int x2 
= m_currentX
; 
1817     m_owner
->GetClientSize( NULL
, &y2 
); 
1818     m_owner
->ClientToScreen( &x2
, &y2 
); 
1821     dc
.SetLogicalFunction( wxINVERT 
); 
1822     dc
.SetPen( wxPen( *wxBLACK
, 2, wxSOLID 
) ); 
1823     dc
.SetBrush( *wxTRANSPARENT_BRUSH 
); 
1827     dc
.DrawLine( x1
, y1
, x2
, y2 
); 
1829     dc
.SetLogicalFunction( wxCOPY 
); 
1831     dc
.SetPen( wxNullPen 
); 
1832     dc
.SetBrush( wxNullBrush 
); 
1835 void wxListHeaderWindow::OnMouse( wxMouseEvent 
&event 
) 
1837     // we want to work with logical coords 
1839     m_owner
->CalcUnscrolledPosition(event
.GetX(), 0, &x
, NULL
); 
1840     int y 
= event
.GetY(); 
1844         SendListEvent(wxEVT_COMMAND_LIST_COL_DRAGGING
, event
.GetPosition()); 
1846         // we don't draw the line beyond our window, but we allow dragging it 
1849         GetClientSize( &w
, NULL 
); 
1850         m_owner
->CalcUnscrolledPosition(w
, 0, &w
, NULL
); 
1853         // erase the line if it was drawn 
1854         if ( m_currentX 
< w 
) 
1857         if (event
.ButtonUp()) 
1860             m_isDragging 
= FALSE
; 
1862             m_owner
->SetColumnWidth( m_column
, m_currentX 
- m_minX 
); 
1863             SendListEvent(wxEVT_COMMAND_LIST_COL_END_DRAG
, event
.GetPosition()); 
1870                 m_currentX 
= m_minX 
+ 7; 
1872             // draw in the new location 
1873             if ( m_currentX 
< w 
) 
1877     else // not dragging 
1880         bool hit_border 
= FALSE
; 
1882         // end of the current column 
1885         // find the column where this event occured 
1887             countCol 
= m_owner
->GetColumnCount(); 
1888         for (col 
= 0; col 
< countCol
; col
++) 
1890             xpos 
+= m_owner
->GetColumnWidth( col 
); 
1893             if ( (abs(x
-xpos
) < 3) && (y 
< 22) ) 
1895                 // near the column border 
1902                 // inside the column 
1909         if ( col 
== countCol 
) 
1912         if (event
.LeftDown() || event
.RightUp()) 
1914             if (hit_border 
&& event
.LeftDown()) 
1916                 if ( SendListEvent(wxEVT_COMMAND_LIST_COL_BEGIN_DRAG
, 
1917                                    event
.GetPosition()) ) 
1919                     m_isDragging 
= TRUE
; 
1924                 //else: column resizing was vetoed by the user code 
1926             else // click on a column 
1928                 SendListEvent( event
.LeftDown() 
1929                                     ? wxEVT_COMMAND_LIST_COL_CLICK
 
1930                                     : wxEVT_COMMAND_LIST_COL_RIGHT_CLICK
, 
1931                                 event
.GetPosition()); 
1934         else if (event
.Moving()) 
1939                 setCursor 
= m_currentCursor 
== wxSTANDARD_CURSOR
; 
1940                 m_currentCursor 
= m_resizeCursor
; 
1944                 setCursor 
= m_currentCursor 
!= wxSTANDARD_CURSOR
; 
1945                 m_currentCursor 
= wxSTANDARD_CURSOR
; 
1949                 SetCursor(*m_currentCursor
); 
1954 void wxListHeaderWindow::OnSetFocus( wxFocusEvent 
&WXUNUSED(event
) ) 
1956     m_owner
->SetFocus(); 
1960 bool wxListHeaderWindow::SendListEvent(wxEventType type
, wxPoint pos
) 
1962     wxWindow 
*parent 
= GetParent(); 
1963     wxListEvent 
le( type
, parent
->GetId() ); 
1964     le
.SetEventObject( parent 
); 
1965     le
.m_pointDrag 
= pos
; 
1967     // the position should be relative to the parent window, not 
1968     // this one for compatibility with MSW and common sense: the 
1969     // user code doesn't know anything at all about this header 
1970     // window, so why should it get positions relative to it? 
1971     le
.m_pointDrag
.y 
-= GetSize().y
; 
1973     le
.m_col 
= m_column
; 
1974     return !parent
->GetEventHandler()->ProcessEvent( le 
) || le
.IsAllowed(); 
1977 //----------------------------------------------------------------------------- 
1978 // wxListRenameTimer (internal) 
1979 //----------------------------------------------------------------------------- 
1981 wxListRenameTimer::wxListRenameTimer( wxListMainWindow 
*owner 
) 
1986 void wxListRenameTimer::Notify() 
1988     m_owner
->OnRenameTimer(); 
1991 //----------------------------------------------------------------------------- 
1992 // wxListTextCtrl (internal) 
1993 //----------------------------------------------------------------------------- 
1995 BEGIN_EVENT_TABLE(wxListTextCtrl
,wxTextCtrl
) 
1996     EVT_CHAR           (wxListTextCtrl::OnChar
) 
1997     EVT_KEY_UP         (wxListTextCtrl::OnKeyUp
) 
1998     EVT_KILL_FOCUS     (wxListTextCtrl::OnKillFocus
) 
2001 wxListTextCtrl::wxListTextCtrl(wxListMainWindow 
*owner
, size_t itemEdit
) 
2002               : m_startValue(owner
->GetItemText(itemEdit
)), 
2003                 m_itemEdited(itemEdit
) 
2008     wxRect rectLabel 
= owner
->GetLineLabelRect(itemEdit
); 
2010     m_owner
->CalcScrolledPosition(rectLabel
.x
, rectLabel
.y
, 
2011                                   &rectLabel
.x
, &rectLabel
.y
); 
2013     (void)Create(owner
, wxID_ANY
, m_startValue
, 
2014                  wxPoint(rectLabel
.x
-4,rectLabel
.y
-4), 
2015                  wxSize(rectLabel
.width
+11,rectLabel
.height
+8)); 
2018 void wxListTextCtrl::Finish() 
2022         wxPendingDelete
.Append(this); 
2026         m_owner
->SetFocus(); 
2030 bool wxListTextCtrl::AcceptChanges() 
2032     const wxString value 
= GetValue(); 
2034     if ( value 
== m_startValue 
) 
2036         // nothing changed, always accept 
2040     if ( !m_owner
->OnRenameAccept(m_itemEdited
, value
) ) 
2042         // vetoed by the user 
2046     // accepted, do rename the item 
2047     m_owner
->SetItemText(m_itemEdited
, value
); 
2052 void wxListTextCtrl::OnChar( wxKeyEvent 
&event 
) 
2054     switch ( event
.m_keyCode 
) 
2057             if ( !AcceptChanges() ) 
2059                 // vetoed by the user code 
2062             //else: fall through 
2066             m_owner
->OnRenameCancelled( m_itemEdited 
); 
2074 void wxListTextCtrl::OnKeyUp( wxKeyEvent 
&event 
) 
2082     // auto-grow the textctrl: 
2083     wxSize parentSize 
= m_owner
->GetSize(); 
2084     wxPoint myPos 
= GetPosition(); 
2085     wxSize mySize 
= GetSize(); 
2087     GetTextExtent(GetValue() + _T("MM"), &sx
, &sy
); 
2088     if (myPos
.x 
+ sx 
> parentSize
.x
) 
2089         sx 
= parentSize
.x 
- myPos
.x
; 
2097 void wxListTextCtrl::OnKillFocus( wxFocusEvent 
&event 
) 
2101         // We must finish regardless of success, otherwise we'll get focus problems 
2104         if ( !AcceptChanges() ) 
2105             m_owner
->OnRenameCancelled( m_itemEdited 
); 
2111 //----------------------------------------------------------------------------- 
2113 //----------------------------------------------------------------------------- 
2115 IMPLEMENT_DYNAMIC_CLASS(wxListMainWindow
,wxScrolledWindow
) 
2117 BEGIN_EVENT_TABLE(wxListMainWindow
,wxScrolledWindow
) 
2118   EVT_PAINT          (wxListMainWindow::OnPaint
) 
2119   EVT_MOUSE_EVENTS   (wxListMainWindow::OnMouse
) 
2120   EVT_CHAR           (wxListMainWindow::OnChar
) 
2121   EVT_KEY_DOWN       (wxListMainWindow::OnKeyDown
) 
2122   EVT_SET_FOCUS      (wxListMainWindow::OnSetFocus
) 
2123   EVT_KILL_FOCUS     (wxListMainWindow::OnKillFocus
) 
2124   EVT_SCROLLWIN      (wxListMainWindow::OnScroll
) 
2127 void wxListMainWindow::Init() 
2132     m_lineTo 
= (size_t)-1; 
2138     m_small_image_list 
= (wxImageListType 
*) NULL
; 
2139     m_normal_image_list 
= (wxImageListType 
*) NULL
; 
2141     m_small_spacing 
= 30; 
2142     m_normal_spacing 
= 40; 
2146     m_isCreated 
= FALSE
; 
2148     m_lastOnSame 
= FALSE
; 
2149     m_renameTimer 
= new wxListRenameTimer( this ); 
2153     m_lineBeforeLastClicked 
= (size_t)-1; 
2158 wxListMainWindow::wxListMainWindow() 
2163     m_highlightUnfocusedBrush 
= (wxBrush 
*) NULL
; 
2166 wxListMainWindow::wxListMainWindow( wxWindow 
*parent
, 
2171                                     const wxString 
&name 
) 
2172                 : wxScrolledWindow( parent
, id
, pos
, size
, 
2173                                     style 
| wxHSCROLL 
| wxVSCROLL
, name 
) 
2177     m_highlightBrush 
= new wxBrush
 
2179                             wxSystemSettings::GetColour
 
2181                                 wxSYS_COLOUR_HIGHLIGHT
 
2186     m_highlightUnfocusedBrush 
= new wxBrush
 
2188                                        wxSystemSettings::GetColour
 
2190                                            wxSYS_COLOUR_BTNSHADOW
 
2198     SetScrollbars( 0, 0, 0, 0, 0, 0 ); 
2200     SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_LISTBOX 
) ); 
2203 wxListMainWindow::~wxListMainWindow() 
2206     WX_CLEAR_LIST(wxListHeaderDataList
, m_columns
); 
2208     delete m_highlightBrush
; 
2209     delete m_highlightUnfocusedBrush
; 
2211     delete m_renameTimer
; 
2214 void wxListMainWindow::CacheLineData(size_t line
) 
2216     wxGenericListCtrl 
*listctrl 
= GetListCtrl(); 
2218     wxListLineData 
*ld 
= GetDummyLine(); 
2220     size_t countCol 
= GetColumnCount(); 
2221     for ( size_t col 
= 0; col 
< countCol
; col
++ ) 
2223         ld
->SetText(col
, listctrl
->OnGetItemText(line
, col
)); 
2226     ld
->SetImage(listctrl
->OnGetItemImage(line
)); 
2227     ld
->SetAttr(listctrl
->OnGetItemAttr(line
)); 
2230 wxListLineData 
*wxListMainWindow::GetDummyLine() const 
2232     wxASSERT_MSG( !IsEmpty(), _T("invalid line index") ); 
2234     wxASSERT_MSG( IsVirtual(), _T("GetDummyLine() shouldn't be called") ); 
2236     wxListMainWindow 
*self 
= wxConstCast(this, wxListMainWindow
); 
2238     // we need to recreate the dummy line if the number of columns in the 
2239     // control changed as it would have the incorrect number of fields 
2241     if ( !m_lines
.IsEmpty() && 
2242             m_lines
[0].m_items
.GetCount() != (size_t)GetColumnCount() ) 
2244         self
->m_lines
.Clear(); 
2247     if ( m_lines
.IsEmpty() ) 
2249         wxListLineData 
*line 
= new wxListLineData(self
); 
2250         self
->m_lines
.Add(line
); 
2252         // don't waste extra memory -- there never going to be anything 
2253         // else/more in this array 
2254         self
->m_lines
.Shrink(); 
2260 // ---------------------------------------------------------------------------- 
2261 // line geometry (report mode only) 
2262 // ---------------------------------------------------------------------------- 
2264 wxCoord 
wxListMainWindow::GetLineHeight() const 
2266     // we cache the line height as calling GetTextExtent() is slow 
2267     if ( !m_lineHeight 
) 
2269         wxListMainWindow 
*self 
= wxConstCast(this, wxListMainWindow
); 
2271         wxClientDC 
dc( self 
); 
2272         dc
.SetFont( GetFont() ); 
2275         dc
.GetTextExtent(_T("H"), NULL
, &y
); 
2277         if ( m_small_image_list 
&& m_small_image_list
->GetImageCount() ) 
2281             m_small_image_list
->GetSize(0, iw
, ih
); 
2286         self
->m_lineHeight 
= y 
+ LINE_SPACING
; 
2289     return m_lineHeight
; 
2292 wxCoord 
wxListMainWindow::GetLineY(size_t line
) const 
2294     wxASSERT_MSG( HasFlag(wxLC_REPORT
), _T("only works in report mode") ); 
2296     return LINE_SPACING 
+ line
*GetLineHeight(); 
2299 wxRect 
wxListMainWindow::GetLineRect(size_t line
) const 
2301     if ( !InReportView() ) 
2302         return GetLine(line
)->m_gi
->m_rectAll
; 
2305     rect
.x 
= HEADER_OFFSET_X
; 
2306     rect
.y 
= GetLineY(line
); 
2307     rect
.width 
= GetHeaderWidth(); 
2308     rect
.height 
= GetLineHeight(); 
2313 wxRect 
wxListMainWindow::GetLineLabelRect(size_t line
) const 
2315     if ( !InReportView() ) 
2316         return GetLine(line
)->m_gi
->m_rectLabel
; 
2319     rect
.x 
= HEADER_OFFSET_X
; 
2320     rect
.y 
= GetLineY(line
); 
2321     rect
.width 
= GetColumnWidth(0); 
2322     rect
.height 
= GetLineHeight(); 
2327 wxRect 
wxListMainWindow::GetLineIconRect(size_t line
) const 
2329     if ( !InReportView() ) 
2330         return GetLine(line
)->m_gi
->m_rectIcon
; 
2332     wxListLineData 
*ld 
= GetLine(line
); 
2333     wxASSERT_MSG( ld
->HasImage(), _T("should have an image") ); 
2336     rect
.x 
= HEADER_OFFSET_X
; 
2337     rect
.y 
= GetLineY(line
); 
2338     GetImageSize(ld
->GetImage(), rect
.width
, rect
.height
); 
2343 wxRect 
wxListMainWindow::GetLineHighlightRect(size_t line
) const 
2345     return InReportView() ? GetLineRect(line
) 
2346                           : GetLine(line
)->m_gi
->m_rectHighlight
; 
2349 long wxListMainWindow::HitTestLine(size_t line
, int x
, int y
) const 
2351     wxASSERT_MSG( line 
< GetItemCount(), _T("invalid line in HitTestLine") ); 
2353     wxListLineData 
*ld 
= GetLine(line
); 
2355     if ( ld
->HasImage() && GetLineIconRect(line
).Inside(x
, y
) ) 
2356         return wxLIST_HITTEST_ONITEMICON
; 
2358     // VS: Testing for "ld->HasText() || InReportView()" instead of 
2359     //     "ld->HasText()" is needed to make empty lines in report view 
2361     if ( ld
->HasText() || InReportView() ) 
2363         wxRect rect 
= InReportView() ? GetLineRect(line
) 
2364                                      : GetLineLabelRect(line
); 
2366         if ( rect
.Inside(x
, y
) ) 
2367             return wxLIST_HITTEST_ONITEMLABEL
; 
2373 // ---------------------------------------------------------------------------- 
2374 // highlight (selection) handling 
2375 // ---------------------------------------------------------------------------- 
2377 bool wxListMainWindow::IsHighlighted(size_t line
) const 
2381         return m_selStore
.IsSelected(line
); 
2385         wxListLineData 
*ld 
= GetLine(line
); 
2386         wxCHECK_MSG( ld
, FALSE
, _T("invalid index in IsHighlighted") ); 
2388         return ld
->IsHighlighted(); 
2392 void wxListMainWindow::HighlightLines( size_t lineFrom
, 
2398         wxArrayInt linesChanged
; 
2399         if ( !m_selStore
.SelectRange(lineFrom
, lineTo
, highlight
, 
2402             // meny items changed state, refresh everything 
2403             RefreshLines(lineFrom
, lineTo
); 
2405         else // only a few items changed state, refresh only them 
2407             size_t count 
= linesChanged
.GetCount(); 
2408             for ( size_t n 
= 0; n 
< count
; n
++ ) 
2410                 RefreshLine(linesChanged
[n
]); 
2414     else // iterate over all items in non report view 
2416         for ( size_t line 
= lineFrom
; line 
<= lineTo
; line
++ ) 
2418             if ( HighlightLine(line
, highlight
) ) 
2426 bool wxListMainWindow::HighlightLine( size_t line
, bool highlight 
) 
2432         changed 
= m_selStore
.SelectItem(line
, highlight
); 
2436         wxListLineData 
*ld 
= GetLine(line
); 
2437         wxCHECK_MSG( ld
, FALSE
, _T("invalid index in HighlightLine") ); 
2439         changed 
= ld
->Highlight(highlight
); 
2444         SendNotify( line
, highlight 
? wxEVT_COMMAND_LIST_ITEM_SELECTED
 
2445                                     : wxEVT_COMMAND_LIST_ITEM_DESELECTED 
); 
2451 void wxListMainWindow::RefreshLine( size_t line 
) 
2453     if ( HasFlag(wxLC_REPORT
) ) 
2455         size_t visibleFrom
, visibleTo
; 
2456         GetVisibleLinesRange(&visibleFrom
, &visibleTo
); 
2458         if ( line 
< visibleFrom 
|| line 
> visibleTo 
) 
2462     wxRect rect 
= GetLineRect(line
); 
2464     CalcScrolledPosition( rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
2465     RefreshRect( rect 
); 
2468 void wxListMainWindow::RefreshLines( size_t lineFrom
, size_t lineTo 
) 
2470     // we suppose that they are ordered by caller 
2471     wxASSERT_MSG( lineFrom 
<= lineTo
, _T("indices in disorder") ); 
2473     wxASSERT_MSG( lineTo 
< GetItemCount(), _T("invalid line range") ); 
2475     if ( HasFlag(wxLC_REPORT
) ) 
2477         size_t visibleFrom
, visibleTo
; 
2478         GetVisibleLinesRange(&visibleFrom
, &visibleTo
); 
2480         if ( lineFrom 
< visibleFrom 
) 
2481             lineFrom 
= visibleFrom
; 
2482         if ( lineTo 
> visibleTo 
) 
2487         rect
.y 
= GetLineY(lineFrom
); 
2488         rect
.width 
= GetClientSize().x
; 
2489         rect
.height 
= GetLineY(lineTo
) - rect
.y 
+ GetLineHeight(); 
2491         CalcScrolledPosition( rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
2492         RefreshRect( rect 
); 
2496         // TODO: this should be optimized... 
2497         for ( size_t line 
= lineFrom
; line 
<= lineTo
; line
++ ) 
2504 void wxListMainWindow::RefreshAfter( size_t lineFrom 
) 
2506     if ( HasFlag(wxLC_REPORT
) ) 
2508         size_t visibleFrom
, visibleTo
; 
2509         GetVisibleLinesRange(&visibleFrom
, &visibleTo
); 
2511         if ( lineFrom 
< visibleFrom 
) 
2512             lineFrom 
= visibleFrom
; 
2513         else if ( lineFrom 
> visibleTo 
) 
2518         rect
.y 
= GetLineY(lineFrom
); 
2519         CalcScrolledPosition( rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
2521         wxSize size 
= GetClientSize(); 
2522         rect
.width 
= size
.x
; 
2523         // refresh till the bottom of the window 
2524         rect
.height 
= size
.y 
- rect
.y
; 
2526         RefreshRect( rect 
); 
2530         // TODO: how to do it more efficiently? 
2535 void wxListMainWindow::RefreshSelected() 
2541     if ( InReportView() ) 
2543         GetVisibleLinesRange(&from
, &to
); 
2548         to 
= GetItemCount() - 1; 
2551     if ( HasCurrent() && m_current 
>= from 
&& m_current 
<= to 
) 
2553         RefreshLine(m_current
); 
2556     for ( size_t line 
= from
; line 
<= to
; line
++ ) 
2558         // NB: the test works as expected even if m_current == -1 
2559         if ( line 
!= m_current 
&& IsHighlighted(line
) ) 
2566 void wxListMainWindow::Freeze() 
2571 void wxListMainWindow::Thaw() 
2573     wxCHECK_RET( m_freezeCount 
> 0, _T("thawing unfrozen list control?") ); 
2575     if ( !--m_freezeCount 
) 
2581 void wxListMainWindow::OnPaint( wxPaintEvent 
&WXUNUSED(event
) ) 
2583     // Note: a wxPaintDC must be constructed even if no drawing is 
2584     // done (a Windows requirement). 
2585     wxPaintDC 
dc( this ); 
2587     if ( IsEmpty() || m_freezeCount 
) 
2589         // nothing to draw or not the moment to draw it 
2595         // delay the repainting until we calculate all the items positions 
2602     CalcScrolledPosition( 0, 0, &dev_x
, &dev_y 
); 
2606     dc
.SetFont( GetFont() ); 
2608     if ( HasFlag(wxLC_REPORT
) ) 
2610         int lineHeight 
= GetLineHeight(); 
2612         size_t visibleFrom
, visibleTo
; 
2613         GetVisibleLinesRange(&visibleFrom
, &visibleTo
); 
2616         wxCoord xOrig
, yOrig
; 
2617         CalcUnscrolledPosition(0, 0, &xOrig
, &yOrig
); 
2619         // tell the caller cache to cache the data 
2622             wxListEvent 
evCache(wxEVT_COMMAND_LIST_CACHE_HINT
, 
2623                                 GetParent()->GetId()); 
2624             evCache
.SetEventObject( GetParent() ); 
2625             evCache
.m_oldItemIndex 
= visibleFrom
; 
2626             evCache
.m_itemIndex 
= visibleTo
; 
2627             GetParent()->GetEventHandler()->ProcessEvent( evCache 
); 
2630         for ( size_t line 
= visibleFrom
; line 
<= visibleTo
; line
++ ) 
2632             rectLine 
= GetLineRect(line
); 
2634             if ( !IsExposed(rectLine
.x 
- xOrig
, rectLine
.y 
- yOrig
, 
2635                             rectLine
.width
, rectLine
.height
) ) 
2637                 // don't redraw unaffected lines to avoid flicker 
2641             GetLine(line
)->DrawInReportMode( &dc
, 
2643                                              GetLineHighlightRect(line
), 
2644                                              IsHighlighted(line
) ); 
2647         if ( HasFlag(wxLC_HRULES
) ) 
2649             wxPen 
pen(GetRuleColour(), 1, wxSOLID
); 
2650             wxSize clientSize 
= GetClientSize(); 
2652             // Don't draw the first one 
2653             for ( size_t i 
= visibleFrom
+1; i 
<= visibleTo
; i
++ ) 
2656                 dc
.SetBrush( *wxTRANSPARENT_BRUSH 
); 
2657                 dc
.DrawLine(0 - dev_x
, i
*lineHeight
, 
2658                             clientSize
.x 
- dev_x
, i
*lineHeight
); 
2661             // Draw last horizontal rule 
2662             if ( visibleTo 
== GetItemCount() - 1 ) 
2665                 dc
.SetBrush( *wxTRANSPARENT_BRUSH 
); 
2666                 dc
.DrawLine(0 - dev_x
, (m_lineTo
+1)*lineHeight
, 
2667                             clientSize
.x 
- dev_x 
, (m_lineTo
+1)*lineHeight 
); 
2671         // Draw vertical rules if required 
2672         if ( HasFlag(wxLC_VRULES
) && !IsEmpty() ) 
2674             wxPen 
pen(GetRuleColour(), 1, wxSOLID
); 
2676             wxRect firstItemRect
; 
2677             wxRect lastItemRect
; 
2678             GetItemRect(visibleFrom
, firstItemRect
); 
2679             GetItemRect(visibleTo
, lastItemRect
); 
2680             int x 
= firstItemRect
.GetX(); 
2682             dc
.SetBrush(* wxTRANSPARENT_BRUSH
); 
2683             for (int col 
= 0; col 
< GetColumnCount(); col
++) 
2685                 int colWidth 
= GetColumnWidth(col
); 
2687                 dc
.DrawLine(x 
- dev_x 
- 2, firstItemRect
.GetY() - 1 - dev_y
, 
2688                             x 
- dev_x 
- 2, lastItemRect
.GetBottom() + 1 - dev_y
); 
2694         size_t count 
= GetItemCount(); 
2695         for ( size_t i 
= 0; i 
< count
; i
++ ) 
2697             GetLine(i
)->Draw( &dc 
); 
2703         // don't draw rect outline under Max if we already have the background 
2704         // color but under other platforms only draw it if we do: it is a bit 
2705         // silly to draw "focus rect" if we don't have focus! 
2710 #endif // __WXMAC__/!__WXMAC__ 
2712             dc
.SetPen( *wxBLACK_PEN 
); 
2713             dc
.SetBrush( *wxTRANSPARENT_BRUSH 
); 
2714             dc
.DrawRectangle( GetLineHighlightRect(m_current
) ); 
2721 void wxListMainWindow::HighlightAll( bool on 
) 
2723     if ( IsSingleSel() ) 
2725         wxASSERT_MSG( !on
, _T("can't do this in a single sel control") ); 
2727         // we just have one item to turn off 
2728         if ( HasCurrent() && IsHighlighted(m_current
) ) 
2730             HighlightLine(m_current
, FALSE
); 
2731             RefreshLine(m_current
); 
2736         HighlightLines(0, GetItemCount() - 1, on
); 
2740 void wxListMainWindow::SendNotify( size_t line
, 
2741                                    wxEventType command
, 
2744     wxListEvent 
le( command
, GetParent()->GetId() ); 
2745     le
.SetEventObject( GetParent() ); 
2746     le
.m_itemIndex 
= line
; 
2748     // set only for events which have position 
2749     if ( point 
!= wxDefaultPosition 
) 
2750         le
.m_pointDrag 
= point
; 
2752     // don't try to get the line info for virtual list controls: the main 
2753     // program has it anyhow and if we did it would result in accessing all 
2754     // the lines, even those which are not visible now and this is precisely 
2755     // what we're trying to avoid 
2756     if ( !IsVirtual() && (command 
!= wxEVT_COMMAND_LIST_DELETE_ITEM
) ) 
2758         if ( line 
!= (size_t)-1 ) 
2760             GetLine(line
)->GetItem( 0, le
.m_item 
); 
2762         //else: this happens for wxEVT_COMMAND_LIST_ITEM_FOCUSED event 
2764     //else: there may be no more such item 
2766     GetParent()->GetEventHandler()->ProcessEvent( le 
); 
2769 void wxListMainWindow::ChangeCurrent(size_t current
) 
2771     m_current 
= current
; 
2773     SendNotify(current
, wxEVT_COMMAND_LIST_ITEM_FOCUSED
); 
2776 void wxListMainWindow::EditLabel( long item 
) 
2778     wxCHECK_RET( (item 
>= 0) && ((size_t)item 
< GetItemCount()), 
2779                  wxT("wrong index in wxGenericListCtrl::EditLabel()") ); 
2781     size_t itemEdit 
= (size_t)item
; 
2783     wxListEvent 
le( wxEVT_COMMAND_LIST_BEGIN_LABEL_EDIT
, GetParent()->GetId() ); 
2784     le
.SetEventObject( GetParent() ); 
2785     le
.m_itemIndex 
= item
; 
2786     wxListLineData 
*data 
= GetLine(itemEdit
); 
2787     wxCHECK_RET( data
, _T("invalid index in EditLabel()") ); 
2788     data
->GetItem( 0, le
.m_item 
); 
2789     if ( GetParent()->GetEventHandler()->ProcessEvent( le 
) && !le
.IsAllowed() ) 
2791         // vetoed by user code 
2795     // We have to call this here because the label in question might just have 
2796     // been added and no screen update taken place. 
2800     wxListTextCtrl 
*text 
= new wxListTextCtrl(this, itemEdit
); 
2805 void wxListMainWindow::OnRenameTimer() 
2807     wxCHECK_RET( HasCurrent(), wxT("unexpected rename timer") ); 
2809     EditLabel( m_current 
); 
2812 bool wxListMainWindow::OnRenameAccept(size_t itemEdit
, const wxString
& value
) 
2814     wxListEvent 
le( wxEVT_COMMAND_LIST_END_LABEL_EDIT
, GetParent()->GetId() ); 
2815     le
.SetEventObject( GetParent() ); 
2816     le
.m_itemIndex 
= itemEdit
; 
2818     wxListLineData 
*data 
= GetLine(itemEdit
); 
2819     wxCHECK_MSG( data
, FALSE
, _T("invalid index in OnRenameAccept()") ); 
2821     data
->GetItem( 0, le
.m_item 
); 
2822     le
.m_item
.m_text 
= value
; 
2823     return !GetParent()->GetEventHandler()->ProcessEvent( le 
) || 
2827 void wxListMainWindow::OnRenameCancelled(size_t itemEdit
) 
2829     // let owner know that the edit was cancelled 
2830     wxListEvent 
le( wxEVT_COMMAND_LIST_END_LABEL_EDIT
, GetParent()->GetId() ); 
2832     le
.SetEditCanceled(TRUE
); 
2834     le
.SetEventObject( GetParent() ); 
2835     le
.m_itemIndex 
= itemEdit
; 
2837     wxListLineData 
*data 
= GetLine(itemEdit
); 
2838     wxCHECK_RET( data
, _T("invalid index in OnRenameCancelled()") ); 
2840     data
->GetItem( 0, le
.m_item 
); 
2842     GetEventHandler()->ProcessEvent( le 
); 
2845 void wxListMainWindow::OnMouse( wxMouseEvent 
&event 
) 
2847     event
.SetEventObject( GetParent() ); 
2848     if ( GetParent()->GetEventHandler()->ProcessEvent( event
) ) 
2851     if ( !HasCurrent() || IsEmpty() ) 
2857     if ( !(event
.Dragging() || event
.ButtonDown() || event
.LeftUp() || 
2858         event
.ButtonDClick()) ) 
2861     int x 
= event
.GetX(); 
2862     int y 
= event
.GetY(); 
2863     CalcUnscrolledPosition( x
, y
, &x
, &y 
); 
2865     // where did we hit it (if we did)? 
2868     size_t count 
= GetItemCount(), 
2871     if ( HasFlag(wxLC_REPORT
) ) 
2873         current 
= y 
/ GetLineHeight(); 
2874         if ( current 
< count 
) 
2875             hitResult 
= HitTestLine(current
, x
, y
); 
2879         // TODO: optimize it too! this is less simple than for report view but 
2880         //       enumerating all items is still not a way to do it!! 
2881         for ( current 
= 0; current 
< count
; current
++ ) 
2883             hitResult 
= HitTestLine(current
, x
, y
); 
2889     if (event
.Dragging()) 
2891         if (m_dragCount 
== 0) 
2893             // we have to report the raw, physical coords as we want to be 
2894             // able to call HitTest(event.m_pointDrag) from the user code to 
2895             // get the item being dragged 
2896             m_dragStart 
= event
.GetPosition(); 
2901         if (m_dragCount 
!= 3) 
2904         int command 
= event
.RightIsDown() ? wxEVT_COMMAND_LIST_BEGIN_RDRAG
 
2905                                           : wxEVT_COMMAND_LIST_BEGIN_DRAG
; 
2907         wxListEvent 
le( command
, GetParent()->GetId() ); 
2908         le
.SetEventObject( GetParent() ); 
2909         le
.m_itemIndex 
= current
; 
2910         le
.m_pointDrag 
= m_dragStart
; 
2911         GetParent()->GetEventHandler()->ProcessEvent( le 
); 
2922         // outside of any item 
2926     bool forceClick 
= FALSE
; 
2927     if (event
.ButtonDClick()) 
2929         m_renameTimer
->Stop(); 
2930         m_lastOnSame 
= FALSE
; 
2932         if ( current 
== m_lineLastClicked 
) 
2934             SendNotify( current
, wxEVT_COMMAND_LIST_ITEM_ACTIVATED 
); 
2940             // the first click was on another item, so don't interpret this as 
2941             // a double click, but as a simple click instead 
2946     if (event
.LeftUp() && m_lastOnSame
) 
2948         if ((current 
== m_current
) && 
2949             (hitResult 
== wxLIST_HITTEST_ONITEMLABEL
) && 
2950             HasFlag(wxLC_EDIT_LABELS
)  ) 
2952             m_renameTimer
->Start( 100, TRUE 
); 
2954         m_lastOnSame 
= FALSE
; 
2956     else if (event
.RightDown()) 
2958         SendNotify( current
, wxEVT_COMMAND_LIST_ITEM_RIGHT_CLICK
, 
2959                     event
.GetPosition() ); 
2961     else if (event
.MiddleDown()) 
2963         SendNotify( current
, wxEVT_COMMAND_LIST_ITEM_MIDDLE_CLICK 
); 
2965     else if ( event
.LeftDown() || forceClick 
) 
2967         m_lineBeforeLastClicked 
= m_lineLastClicked
; 
2968         m_lineLastClicked 
= current
; 
2970         size_t oldCurrent 
= m_current
; 
2972         if ( IsSingleSel() || !(event
.ControlDown() || event
.ShiftDown()) ) 
2974             HighlightAll( FALSE 
); 
2976             ChangeCurrent(current
); 
2978             ReverseHighlight(m_current
); 
2980         else // multi sel & either ctrl or shift is down 
2982             if (event
.ControlDown()) 
2984                 ChangeCurrent(current
); 
2986                 ReverseHighlight(m_current
); 
2988             else if (event
.ShiftDown()) 
2990                 ChangeCurrent(current
); 
2992                 size_t lineFrom 
= oldCurrent
, 
2995                 if ( lineTo 
< lineFrom 
) 
2998                     lineFrom 
= m_current
; 
3001                 HighlightLines(lineFrom
, lineTo
); 
3003             else // !ctrl, !shift 
3005                 // test in the enclosing if should make it impossible 
3006                 wxFAIL_MSG( _T("how did we get here?") ); 
3010         if (m_current 
!= oldCurrent
) 
3012             RefreshLine( oldCurrent 
); 
3015         // forceClick is only set if the previous click was on another item 
3016         m_lastOnSame 
= !forceClick 
&& (m_current 
== oldCurrent
); 
3020 void wxListMainWindow::MoveToItem(size_t item
) 
3022     if ( item 
== (size_t)-1 ) 
3025     wxRect rect 
= GetLineRect(item
); 
3027     int client_w
, client_h
; 
3028     GetClientSize( &client_w
, &client_h 
); 
3030     const int hLine 
= GetLineHeight(); 
3032     int view_x 
= SCROLL_UNIT_X
*GetScrollPos( wxHORIZONTAL 
); 
3033     int view_y 
= hLine
*GetScrollPos( wxVERTICAL 
); 
3035     if ( HasFlag(wxLC_REPORT
) ) 
3037         // the next we need the range of lines shown it might be different, so 
3039         ResetVisibleLinesRange(); 
3041         if (rect
.y 
< view_y 
) 
3042             Scroll( -1, rect
.y
/hLine 
); 
3043         if (rect
.y
+rect
.height
+5 > view_y
+client_h
) 
3044             Scroll( -1, (rect
.y
+rect
.height
-client_h
+hLine
)/hLine 
); 
3048         if (rect
.x
-view_x 
< 5) 
3049             Scroll( (rect
.x
-5)/SCROLL_UNIT_X
, -1 ); 
3050         if (rect
.x
+rect
.width
-5 > view_x
+client_w
) 
3051             Scroll( (rect
.x
+rect
.width
-client_w
+SCROLL_UNIT_X
)/SCROLL_UNIT_X
, -1 ); 
3055 // ---------------------------------------------------------------------------- 
3056 // keyboard handling 
3057 // ---------------------------------------------------------------------------- 
3059 void wxListMainWindow::OnArrowChar(size_t newCurrent
, const wxKeyEvent
& event
) 
3061     wxCHECK_RET( newCurrent 
< (size_t)GetItemCount(), 
3062                  _T("invalid item index in OnArrowChar()") ); 
3064     size_t oldCurrent 
= m_current
; 
3066     // in single selection we just ignore Shift as we can't select several 
3068     if ( event
.ShiftDown() && !IsSingleSel() ) 
3070         ChangeCurrent(newCurrent
); 
3072         // refresh the old focus to remove it 
3073         RefreshLine( oldCurrent 
); 
3075         // select all the items between the old and the new one 
3076         if ( oldCurrent 
> newCurrent 
) 
3078             newCurrent 
= oldCurrent
; 
3079             oldCurrent 
= m_current
; 
3082         HighlightLines(oldCurrent
, newCurrent
); 
3086         // all previously selected items are unselected unless ctrl is held 
3087         if ( !event
.ControlDown() ) 
3088             HighlightAll(FALSE
); 
3090         ChangeCurrent(newCurrent
); 
3092         // refresh the old focus to remove it 
3093         RefreshLine( oldCurrent 
); 
3095         if ( !event
.ControlDown() ) 
3097             HighlightLine( m_current
, TRUE 
); 
3102     RefreshLine( m_current 
); 
3107 void wxListMainWindow::OnKeyDown( wxKeyEvent 
&event 
) 
3109     wxWindow 
*parent 
= GetParent(); 
3111     /* we propagate the key event up */ 
3112     wxKeyEvent 
ke( wxEVT_KEY_DOWN 
); 
3113     ke
.m_shiftDown 
= event
.m_shiftDown
; 
3114     ke
.m_controlDown 
= event
.m_controlDown
; 
3115     ke
.m_altDown 
= event
.m_altDown
; 
3116     ke
.m_metaDown 
= event
.m_metaDown
; 
3117     ke
.m_keyCode 
= event
.m_keyCode
; 
3120     ke
.SetEventObject( parent 
); 
3121     if (parent
->GetEventHandler()->ProcessEvent( ke 
)) return; 
3126 void wxListMainWindow::OnChar( wxKeyEvent 
&event 
) 
3128     wxWindow 
*parent 
= GetParent(); 
3130     /* we send a list_key event up */ 
3133         wxListEvent 
le( wxEVT_COMMAND_LIST_KEY_DOWN
, GetParent()->GetId() ); 
3134         le
.m_itemIndex 
= m_current
; 
3135         GetLine(m_current
)->GetItem( 0, le
.m_item 
); 
3136         le
.m_code 
= event
.GetKeyCode(); 
3137         le
.SetEventObject( parent 
); 
3138         parent
->GetEventHandler()->ProcessEvent( le 
); 
3141     /* we propagate the char event up */ 
3142     wxKeyEvent 
ke( wxEVT_CHAR 
); 
3143     ke
.m_shiftDown 
= event
.m_shiftDown
; 
3144     ke
.m_controlDown 
= event
.m_controlDown
; 
3145     ke
.m_altDown 
= event
.m_altDown
; 
3146     ke
.m_metaDown 
= event
.m_metaDown
; 
3147     ke
.m_keyCode 
= event
.m_keyCode
; 
3150     ke
.SetEventObject( parent 
); 
3151     if (parent
->GetEventHandler()->ProcessEvent( ke 
)) return; 
3153     if (event
.GetKeyCode() == WXK_TAB
) 
3155         wxNavigationKeyEvent nevent
; 
3156         nevent
.SetWindowChange( event
.ControlDown() ); 
3157         nevent
.SetDirection( !event
.ShiftDown() ); 
3158         nevent
.SetEventObject( GetParent()->GetParent() ); 
3159         nevent
.SetCurrentFocus( m_parent 
); 
3160         if (GetParent()->GetParent()->GetEventHandler()->ProcessEvent( nevent 
)) 
3164     /* no item -> nothing to do */ 
3171     switch (event
.GetKeyCode()) 
3174             if ( m_current 
> 0 ) 
3175                 OnArrowChar( m_current 
- 1, event 
); 
3179             if ( m_current 
< (size_t)GetItemCount() - 1 ) 
3180                 OnArrowChar( m_current 
+ 1, event 
); 
3185                 OnArrowChar( GetItemCount() - 1, event 
); 
3190                 OnArrowChar( 0, event 
); 
3195                 int steps 
= HasFlag(wxLC_REPORT
) ? m_linesPerPage 
- 1 : m_current 
% m_linesPerPage
; 
3197                 int index 
= m_current 
- steps
; 
3201                 OnArrowChar( index
, event 
); 
3207                 int steps 
= HasFlag(wxLC_REPORT
) 
3208                                ? m_linesPerPage 
- 1 
3209                                : m_linesPerPage 
- (m_current 
% m_linesPerPage
) - 1; 
3211                 size_t index 
= m_current 
+ steps
; 
3212                 size_t count 
= GetItemCount(); 
3213                 if ( index 
>= count 
) 
3216                 OnArrowChar( index
, event 
); 
3221             if ( !HasFlag(wxLC_REPORT
) ) 
3223                 int index 
= m_current 
- m_linesPerPage
; 
3227                 OnArrowChar( index
, event 
); 
3232             if ( !HasFlag(wxLC_REPORT
) ) 
3234                 size_t index 
= m_current 
+ m_linesPerPage
; 
3236                 size_t count 
= GetItemCount(); 
3237                 if ( index 
>= count 
) 
3240                 OnArrowChar( index
, event 
); 
3245             if ( IsSingleSel() ) 
3247                 SendNotify( m_current
, wxEVT_COMMAND_LIST_ITEM_ACTIVATED 
); 
3249                 if ( IsHighlighted(m_current
) ) 
3251                     // don't unselect the item in single selection mode 
3254                 //else: select it in ReverseHighlight() below if unselected 
3257             ReverseHighlight(m_current
); 
3262             SendNotify( m_current
, wxEVT_COMMAND_LIST_ITEM_ACTIVATED 
); 
3270 // ---------------------------------------------------------------------------- 
3272 // ---------------------------------------------------------------------------- 
3274 void wxListMainWindow::SetFocus() 
3276     // VS: wxListMainWindow derives from wxPanel (via wxScrolledWindow) and wxPanel 
3277     //     overrides SetFocus in such way that it does never change focus from 
3278     //     panel's child to the panel itself. Unfortunately, we must be able to change 
3279     //     focus to the panel from wxListTextCtrl because the text control should 
3280     //     disappear when the user clicks outside it. 
3282     wxWindow 
*oldFocus 
= FindFocus(); 
3284     if ( oldFocus 
&& oldFocus
->GetParent() == this ) 
3286         wxWindow::SetFocus(); 
3290         wxScrolledWindow::SetFocus(); 
3294 void wxListMainWindow::OnSetFocus( wxFocusEvent 
&WXUNUSED(event
) ) 
3298         wxFocusEvent 
event( wxEVT_SET_FOCUS
, GetParent()->GetId() ); 
3299         event
.SetEventObject( GetParent() ); 
3300         if ( GetParent()->GetEventHandler()->ProcessEvent( event
) ) 
3304     // wxGTK sends us EVT_SET_FOCUS events even if we had never got 
3305     // EVT_KILL_FOCUS before which means that we finish by redrawing the items 
3306     // which are already drawn correctly resulting in horrible flicker - avoid 
3316 void wxListMainWindow::OnKillFocus( wxFocusEvent 
&WXUNUSED(event
) ) 
3320         wxFocusEvent 
event( wxEVT_KILL_FOCUS
, GetParent()->GetId() ); 
3321         event
.SetEventObject( GetParent() ); 
3322         if ( GetParent()->GetEventHandler()->ProcessEvent( event
) ) 
3329 void wxListMainWindow::DrawImage( int index
, wxDC 
*dc
, int x
, int y 
) 
3331     if ( HasFlag(wxLC_ICON
) && (m_normal_image_list
)) 
3333         m_normal_image_list
->Draw( index
, *dc
, x
, y
, wxIMAGELIST_DRAW_TRANSPARENT 
); 
3335     else if ( HasFlag(wxLC_SMALL_ICON
) && (m_small_image_list
)) 
3337         m_small_image_list
->Draw( index
, *dc
, x
, y
, wxIMAGELIST_DRAW_TRANSPARENT 
); 
3339     else if ( HasFlag(wxLC_LIST
) && (m_small_image_list
)) 
3341         m_small_image_list
->Draw( index
, *dc
, x
, y
, wxIMAGELIST_DRAW_TRANSPARENT 
); 
3343     else if ( HasFlag(wxLC_REPORT
) && (m_small_image_list
)) 
3345         m_small_image_list
->Draw( index
, *dc
, x
, y
, wxIMAGELIST_DRAW_TRANSPARENT 
); 
3349 void wxListMainWindow::GetImageSize( int index
, int &width
, int &height 
) const 
3351     if ( HasFlag(wxLC_ICON
) && m_normal_image_list 
) 
3353         m_normal_image_list
->GetSize( index
, width
, height 
); 
3355     else if ( HasFlag(wxLC_SMALL_ICON
) && m_small_image_list 
) 
3357         m_small_image_list
->GetSize( index
, width
, height 
); 
3359     else if ( HasFlag(wxLC_LIST
) && m_small_image_list 
) 
3361         m_small_image_list
->GetSize( index
, width
, height 
); 
3363     else if ( HasFlag(wxLC_REPORT
) && m_small_image_list 
) 
3365         m_small_image_list
->GetSize( index
, width
, height 
); 
3374 int wxListMainWindow::GetTextLength( const wxString 
&s 
) const 
3376     wxClientDC 
dc( wxConstCast(this, wxListMainWindow
) ); 
3377     dc
.SetFont( GetFont() ); 
3380     dc
.GetTextExtent( s
, &lw
, NULL 
); 
3382     return lw 
+ AUTOSIZE_COL_MARGIN
; 
3385 void wxListMainWindow::SetImageList( wxImageListType 
*imageList
, int which 
) 
3389     // calc the spacing from the icon size 
3392     if ((imageList
) && (imageList
->GetImageCount()) ) 
3394         imageList
->GetSize(0, width
, height
); 
3397     if (which 
== wxIMAGE_LIST_NORMAL
) 
3399         m_normal_image_list 
= imageList
; 
3400         m_normal_spacing 
= width 
+ 8; 
3403     if (which 
== wxIMAGE_LIST_SMALL
) 
3405         m_small_image_list 
= imageList
; 
3406         m_small_spacing 
= width 
+ 14; 
3407         m_lineHeight 
= 0;  // ensure that the line height will be recalc'd 
3411 void wxListMainWindow::SetItemSpacing( int spacing
, bool isSmall 
) 
3416         m_small_spacing 
= spacing
; 
3420         m_normal_spacing 
= spacing
; 
3424 int wxListMainWindow::GetItemSpacing( bool isSmall 
) 
3426     return isSmall 
? m_small_spacing 
: m_normal_spacing
; 
3429 // ---------------------------------------------------------------------------- 
3431 // ---------------------------------------------------------------------------- 
3433 void wxListMainWindow::SetColumn( int col
, wxListItem 
&item 
) 
3435     wxListHeaderDataList::compatibility_iterator node 
= m_columns
.Item( col 
); 
3437     wxCHECK_RET( node
, _T("invalid column index in SetColumn") ); 
3439     if ( item
.m_width 
== wxLIST_AUTOSIZE_USEHEADER 
) 
3440         item
.m_width 
= GetTextLength( item
.m_text 
); 
3442     wxListHeaderData 
*column 
= node
->GetData(); 
3443     column
->SetItem( item 
); 
3445     wxListHeaderWindow 
*headerWin 
= GetListCtrl()->m_headerWin
; 
3447         headerWin
->m_dirty 
= TRUE
; 
3451     // invalidate it as it has to be recalculated 
3455 void wxListMainWindow::SetColumnWidth( int col
, int width 
) 
3457     wxCHECK_RET( col 
>= 0 && col 
< GetColumnCount(), 
3458                  _T("invalid column index") ); 
3460     wxCHECK_RET( HasFlag(wxLC_REPORT
), 
3461                  _T("SetColumnWidth() can only be called in report mode.") ); 
3464     wxListHeaderWindow 
*headerWin 
= GetListCtrl()->m_headerWin
; 
3466         headerWin
->m_dirty 
= TRUE
; 
3468     wxListHeaderDataList::compatibility_iterator node 
= m_columns
.Item( col 
); 
3469     wxCHECK_RET( node
, _T("no column?") ); 
3471     wxListHeaderData 
*column 
= node
->GetData(); 
3473     size_t count 
= GetItemCount(); 
3475     if (width 
== wxLIST_AUTOSIZE_USEHEADER
) 
3477         width 
= GetTextLength(column
->GetText()); 
3479     else if ( width 
== wxLIST_AUTOSIZE 
) 
3483             // TODO: determine the max width somehow... 
3484             width 
= WIDTH_COL_DEFAULT
; 
3488             wxClientDC 
dc(this); 
3489             dc
.SetFont( GetFont() ); 
3491             int max 
= AUTOSIZE_COL_MARGIN
; 
3493             for ( size_t i 
= 0; i 
< count
; i
++ ) 
3495                 wxListLineData 
*line 
= GetLine(i
); 
3496                 wxListItemDataList::compatibility_iterator n 
= line
->m_items
.Item( col 
); 
3498                 wxCHECK_RET( n
, _T("no subitem?") ); 
3500                 wxListItemData 
*item 
= n
->GetData(); 
3503                 if (item
->HasImage()) 
3506                     GetImageSize( item
->GetImage(), ix
, iy 
); 
3510                 if (item
->HasText()) 
3513                     dc
.GetTextExtent( item
->GetText(), &w
, NULL 
); 
3521             width 
= max 
+ AUTOSIZE_COL_MARGIN
; 
3525     column
->SetWidth( width 
); 
3527     // invalidate it as it has to be recalculated 
3531 int wxListMainWindow::GetHeaderWidth() const 
3533     if ( !m_headerWidth 
) 
3535         wxListMainWindow 
*self 
= wxConstCast(this, wxListMainWindow
); 
3537         size_t count 
= GetColumnCount(); 
3538         for ( size_t col 
= 0; col 
< count
; col
++ ) 
3540             self
->m_headerWidth 
+= GetColumnWidth(col
); 
3544     return m_headerWidth
; 
3547 void wxListMainWindow::GetColumn( int col
, wxListItem 
&item 
) const 
3549     wxListHeaderDataList::compatibility_iterator node 
= m_columns
.Item( col 
); 
3550     wxCHECK_RET( node
, _T("invalid column index in GetColumn") ); 
3552     wxListHeaderData 
*column 
= node
->GetData(); 
3553     column
->GetItem( item 
); 
3556 int wxListMainWindow::GetColumnWidth( int col 
) const 
3558     wxListHeaderDataList::compatibility_iterator node 
= m_columns
.Item( col 
); 
3559     wxCHECK_MSG( node
, 0, _T("invalid column index") ); 
3561     wxListHeaderData 
*column 
= node
->GetData(); 
3562     return column
->GetWidth(); 
3565 // ---------------------------------------------------------------------------- 
3567 // ---------------------------------------------------------------------------- 
3569 void wxListMainWindow::SetItem( wxListItem 
&item 
) 
3571     long id 
= item
.m_itemId
; 
3572     wxCHECK_RET( id 
>= 0 && (size_t)id 
< GetItemCount(), 
3573                  _T("invalid item index in SetItem") ); 
3577         wxListLineData 
*line 
= GetLine((size_t)id
); 
3578         line
->SetItem( item
.m_col
, item 
); 
3581     // update the item on screen 
3583     GetItemRect(id
, rectItem
); 
3584     RefreshRect(rectItem
); 
3587 void wxListMainWindow::SetItemState( long litem
, long state
, long stateMask 
) 
3589      wxCHECK_RET( litem 
>= 0 && (size_t)litem 
< GetItemCount(), 
3590                   _T("invalid list ctrl item index in SetItem") ); 
3592     size_t oldCurrent 
= m_current
; 
3593     size_t item 
= (size_t)litem
;    // safe because of the check above 
3595     // do we need to change the focus? 
3596     if ( stateMask 
& wxLIST_STATE_FOCUSED 
) 
3598         if ( state 
& wxLIST_STATE_FOCUSED 
) 
3600             // don't do anything if this item is already focused 
3601             if ( item 
!= m_current 
) 
3603                 ChangeCurrent(item
); 
3605                 if ( oldCurrent 
!= (size_t)-1 ) 
3607                     if ( IsSingleSel() ) 
3609                         HighlightLine(oldCurrent
, FALSE
); 
3612                     RefreshLine(oldCurrent
); 
3615                 RefreshLine( m_current 
); 
3620             // don't do anything if this item is not focused 
3621             if ( item 
== m_current 
) 
3625                 if ( IsSingleSel() ) 
3627                     // we must unselect the old current item as well or we 
3628                     // might end up with more than one selected item in a 
3629                     // single selection control 
3630                     HighlightLine(oldCurrent
, FALSE
); 
3633                 RefreshLine( oldCurrent 
); 
3638     // do we need to change the selection state? 
3639     if ( stateMask 
& wxLIST_STATE_SELECTED 
) 
3641         bool on 
= (state 
& wxLIST_STATE_SELECTED
) != 0; 
3643         if ( IsSingleSel() ) 
3647                 // selecting the item also makes it the focused one in the 
3649                 if ( m_current 
!= item 
) 
3651                     ChangeCurrent(item
); 
3653                     if ( oldCurrent 
!= (size_t)-1 ) 
3655                         HighlightLine( oldCurrent
, FALSE 
); 
3656                         RefreshLine( oldCurrent 
); 
3662                 // only the current item may be selected anyhow 
3663                 if ( item 
!= m_current 
) 
3668         if ( HighlightLine(item
, on
) ) 
3675 int wxListMainWindow::GetItemState( long item
, long stateMask 
) const 
3677     wxCHECK_MSG( item 
>= 0 && (size_t)item 
< GetItemCount(), 0, 
3678                  _T("invalid list ctrl item index in GetItemState()") ); 
3680     int ret 
= wxLIST_STATE_DONTCARE
; 
3682     if ( stateMask 
& wxLIST_STATE_FOCUSED 
) 
3684         if ( (size_t)item 
== m_current 
) 
3685             ret 
|= wxLIST_STATE_FOCUSED
; 
3688     if ( stateMask 
& wxLIST_STATE_SELECTED 
) 
3690         if ( IsHighlighted(item
) ) 
3691             ret 
|= wxLIST_STATE_SELECTED
; 
3697 void wxListMainWindow::GetItem( wxListItem 
&item 
) const 
3699     wxCHECK_RET( item
.m_itemId 
>= 0 && (size_t)item
.m_itemId 
< GetItemCount(), 
3700                  _T("invalid item index in GetItem") ); 
3702     wxListLineData 
*line 
= GetLine((size_t)item
.m_itemId
); 
3703     line
->GetItem( item
.m_col
, item 
); 
3706 // ---------------------------------------------------------------------------- 
3708 // ---------------------------------------------------------------------------- 
3710 size_t wxListMainWindow::GetItemCount() const 
3712     return IsVirtual() ? m_countVirt 
: m_lines
.GetCount(); 
3715 void wxListMainWindow::SetItemCount(long count
) 
3717     m_selStore
.SetItemCount(count
); 
3718     m_countVirt 
= count
; 
3720     ResetVisibleLinesRange(); 
3722     // scrollbars must be reset 
3726 int wxListMainWindow::GetSelectedItemCount() const 
3728     // deal with the quick case first 
3729     if ( IsSingleSel() ) 
3731         return HasCurrent() ? IsHighlighted(m_current
) : FALSE
; 
3734     // virtual controls remmebers all its selections itself 
3736         return m_selStore
.GetSelectedCount(); 
3738     // TODO: we probably should maintain the number of items selected even for 
3739     //       non virtual controls as enumerating all lines is really slow... 
3740     size_t countSel 
= 0; 
3741     size_t count 
= GetItemCount(); 
3742     for ( size_t line 
= 0; line 
< count
; line
++ ) 
3744         if ( GetLine(line
)->IsHighlighted() ) 
3751 // ---------------------------------------------------------------------------- 
3752 // item position/size 
3753 // ---------------------------------------------------------------------------- 
3755 wxRect 
wxListMainWindow::GetViewRect() const 
3757     wxASSERT_MSG( !HasFlag(wxLC_REPORT 
| wxLC_LIST
), 
3758                     _T("wxListCtrl::GetViewRect() only works in icon mode") ); 
3760     // we need to find the longest/tallest label 
3763     const int count 
= GetItemCount(); 
3766         for ( int i 
= 0; i 
< count
; i
++ ) 
3771             wxCoord x 
= r
.GetRight(), 
3781     // some fudge needed to make it look prettier 
3782     xMax 
+= 2*EXTRA_BORDER_X
; 
3783     yMax 
+= 2*EXTRA_BORDER_Y
; 
3785     // account for the scrollbars if necessary 
3786     const wxSize sizeAll 
= GetClientSize(); 
3787     if ( xMax 
> sizeAll
.x 
) 
3788         yMax 
+= wxSystemSettings::GetMetric(wxSYS_HSCROLL_Y
); 
3789     if ( yMax 
> sizeAll
.y 
) 
3790         xMax 
+= wxSystemSettings::GetMetric(wxSYS_VSCROLL_X
); 
3792     return wxRect(0, 0, xMax
, yMax
); 
3795 void wxListMainWindow::GetItemRect( long index
, wxRect 
&rect 
) const 
3797     wxCHECK_RET( index 
>= 0 && (size_t)index 
< GetItemCount(), 
3798                  _T("invalid index in GetItemRect") ); 
3800     // ensure that we're laid out, otherwise we could return nonsense 
3803         wxConstCast(this, wxListMainWindow
)-> 
3804             RecalculatePositions(TRUE 
/* no refresh */); 
3807     rect 
= GetLineRect((size_t)index
); 
3809     CalcScrolledPosition(rect
.x
, rect
.y
, &rect
.x
, &rect
.y
); 
3812 bool wxListMainWindow::GetItemPosition(long item
, wxPoint
& pos
) const 
3815     GetItemRect(item
, rect
); 
3823 // ---------------------------------------------------------------------------- 
3824 // geometry calculation 
3825 // ---------------------------------------------------------------------------- 
3827 void wxListMainWindow::RecalculatePositions(bool noRefresh
) 
3829     wxClientDC 
dc( this ); 
3830     dc
.SetFont( GetFont() ); 
3832     const size_t count 
= GetItemCount(); 
3835     if ( HasFlag(wxLC_ICON
) ) 
3836         iconSpacing 
= m_normal_spacing
; 
3837     else if ( HasFlag(wxLC_SMALL_ICON
) ) 
3838         iconSpacing 
= m_small_spacing
; 
3842     // Note that we do not call GetClientSize() here but 
3843     // GetSize() and substract the border size for sunken 
3844     // borders manually. This is technically incorrect, 
3845     // but we need to know the client area's size WITHOUT 
3846     // scrollbars here. Since we don't know if there are 
3847     // any scrollbars, we use GetSize() instead. Another 
3848     // solution would be to call SetScrollbars() here to 
3849     // remove the scrollbars and call GetClientSize() then, 
3850     // but this might result in flicker and - worse - will 
3851     // reset the scrollbars to 0 which is not good at all 
3852     // if you resize a dialog/window, but don't want to 
3853     // reset the window scrolling. RR. 
3854     // Furthermore, we actually do NOT subtract the border 
3855     // width as 2 pixels is just the extra space which we 
3856     // need around the actual content in the window. Other- 
3857     // wise the text would e.g. touch the upper border. RR. 
3860     GetSize( &clientWidth
, &clientHeight 
); 
3862     const int lineHeight 
= GetLineHeight(); 
3864     if ( HasFlag(wxLC_REPORT
) ) 
3866         // all lines have the same height and we scroll one line per step 
3867         int entireHeight 
= count
*lineHeight 
+ LINE_SPACING
; 
3869         m_linesPerPage 
= clientHeight 
/ lineHeight
; 
3871         ResetVisibleLinesRange(); 
3873         SetScrollbars( SCROLL_UNIT_X
, lineHeight
, 
3874                        GetHeaderWidth() / SCROLL_UNIT_X
, 
3875                        (entireHeight 
+ lineHeight 
- 1) / lineHeight
, 
3876                        GetScrollPos(wxHORIZONTAL
), 
3877                        GetScrollPos(wxVERTICAL
), 
3882         // we have 3 different layout strategies: either layout all items 
3883         // horizontally/vertically (wxLC_ALIGN_XXX styles explicitly given) or 
3884         // to arrange them in top to bottom, left to right (don't ask me why 
3885         // not the other way round...) order 
3886         if ( HasFlag(wxLC_ALIGN_LEFT 
| wxLC_ALIGN_TOP
) ) 
3888             int x 
= EXTRA_BORDER_X
; 
3889             int y 
= EXTRA_BORDER_Y
; 
3891             wxCoord widthMax 
= 0; 
3894             for ( i 
= 0; i 
< count
; i
++ ) 
3896                 wxListLineData 
*line 
= GetLine(i
); 
3897                 line
->CalculateSize( &dc
, iconSpacing 
); 
3898                 line
->SetPosition( x
, y
, iconSpacing 
); 
3900                 wxSize sizeLine 
= GetLineSize(i
); 
3902                 if ( HasFlag(wxLC_ALIGN_TOP
) ) 
3904                     if ( sizeLine
.x 
> widthMax 
) 
3905                         widthMax 
= sizeLine
.x
; 
3909                 else // wxLC_ALIGN_LEFT 
3911                     x 
+= sizeLine
.x 
+ MARGIN_BETWEEN_ROWS
; 
3915             if ( HasFlag(wxLC_ALIGN_TOP
) ) 
3917                 // traverse the items again and tweak their sizes so that they are 
3918                 // all the same in a row 
3919                 for ( i 
= 0; i 
< count
; i
++ ) 
3921                     wxListLineData 
*line 
= GetLine(i
); 
3922                     line
->m_gi
->ExtendWidth(widthMax
); 
3931                 (x 
+ SCROLL_UNIT_X
) / SCROLL_UNIT_X
, 
3932                 (y 
+ lineHeight
) / lineHeight
, 
3933                 GetScrollPos( wxHORIZONTAL 
), 
3934                 GetScrollPos( wxVERTICAL 
), 
3938         else // "flowed" arrangement, the most complicated case 
3940             // at first we try without any scrollbars, if the items don't fit into 
3941             // the window, we recalculate after subtracting the space taken by the 
3944             int entireWidth 
= 0; 
3946             for (int tries 
= 0; tries 
< 2; tries
++) 
3948                 entireWidth 
= 2*EXTRA_BORDER_X
; 
3952                     // Now we have decided that the items do not fit into the 
3953                     // client area, so we need a scrollbar 
3954                     entireWidth 
+= SCROLL_UNIT_X
; 
3957                 int x 
= EXTRA_BORDER_X
; 
3958                 int y 
= EXTRA_BORDER_Y
; 
3959                 int maxWidthInThisRow 
= 0; 
3962                 int currentlyVisibleLines 
= 0; 
3964                 for (size_t i 
= 0; i 
< count
; i
++) 
3966                     currentlyVisibleLines
++; 
3967                     wxListLineData 
*line 
= GetLine(i
); 
3968                     line
->CalculateSize( &dc
, iconSpacing 
); 
3969                     line
->SetPosition( x
, y
, iconSpacing 
); 
3971                     wxSize sizeLine 
= GetLineSize(i
); 
3973                     if ( maxWidthInThisRow 
< sizeLine
.x 
) 
3974                         maxWidthInThisRow 
= sizeLine
.x
; 
3977                     if (currentlyVisibleLines 
> m_linesPerPage
) 
3978                         m_linesPerPage 
= currentlyVisibleLines
; 
3980                     if ( y 
+ sizeLine
.y 
>= clientHeight 
) 
3982                         currentlyVisibleLines 
= 0; 
3984                         maxWidthInThisRow 
+= MARGIN_BETWEEN_ROWS
; 
3985                         x 
+= maxWidthInThisRow
; 
3986                         entireWidth 
+= maxWidthInThisRow
; 
3987                         maxWidthInThisRow 
= 0; 
3990                     // We have reached the last item. 
3991                     if ( i 
== count 
- 1 ) 
3992                         entireWidth 
+= maxWidthInThisRow
; 
3994                     if ( (tries 
== 0) && 
3995                             (entireWidth 
+ SCROLL_UNIT_X 
> clientWidth
) ) 
3997                         clientHeight 
-= wxSystemSettings:: 
3998                                             GetMetric(wxSYS_HSCROLL_Y
); 
4003                     if ( i 
== count 
- 1 ) 
4004                         tries 
= 1;  // Everything fits, no second try required. 
4012                 (entireWidth 
+ SCROLL_UNIT_X
) / SCROLL_UNIT_X
, 
4014                 GetScrollPos( wxHORIZONTAL 
), 
4023         // FIXME: why should we call it from here? 
4030 void wxListMainWindow::RefreshAll() 
4035     wxListHeaderWindow 
*headerWin 
= GetListCtrl()->m_headerWin
; 
4036     if ( headerWin 
&& headerWin
->m_dirty 
) 
4038         headerWin
->m_dirty 
= FALSE
; 
4039         headerWin
->Refresh(); 
4043 void wxListMainWindow::UpdateCurrent() 
4045     if ( !HasCurrent() && !IsEmpty() ) 
4051 long wxListMainWindow::GetNextItem( long item
, 
4052                                     int WXUNUSED(geometry
), 
4056          max 
= GetItemCount(); 
4057     wxCHECK_MSG( (ret 
== -1) || (ret 
< max
), -1, 
4058                  _T("invalid listctrl index in GetNextItem()") ); 
4060     // notice that we start with the next item (or the first one if item == -1) 
4061     // and this is intentional to allow writing a simple loop to iterate over 
4062     // all selected items 
4066         // this is not an error because the index was ok initially, just no 
4077     size_t count 
= GetItemCount(); 
4078     for ( size_t line 
= (size_t)ret
; line 
< count
; line
++ ) 
4080         if ( (state 
& wxLIST_STATE_FOCUSED
) && (line 
== m_current
) ) 
4083         if ( (state 
& wxLIST_STATE_SELECTED
) && IsHighlighted(line
) ) 
4090 // ---------------------------------------------------------------------------- 
4092 // ---------------------------------------------------------------------------- 
4094 void wxListMainWindow::DeleteItem( long lindex 
) 
4096     size_t count 
= GetItemCount(); 
4098     wxCHECK_RET( (lindex 
>= 0) && ((size_t)lindex 
< count
), 
4099                  _T("invalid item index in DeleteItem") ); 
4101     size_t index 
= (size_t)lindex
; 
4103     // we don't need to adjust the index for the previous items 
4104     if ( HasCurrent() && m_current 
>= index 
) 
4106         // if the current item is being deleted, we want the next one to 
4107         // become selected - unless there is no next one - so don't adjust 
4108         // m_current in this case 
4109         if ( m_current 
!= index 
|| m_current 
== count 
- 1 ) 
4115     if ( InReportView() ) 
4117         ResetVisibleLinesRange(); 
4124         m_selStore
.OnItemDelete(index
); 
4128         m_lines
.RemoveAt( index 
); 
4131     // we need to refresh the (vert) scrollbar as the number of items changed 
4134     SendNotify( index
, wxEVT_COMMAND_LIST_DELETE_ITEM 
); 
4136     RefreshAfter(index
); 
4139 void wxListMainWindow::DeleteColumn( int col 
) 
4141     wxListHeaderDataList::compatibility_iterator node 
= m_columns
.Item( col 
); 
4143     wxCHECK_RET( node
, wxT("invalid column index in DeleteColumn()") ); 
4146     delete node
->GetData(); 
4147     m_columns
.Erase( node 
); 
4151         // update all the items 
4152         for ( size_t i 
= 0; i 
< m_lines
.GetCount(); i
++ ) 
4154             wxListLineData 
* const line 
= GetLine(i
); 
4155             wxListItemDataList::compatibility_iterator n 
= line
->m_items
.Item( col 
); 
4156             delete n
->GetData(); 
4157             line
->m_items
.Erase(n
); 
4161     // invalidate it as it has to be recalculated 
4165 void wxListMainWindow::DoDeleteAllItems() 
4169         // nothing to do - in particular, don't send the event 
4175     // to make the deletion of all items faster, we don't send the 
4176     // notifications for each item deletion in this case but only one event 
4177     // for all of them: this is compatible with wxMSW and documented in 
4178     // DeleteAllItems() description 
4180     wxListEvent 
event( wxEVT_COMMAND_LIST_DELETE_ALL_ITEMS
, GetParent()->GetId() ); 
4181     event
.SetEventObject( GetParent() ); 
4182     GetParent()->GetEventHandler()->ProcessEvent( event 
); 
4191     if ( InReportView() ) 
4193         ResetVisibleLinesRange(); 
4199 void wxListMainWindow::DeleteAllItems() 
4203     RecalculatePositions(); 
4206 void wxListMainWindow::DeleteEverything() 
4208     WX_CLEAR_LIST(wxListHeaderDataList
, m_columns
); 
4213 // ---------------------------------------------------------------------------- 
4214 // scanning for an item 
4215 // ---------------------------------------------------------------------------- 
4217 void wxListMainWindow::EnsureVisible( long index 
) 
4219     wxCHECK_RET( index 
>= 0 && (size_t)index 
< GetItemCount(), 
4220                  _T("invalid index in EnsureVisible") ); 
4222     // We have to call this here because the label in question might just have 
4223     // been added and its position is not known yet 
4226         RecalculatePositions(TRUE 
/* no refresh */); 
4229     MoveToItem((size_t)index
); 
4232 long wxListMainWindow::FindItem(long start
, const wxString
& str
, bool WXUNUSED(partial
) ) 
4239     size_t count 
= GetItemCount(); 
4240     for ( size_t i 
= (size_t)pos
; i 
< count
; i
++ ) 
4242         wxListLineData 
*line 
= GetLine(i
); 
4243         if ( line
->GetText(0) == tmp 
) 
4250 long wxListMainWindow::FindItem(long start
, long data
) 
4256     size_t count 
= GetItemCount(); 
4257     for (size_t i 
= (size_t)pos
; i 
< count
; i
++) 
4259         wxListLineData 
*line 
= GetLine(i
); 
4261         line
->GetItem( 0, item 
); 
4262         if (item
.m_data 
== data
) 
4269 long wxListMainWindow::HitTest( int x
, int y
, int &flags 
) 
4271     CalcUnscrolledPosition( x
, y
, &x
, &y 
); 
4273     size_t count 
= GetItemCount(); 
4275     if ( HasFlag(wxLC_REPORT
) ) 
4277         size_t current 
= y 
/ GetLineHeight(); 
4278         if ( current 
< count 
) 
4280             flags 
= HitTestLine(current
, x
, y
); 
4287         // TODO: optimize it too! this is less simple than for report view but 
4288         //       enumerating all items is still not a way to do it!! 
4289         for ( size_t current 
= 0; current 
< count
; current
++ ) 
4291             flags 
= HitTestLine(current
, x
, y
); 
4300 // ---------------------------------------------------------------------------- 
4302 // ---------------------------------------------------------------------------- 
4304 void wxListMainWindow::InsertItem( wxListItem 
&item 
) 
4306     wxASSERT_MSG( !IsVirtual(), _T("can't be used with virtual control") ); 
4308     size_t count 
= GetItemCount(); 
4309     wxCHECK_RET( item
.m_itemId 
>= 0 && (size_t)item
.m_itemId 
<= count
, 
4310                  _T("invalid item index") ); 
4312     size_t id 
= item
.m_itemId
; 
4317     // this is unused variable 
4320     if ( HasFlag(wxLC_REPORT
) ) 
4323         // this is unused variable 
4326         ResetVisibleLinesRange(); 
4328     else if ( HasFlag(wxLC_LIST
) ) 
4330         // this is unused variable 
4335     else if ( HasFlag(wxLC_ICON
) ) 
4337         // this is unused variable 
4342     else if ( HasFlag(wxLC_SMALL_ICON
) ) 
4344         // this is unused variable 
4345         mode 
= wxLC_ICON
;  // no typo 
4351         wxFAIL_MSG( _T("unknown mode") ); 
4354     wxListLineData 
*line 
= new wxListLineData(this); 
4356     line
->SetItem( 0, item 
); 
4358     m_lines
.Insert( line
, id 
); 
4362     // If an item is selected at or below the point of insertion, we need to 
4363     // increment the member variables because the current row's index has gone 
4365     if ( HasCurrent() && m_current 
>= id 
) 
4370     SendNotify(id
, wxEVT_COMMAND_LIST_INSERT_ITEM
); 
4372     RefreshLines(id
, GetItemCount() - 1); 
4375 void wxListMainWindow::InsertColumn( long col
, wxListItem 
&item 
) 
4378     if ( HasFlag(wxLC_REPORT
) ) 
4380         if (item
.m_width 
== wxLIST_AUTOSIZE_USEHEADER
) 
4381             item
.m_width 
= GetTextLength( item
.m_text 
); 
4383         wxListHeaderData 
*column 
= new wxListHeaderData( item 
); 
4384         bool insert 
= (col 
>= 0) && ((size_t)col 
< m_columns
.GetCount()); 
4387             wxListHeaderDataList::compatibility_iterator node 
= m_columns
.Item( col 
); 
4388             m_columns
.Insert( node
, column 
); 
4392             m_columns
.Append( column 
); 
4397             // update all the items 
4398             for ( size_t i 
= 0; i 
< m_lines
.GetCount(); i
++ ) 
4400                 wxListLineData 
* const line 
= GetLine(i
); 
4401                 wxListItemData 
* const data 
= new wxListItemData(this); 
4403                     line
->m_items
.Insert(col
, data
); 
4405                     line
->m_items
.Append(data
); 
4409         // invalidate it as it has to be recalculated 
4414 // ---------------------------------------------------------------------------- 
4416 // ---------------------------------------------------------------------------- 
4418 wxListCtrlCompare list_ctrl_compare_func_2
; 
4419 long              list_ctrl_compare_data
; 
4421 int LINKAGEMODE 
list_ctrl_compare_func_1( wxListLineData 
**arg1
, wxListLineData 
**arg2 
) 
4423     wxListLineData 
*line1 
= *arg1
; 
4424     wxListLineData 
*line2 
= *arg2
; 
4426     line1
->GetItem( 0, item 
); 
4427     long data1 
= item
.m_data
; 
4428     line2
->GetItem( 0, item 
); 
4429     long data2 
= item
.m_data
; 
4430     return list_ctrl_compare_func_2( data1
, data2
, list_ctrl_compare_data 
); 
4433 void wxListMainWindow::SortItems( wxListCtrlCompare fn
, long data 
) 
4435     list_ctrl_compare_func_2 
= fn
; 
4436     list_ctrl_compare_data 
= data
; 
4437     m_lines
.Sort( list_ctrl_compare_func_1 
); 
4441 // ---------------------------------------------------------------------------- 
4443 // ---------------------------------------------------------------------------- 
4445 void wxListMainWindow::OnScroll(wxScrollWinEvent
& event
) 
4447     // update our idea of which lines are shown when we redraw the window the 
4449     ResetVisibleLinesRange(); 
4452 #if defined(__WXGTK__) && !defined(__WXUNIVERSAL__) 
4453     wxScrolledWindow::OnScroll(event
); 
4455     HandleOnScroll( event 
); 
4458     if ( event
.GetOrientation() == wxHORIZONTAL 
&& HasHeader() ) 
4460         wxGenericListCtrl
* lc 
= GetListCtrl(); 
4461         wxCHECK_RET( lc
, _T("no listctrl window?") ); 
4463         lc
->m_headerWin
->Refresh(); 
4464         lc
->m_headerWin
->Update(); 
4468 int wxListMainWindow::GetCountPerPage() const 
4470     if ( !m_linesPerPage 
) 
4472         wxConstCast(this, wxListMainWindow
)-> 
4473             m_linesPerPage 
= GetClientSize().y 
/ GetLineHeight(); 
4476     return m_linesPerPage
; 
4479 void wxListMainWindow::GetVisibleLinesRange(size_t *from
, size_t *to
) 
4481     wxASSERT_MSG( HasFlag(wxLC_REPORT
), _T("this is for report mode only") ); 
4483     if ( m_lineFrom 
== (size_t)-1 ) 
4485         size_t count 
= GetItemCount(); 
4488             m_lineFrom 
= GetScrollPos(wxVERTICAL
); 
4490             // this may happen if SetScrollbars() hadn't been called yet 
4491             if ( m_lineFrom 
>= count 
) 
4492                 m_lineFrom 
= count 
- 1; 
4494             // we redraw one extra line but this is needed to make the redrawing 
4495             // logic work when there is a fractional number of lines on screen 
4496             m_lineTo 
= m_lineFrom 
+ m_linesPerPage
; 
4497             if ( m_lineTo 
>= count 
) 
4498                 m_lineTo 
= count 
- 1; 
4500         else // empty control 
4503             m_lineTo 
= (size_t)-1; 
4507     wxASSERT_MSG( IsEmpty() || 
4508                   (m_lineFrom 
<= m_lineTo 
&& m_lineTo 
< GetItemCount()), 
4509                   _T("GetVisibleLinesRange() returns incorrect result") ); 
4517 // ------------------------------------------------------------------------------------- 
4518 // wxGenericListCtrl 
4519 // ------------------------------------------------------------------------------------- 
4521 IMPLEMENT_DYNAMIC_CLASS(wxGenericListCtrl
, wxControl
) 
4523 BEGIN_EVENT_TABLE(wxGenericListCtrl
,wxControl
) 
4524   EVT_SIZE(wxGenericListCtrl::OnSize
) 
4527 wxGenericListCtrl::wxGenericListCtrl() 
4529     m_imageListNormal 
= (wxImageListType 
*) NULL
; 
4530     m_imageListSmall 
= (wxImageListType 
*) NULL
; 
4531     m_imageListState 
= (wxImageListType 
*) NULL
; 
4533     m_ownsImageListNormal 
= 
4534     m_ownsImageListSmall 
= 
4535     m_ownsImageListState 
= FALSE
; 
4537     m_mainWin 
= (wxListMainWindow
*) NULL
; 
4538     m_headerWin 
= (wxListHeaderWindow
*) NULL
; 
4542 wxGenericListCtrl::~wxGenericListCtrl() 
4544     if (m_ownsImageListNormal
) 
4545         delete m_imageListNormal
; 
4546     if (m_ownsImageListSmall
) 
4547         delete m_imageListSmall
; 
4548     if (m_ownsImageListState
) 
4549         delete m_imageListState
; 
4552 void wxGenericListCtrl::CalculateAndSetHeaderHeight() 
4556         // we use 'g' to get the descent, too 
4558         m_headerWin
->GetTextExtent(wxT("Hg"), &w
, &h
, &d
); 
4559         h 
+= d 
+ 2 * HEADER_OFFSET_Y 
+ EXTRA_HEIGHT
; 
4561         // only update if changed 
4562         if ( h 
!= m_headerHeight 
) 
4566             m_headerWin
->SetSize(m_headerWin
->GetSize().x
, m_headerHeight
); 
4568             if ( HasFlag(wxLC_REPORT
) && !HasFlag(wxLC_NO_HEADER
) ) 
4569                 ResizeReportView(TRUE
); 
4574 void wxGenericListCtrl::CreateHeaderWindow() 
4576     m_headerWin 
= new wxListHeaderWindow
 
4578                         this, -1, m_mainWin
, 
4580                         wxSize(GetClientSize().x
, m_headerHeight
), 
4583     CalculateAndSetHeaderHeight(); 
4586 bool wxGenericListCtrl::Create(wxWindow 
*parent
, 
4591                         const wxValidator 
&validator
, 
4592                         const wxString 
&name
) 
4596     m_imageListState 
= (wxImageListType 
*) NULL
; 
4597     m_ownsImageListNormal 
= 
4598     m_ownsImageListSmall 
= 
4599     m_ownsImageListState 
= FALSE
; 
4601     m_mainWin 
= (wxListMainWindow
*) NULL
; 
4602     m_headerWin 
= (wxListHeaderWindow
*) NULL
; 
4606     if ( !(style 
& wxLC_MASK_TYPE
) ) 
4608         style 
= style 
| wxLC_LIST
; 
4611     if ( !wxControl::Create( parent
, id
, pos
, size
, style
, validator
, name 
) ) 
4614     // don't create the inner window with the border 
4615     style 
&= ~wxBORDER_MASK
; 
4617     m_mainWin 
= new wxListMainWindow( this, -1, wxPoint(0,0), size
, style 
); 
4619     if ( HasFlag(wxLC_REPORT
) ) 
4621         CreateHeaderWindow(); 
4623         if ( HasFlag(wxLC_NO_HEADER
) ) 
4625             // VZ: why do we create it at all then? 
4626             m_headerWin
->Show( FALSE 
); 
4633 void wxGenericListCtrl::SetSingleStyle( long style
, bool add 
) 
4635     wxASSERT_MSG( !(style 
& wxLC_VIRTUAL
), 
4636                   _T("wxLC_VIRTUAL can't be [un]set") ); 
4638     long flag 
= GetWindowStyle(); 
4642         if (style 
& wxLC_MASK_TYPE
) 
4643             flag 
&= ~(wxLC_MASK_TYPE 
| wxLC_VIRTUAL
); 
4644         if (style 
& wxLC_MASK_ALIGN
) 
4645             flag 
&= ~wxLC_MASK_ALIGN
; 
4646         if (style 
& wxLC_MASK_SORT
) 
4647             flag 
&= ~wxLC_MASK_SORT
; 
4659     SetWindowStyleFlag( flag 
); 
4662 void wxGenericListCtrl::SetWindowStyleFlag( long flag 
) 
4666         m_mainWin
->DeleteEverything(); 
4668         // has the header visibility changed? 
4669         bool hasHeader 
= HasFlag(wxLC_REPORT
) && !HasFlag(wxLC_NO_HEADER
), 
4670              willHaveHeader 
= (flag 
& wxLC_REPORT
) && !(flag 
& wxLC_NO_HEADER
); 
4672         if ( hasHeader 
!= willHaveHeader 
) 
4679                     // don't delete, just hide, as we can reuse it later 
4680                     m_headerWin
->Show(FALSE
); 
4682                 //else: nothing to do 
4684             else // must show header 
4688                     CreateHeaderWindow(); 
4690                 else // already have it, just show 
4692                     m_headerWin
->Show( TRUE 
); 
4696             ResizeReportView(willHaveHeader
); 
4700     wxWindow::SetWindowStyleFlag( flag 
); 
4703 bool wxGenericListCtrl::GetColumn(int col
, wxListItem 
&item
) const 
4705     m_mainWin
->GetColumn( col
, item 
); 
4709 bool wxGenericListCtrl::SetColumn( int col
, wxListItem
& item 
) 
4711     m_mainWin
->SetColumn( col
, item 
); 
4715 int wxGenericListCtrl::GetColumnWidth( int col 
) const 
4717     return m_mainWin
->GetColumnWidth( col 
); 
4720 bool wxGenericListCtrl::SetColumnWidth( int col
, int width 
) 
4722     m_mainWin
->SetColumnWidth( col
, width 
); 
4726 int wxGenericListCtrl::GetCountPerPage() const 
4728   return m_mainWin
->GetCountPerPage();  // different from Windows ? 
4731 bool wxGenericListCtrl::GetItem( wxListItem 
&info 
) const 
4733     m_mainWin
->GetItem( info 
); 
4737 bool wxGenericListCtrl::SetItem( wxListItem 
&info 
) 
4739     m_mainWin
->SetItem( info 
); 
4743 long wxGenericListCtrl::SetItem( long index
, int col
, const wxString
& label
, int imageId 
) 
4746     info
.m_text 
= label
; 
4747     info
.m_mask 
= wxLIST_MASK_TEXT
; 
4748     info
.m_itemId 
= index
; 
4752         info
.m_image 
= imageId
; 
4753         info
.m_mask 
|= wxLIST_MASK_IMAGE
; 
4755     m_mainWin
->SetItem(info
); 
4759 int wxGenericListCtrl::GetItemState( long item
, long stateMask 
) const 
4761     return m_mainWin
->GetItemState( item
, stateMask 
); 
4764 bool wxGenericListCtrl::SetItemState( long item
, long state
, long stateMask 
) 
4766     m_mainWin
->SetItemState( item
, state
, stateMask 
); 
4770 bool wxGenericListCtrl::SetItemImage( long item
, int image
, int WXUNUSED(selImage
) ) 
4773     info
.m_image 
= image
; 
4774     info
.m_mask 
= wxLIST_MASK_IMAGE
; 
4775     info
.m_itemId 
= item
; 
4776     m_mainWin
->SetItem( info 
); 
4780 wxString 
wxGenericListCtrl::GetItemText( long item 
) const 
4782     return m_mainWin
->GetItemText(item
); 
4785 void wxGenericListCtrl::SetItemText( long item
, const wxString
& str 
) 
4787     m_mainWin
->SetItemText(item
, str
); 
4790 long wxGenericListCtrl::GetItemData( long item 
) const 
4793     info
.m_itemId 
= item
; 
4794     m_mainWin
->GetItem( info 
); 
4798 bool wxGenericListCtrl::SetItemData( long item
, long data 
) 
4801     info
.m_mask 
= wxLIST_MASK_DATA
; 
4802     info
.m_itemId 
= item
; 
4804     m_mainWin
->SetItem( info 
); 
4808 wxRect 
wxGenericListCtrl::GetViewRect() const 
4810     return m_mainWin
->GetViewRect(); 
4813 bool wxGenericListCtrl::GetItemRect( long item
, wxRect 
&rect
,  int WXUNUSED(code
) ) const 
4815     m_mainWin
->GetItemRect( item
, rect 
); 
4816     if ( m_mainWin
->HasHeader() ) 
4817         rect
.y 
+= m_headerHeight 
+ 1; 
4821 bool wxGenericListCtrl::GetItemPosition( long item
, wxPoint
& pos 
) const 
4823     m_mainWin
->GetItemPosition( item
, pos 
); 
4827 bool wxGenericListCtrl::SetItemPosition( long WXUNUSED(item
), const wxPoint
& WXUNUSED(pos
) ) 
4832 int wxGenericListCtrl::GetItemCount() const 
4834     return m_mainWin
->GetItemCount(); 
4837 int wxGenericListCtrl::GetColumnCount() const 
4839     return m_mainWin
->GetColumnCount(); 
4842 void wxGenericListCtrl::SetItemSpacing( int spacing
, bool isSmall 
) 
4844     m_mainWin
->SetItemSpacing( spacing
, isSmall 
); 
4847 wxSize 
wxGenericListCtrl::GetItemSpacing() const 
4849     const int spacing 
= m_mainWin
->GetItemSpacing(HasFlag(wxLC_SMALL_ICON
)); 
4851     return wxSize(spacing
, spacing
); 
4854 int wxGenericListCtrl::GetItemSpacing( bool isSmall 
) const 
4856     return m_mainWin
->GetItemSpacing( isSmall 
); 
4859 void wxGenericListCtrl::SetItemTextColour( long item
, const wxColour 
&col 
) 
4862     info
.m_itemId 
= item
; 
4863     info
.SetTextColour( col 
); 
4864     m_mainWin
->SetItem( info 
); 
4867 wxColour 
wxGenericListCtrl::GetItemTextColour( long item 
) const 
4870     info
.m_itemId 
= item
; 
4871     m_mainWin
->GetItem( info 
); 
4872     return info
.GetTextColour(); 
4875 void wxGenericListCtrl::SetItemBackgroundColour( long item
, const wxColour 
&col 
) 
4878     info
.m_itemId 
= item
; 
4879     info
.SetBackgroundColour( col 
); 
4880     m_mainWin
->SetItem( info 
); 
4883 wxColour 
wxGenericListCtrl::GetItemBackgroundColour( long item 
) const 
4886     info
.m_itemId 
= item
; 
4887     m_mainWin
->GetItem( info 
); 
4888     return info
.GetBackgroundColour(); 
4891 int wxGenericListCtrl::GetSelectedItemCount() const 
4893     return m_mainWin
->GetSelectedItemCount(); 
4896 wxColour 
wxGenericListCtrl::GetTextColour() const 
4898     return GetForegroundColour(); 
4901 void wxGenericListCtrl::SetTextColour(const wxColour
& col
) 
4903     SetForegroundColour(col
); 
4906 long wxGenericListCtrl::GetTopItem() const 
4909     m_mainWin
->GetVisibleLinesRange(&top
, NULL
); 
4913 long wxGenericListCtrl::GetNextItem( long item
, int geom
, int state 
) const 
4915     return m_mainWin
->GetNextItem( item
, geom
, state 
); 
4918 wxImageListType 
*wxGenericListCtrl::GetImageList(int which
) const 
4920     if (which 
== wxIMAGE_LIST_NORMAL
) 
4922         return m_imageListNormal
; 
4924     else if (which 
== wxIMAGE_LIST_SMALL
) 
4926         return m_imageListSmall
; 
4928     else if (which 
== wxIMAGE_LIST_STATE
) 
4930         return m_imageListState
; 
4932     return (wxImageListType 
*) NULL
; 
4935 void wxGenericListCtrl::SetImageList( wxImageListType 
*imageList
, int which 
) 
4937     if ( which 
== wxIMAGE_LIST_NORMAL 
) 
4939         if (m_ownsImageListNormal
) delete m_imageListNormal
; 
4940         m_imageListNormal 
= imageList
; 
4941         m_ownsImageListNormal 
= FALSE
; 
4943     else if ( which 
== wxIMAGE_LIST_SMALL 
) 
4945         if (m_ownsImageListSmall
) delete m_imageListSmall
; 
4946         m_imageListSmall 
= imageList
; 
4947         m_ownsImageListSmall 
= FALSE
; 
4949     else if ( which 
== wxIMAGE_LIST_STATE 
) 
4951         if (m_ownsImageListState
) delete m_imageListState
; 
4952         m_imageListState 
= imageList
; 
4953         m_ownsImageListState 
= FALSE
; 
4956     m_mainWin
->SetImageList( imageList
, which 
); 
4959 void wxGenericListCtrl::AssignImageList(wxImageListType 
*imageList
, int which
) 
4961     SetImageList(imageList
, which
); 
4962     if ( which 
== wxIMAGE_LIST_NORMAL 
) 
4963         m_ownsImageListNormal 
= TRUE
; 
4964     else if ( which 
== wxIMAGE_LIST_SMALL 
) 
4965         m_ownsImageListSmall 
= TRUE
; 
4966     else if ( which 
== wxIMAGE_LIST_STATE 
) 
4967         m_ownsImageListState 
= TRUE
; 
4970 bool wxGenericListCtrl::Arrange( int WXUNUSED(flag
) ) 
4975 bool wxGenericListCtrl::DeleteItem( long item 
) 
4977     m_mainWin
->DeleteItem( item 
); 
4981 bool wxGenericListCtrl::DeleteAllItems() 
4983     m_mainWin
->DeleteAllItems(); 
4987 bool wxGenericListCtrl::DeleteAllColumns() 
4989     size_t count 
= m_mainWin
->m_columns
.GetCount(); 
4990     for ( size_t n 
= 0; n 
< count
; n
++ ) 
4996 void wxGenericListCtrl::ClearAll() 
4998     m_mainWin
->DeleteEverything(); 
5001 bool wxGenericListCtrl::DeleteColumn( int col 
) 
5003     m_mainWin
->DeleteColumn( col 
); 
5005     // if we don't have the header any longer, we need to relayout the window 
5006     if ( !GetColumnCount() ) 
5008         ResizeReportView(FALSE 
/* no header */); 
5014 void wxGenericListCtrl::Edit( long item 
) 
5016     m_mainWin
->EditLabel( item 
); 
5019 bool wxGenericListCtrl::EnsureVisible( long item 
) 
5021     m_mainWin
->EnsureVisible( item 
); 
5025 long wxGenericListCtrl::FindItem( long start
, const wxString
& str
,  bool partial 
) 
5027     return m_mainWin
->FindItem( start
, str
, partial 
); 
5030 long wxGenericListCtrl::FindItem( long start
, long data 
) 
5032     return m_mainWin
->FindItem( start
, data 
); 
5035 long wxGenericListCtrl::FindItem( long WXUNUSED(start
), const wxPoint
& WXUNUSED(pt
), 
5036                            int WXUNUSED(direction
)) 
5041 long wxGenericListCtrl::HitTest( const wxPoint 
&point
, int &flags 
) 
5043     return m_mainWin
->HitTest( (int)point
.x
, (int)point
.y
, flags 
); 
5046 long wxGenericListCtrl::InsertItem( wxListItem
& info 
) 
5048     m_mainWin
->InsertItem( info 
); 
5049     return info
.m_itemId
; 
5052 long wxGenericListCtrl::InsertItem( long index
, const wxString 
&label 
) 
5055     info
.m_text 
= label
; 
5056     info
.m_mask 
= wxLIST_MASK_TEXT
; 
5057     info
.m_itemId 
= index
; 
5058     return InsertItem( info 
); 
5061 long wxGenericListCtrl::InsertItem( long index
, int imageIndex 
) 
5064     info
.m_mask 
= wxLIST_MASK_IMAGE
; 
5065     info
.m_image 
= imageIndex
; 
5066     info
.m_itemId 
= index
; 
5067     return InsertItem( info 
); 
5070 long wxGenericListCtrl::InsertItem( long index
, const wxString 
&label
, int imageIndex 
) 
5073     info
.m_text 
= label
; 
5074     info
.m_image 
= imageIndex
; 
5075     info
.m_mask 
= wxLIST_MASK_TEXT 
| wxLIST_MASK_IMAGE
; 
5076     info
.m_itemId 
= index
; 
5077     return InsertItem( info 
); 
5080 long wxGenericListCtrl::InsertColumn( long col
, wxListItem 
&item 
) 
5082     wxCHECK_MSG( m_headerWin
, -1, _T("can't add column in non report mode") ); 
5084     m_mainWin
->InsertColumn( col
, item 
); 
5086     // if we hadn't had header before and have it now we need to relayout the 
5088     if ( GetColumnCount() == 1 && m_mainWin
->HasHeader() ) 
5090         ResizeReportView(TRUE 
/* have header */); 
5093     m_headerWin
->Refresh(); 
5098 long wxGenericListCtrl::InsertColumn( long col
, const wxString 
&heading
, 
5099                                int format
, int width 
) 
5102     item
.m_mask 
= wxLIST_MASK_TEXT 
| wxLIST_MASK_FORMAT
; 
5103     item
.m_text 
= heading
; 
5106         item
.m_mask 
|= wxLIST_MASK_WIDTH
; 
5107         item
.m_width 
= width
; 
5109     item
.m_format 
= format
; 
5111     return InsertColumn( col
, item 
); 
5114 bool wxGenericListCtrl::ScrollList( int WXUNUSED(dx
), int WXUNUSED(dy
) ) 
5120 // fn is a function which takes 3 long arguments: item1, item2, data. 
5121 // item1 is the long data associated with a first item (NOT the index). 
5122 // item2 is the long data associated with a second item (NOT the index). 
5123 // data is the same value as passed to SortItems. 
5124 // The return value is a negative number if the first item should precede the second 
5125 // item, a positive number of the second item should precede the first, 
5126 // or zero if the two items are equivalent. 
5127 // data is arbitrary data to be passed to the sort function. 
5129 bool wxGenericListCtrl::SortItems( wxListCtrlCompare fn
, long data 
) 
5131     m_mainWin
->SortItems( fn
, data 
); 
5135 // ---------------------------------------------------------------------------- 
5137 // ---------------------------------------------------------------------------- 
5139 void wxGenericListCtrl::OnSize(wxSizeEvent
& WXUNUSED(event
)) 
5144     ResizeReportView(m_mainWin
->HasHeader()); 
5146     m_mainWin
->RecalculatePositions(); 
5149 void wxGenericListCtrl::ResizeReportView(bool showHeader
) 
5152     GetClientSize( &cw
, &ch 
); 
5156         m_headerWin
->SetSize( 0, 0, cw
, m_headerHeight 
); 
5157         m_mainWin
->SetSize( 0, m_headerHeight 
+ 1, cw
, ch 
- m_headerHeight 
- 1 ); 
5159     else // no header window 
5161         m_mainWin
->SetSize( 0, 0, cw
, ch 
); 
5165 void wxGenericListCtrl::OnInternalIdle() 
5167     wxWindow::OnInternalIdle(); 
5169     // do it only if needed 
5170     if ( !m_mainWin
->m_dirty 
) 
5173     m_mainWin
->RecalculatePositions(); 
5176 // ---------------------------------------------------------------------------- 
5178 // ---------------------------------------------------------------------------- 
5180 bool wxGenericListCtrl::SetBackgroundColour( const wxColour 
&colour 
) 
5184         m_mainWin
->SetBackgroundColour( colour 
); 
5185         m_mainWin
->m_dirty 
= TRUE
; 
5191 bool wxGenericListCtrl::SetForegroundColour( const wxColour 
&colour 
) 
5193     if ( !wxWindow::SetForegroundColour( colour 
) ) 
5198         m_mainWin
->SetForegroundColour( colour 
); 
5199         m_mainWin
->m_dirty 
= TRUE
; 
5204         m_headerWin
->SetForegroundColour( colour 
); 
5210 bool wxGenericListCtrl::SetFont( const wxFont 
&font 
) 
5212     if ( !wxWindow::SetFont( font 
) ) 
5217         m_mainWin
->SetFont( font 
); 
5218         m_mainWin
->m_dirty 
= TRUE
; 
5223         m_headerWin
->SetFont( font 
); 
5224         CalculateAndSetHeaderHeight(); 
5232 // ---------------------------------------------------------------------------- 
5233 // methods forwarded to m_mainWin 
5234 // ---------------------------------------------------------------------------- 
5236 #if wxUSE_DRAG_AND_DROP 
5238 void wxGenericListCtrl::SetDropTarget( wxDropTarget 
*dropTarget 
) 
5240     m_mainWin
->SetDropTarget( dropTarget 
); 
5243 wxDropTarget 
*wxGenericListCtrl::GetDropTarget() const 
5245     return m_mainWin
->GetDropTarget(); 
5248 #endif // wxUSE_DRAG_AND_DROP 
5250 bool wxGenericListCtrl::SetCursor( const wxCursor 
&cursor 
) 
5252     return m_mainWin 
? m_mainWin
->wxWindow::SetCursor(cursor
) : FALSE
; 
5255 wxColour 
wxGenericListCtrl::GetBackgroundColour() const 
5257     return m_mainWin 
? m_mainWin
->GetBackgroundColour() : wxColour(); 
5260 wxColour 
wxGenericListCtrl::GetForegroundColour() const 
5262     return m_mainWin 
? m_mainWin
->GetForegroundColour() : wxColour(); 
5265 bool wxGenericListCtrl::DoPopupMenu( wxMenu 
*menu
, int x
, int y 
) 
5268     return m_mainWin
->PopupMenu( menu
, x
, y 
); 
5271 #endif // wxUSE_MENUS 
5274 void wxGenericListCtrl::SetFocus() 
5276     /* The test in window.cpp fails as we are a composite 
5277        window, so it checks against "this", but not m_mainWin. */ 
5278     if ( FindFocus() != this ) 
5279         m_mainWin
->SetFocus(); 
5282 // ---------------------------------------------------------------------------- 
5283 // virtual list control support 
5284 // ---------------------------------------------------------------------------- 
5286 wxString 
wxGenericListCtrl::OnGetItemText(long WXUNUSED(item
), long WXUNUSED(col
)) const 
5288     // this is a pure virtual function, in fact - which is not really pure 
5289     // because the controls which are not virtual don't need to implement it 
5290     wxFAIL_MSG( _T("wxGenericListCtrl::OnGetItemText not supposed to be called") ); 
5292     return wxEmptyString
; 
5295 int wxGenericListCtrl::OnGetItemImage(long WXUNUSED(item
)) const 
5298     wxFAIL_MSG( _T("wxGenericListCtrl::OnGetItemImage not supposed to be called") ); 
5304 wxGenericListCtrl::OnGetItemAttr(long WXUNUSED_UNLESS_DEBUG(item
)) const 
5306     wxASSERT_MSG( item 
>= 0 && item 
< GetItemCount(), 
5307                   _T("invalid item index in OnGetItemAttr()") ); 
5309     // no attributes by default 
5313 void wxGenericListCtrl::SetItemCount(long count
) 
5315     wxASSERT_MSG( IsVirtual(), _T("this is for virtual controls only") ); 
5317     m_mainWin
->SetItemCount(count
); 
5320 void wxGenericListCtrl::RefreshItem(long item
) 
5322     m_mainWin
->RefreshLine(item
); 
5325 void wxGenericListCtrl::RefreshItems(long itemFrom
, long itemTo
) 
5327     m_mainWin
->RefreshLines(itemFrom
, itemTo
); 
5331  * Generic wxListCtrl is more or less a container for two other 
5332  * windows which drawings are done upon. These are namely 
5333  * 'm_headerWin' and 'm_mainWin'. 
5334  * Here we override 'virtual wxWindow::Refresh()' to mimic the 
5335  * behaviour wxListCtrl has under wxMSW. 
5337 void wxGenericListCtrl::Refresh(bool eraseBackground
, const wxRect 
*rect
) 
5341         // The easy case, no rectangle specified. 
5343             m_headerWin
->Refresh(eraseBackground
); 
5346             m_mainWin
->Refresh(eraseBackground
); 
5350         // Refresh the header window 
5353             wxRect rectHeader 
= m_headerWin
->GetRect(); 
5354             rectHeader
.Intersect(*rect
); 
5355             if (rectHeader
.GetWidth() && rectHeader
.GetHeight()) 
5358                 m_headerWin
->GetPosition(&x
, &y
); 
5359                 rectHeader
.Offset(-x
, -y
); 
5360                 m_headerWin
->Refresh(eraseBackground
, &rectHeader
); 
5365         // Refresh the main window 
5368             wxRect rectMain 
= m_mainWin
->GetRect(); 
5369             rectMain
.Intersect(*rect
); 
5370             if (rectMain
.GetWidth() && rectMain
.GetHeight()) 
5373                 m_mainWin
->GetPosition(&x
, &y
); 
5374                 rectMain
.Offset(-x
, -y
); 
5375                 m_mainWin
->Refresh(eraseBackground
, &rectMain
); 
5381 void wxGenericListCtrl::Freeze() 
5383     m_mainWin
->Freeze(); 
5386 void wxGenericListCtrl::Thaw() 
5391 #endif // wxUSE_LISTCTRL