1 /////////////////////////////////////////////////////////////////////////////
4 // Author: Robert Roebling
6 // Copyright: (c) 1998 Robert Roebling
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
11 #pragma implementation "listctrl.h"
14 // For compilers that support precompilation, includes "wx.h".
15 #include "wx/wxprec.h"
21 #include "wx/dcscreen.h"
23 #include "wx/listctrl.h"
24 #include "wx/generic/imaglist.h"
26 #ifndef wxUSE_GENERIC_LIST_EXTENSIONS
27 #define wxUSE_GENERIC_LIST_EXTENSIONS 0
30 //-----------------------------------------------------------------------------
32 //-----------------------------------------------------------------------------
34 IMPLEMENT_DYNAMIC_CLASS(wxListItemData
,wxObject
);
36 wxListItemData::wxListItemData()
47 wxListItemData::wxListItemData( const wxListItem
&info
)
56 void wxListItemData::SetItem( const wxListItem
&info
)
58 if (info
.m_mask
& wxLIST_MASK_TEXT
) m_text
= info
.m_text
;
59 if (info
.m_mask
& wxLIST_MASK_IMAGE
) m_image
= info
.m_image
;
60 if (info
.m_mask
& wxLIST_MASK_DATA
) m_data
= info
.m_data
;
62 if ( info
.HasAttributes() )
65 *m_attr
= *info
.GetAttributes();
67 m_attr
= new wxListItemAttr(*info
.GetAttributes());
72 m_width
= info
.m_width
;
76 void wxListItemData::SetText( const wxString
&s
)
81 void wxListItemData::SetImage( int image
)
86 void wxListItemData::SetData( long data
)
91 void wxListItemData::SetPosition( int x
, int y
)
97 void wxListItemData::SetSize( int width
, int height
)
99 if (width
!= -1) m_width
= width
;
100 if (height
!= -1) m_height
= height
;
103 bool wxListItemData::HasImage() const
105 return (m_image
>= 0);
108 bool wxListItemData::HasText() const
110 return (!m_text
.IsNull());
113 bool wxListItemData::IsHit( int x
, int y
) const
115 return ((x
>= m_xpos
) && (x
<= m_xpos
+m_width
) && (y
>= m_ypos
) && (y
<= m_ypos
+m_height
));
118 void wxListItemData::GetText( wxString
&s
)
123 int wxListItemData::GetX() const
128 int wxListItemData::GetY() const
133 int wxListItemData::GetWidth() const
138 int wxListItemData::GetHeight() const
143 int wxListItemData::GetImage() const
148 void wxListItemData::GetItem( wxListItem
&info
) const
150 info
.m_text
= m_text
;
151 info
.m_image
= m_image
;
152 info
.m_data
= m_data
;
156 if ( m_attr
->HasTextColour() )
157 info
.SetTextColour(m_attr
->GetTextColour());
158 if ( m_attr
->HasBackgroundColour() )
159 info
.SetBackgroundColour(m_attr
->GetBackgroundColour());
160 if ( m_attr
->HasFont() )
161 info
.SetFont(m_attr
->GetFont());
165 //-----------------------------------------------------------------------------
167 //-----------------------------------------------------------------------------
169 IMPLEMENT_DYNAMIC_CLASS(wxListHeaderData
,wxObject
);
171 wxListHeaderData::wxListHeaderData()
182 wxListHeaderData::wxListHeaderData( const wxListItem
&item
)
190 void wxListHeaderData::SetItem( const wxListItem
&item
)
192 m_mask
= item
.m_mask
;
193 m_text
= item
.m_text
;
194 m_image
= item
.m_image
;
195 m_format
= item
.m_format
;
196 m_width
= item
.m_width
;
197 if (m_width
< 0) m_width
= 80;
198 if (m_width
< 6) m_width
= 6;
201 void wxListHeaderData::SetPosition( int x
, int y
)
207 void wxListHeaderData::SetHeight( int h
)
212 void wxListHeaderData::SetWidth( int w
)
215 if (m_width
< 0) m_width
= 80;
216 if (m_width
< 6) m_width
= 6;
219 void wxListHeaderData::SetFormat( int format
)
224 bool wxListHeaderData::HasImage() const
226 return (m_image
!= 0);
229 bool wxListHeaderData::HasText() const
231 return (m_text
.Length() > 0);
234 bool wxListHeaderData::IsHit( int x
, int y
) const
236 return ((x
>= m_xpos
) && (x
<= m_xpos
+m_width
) && (y
>= m_ypos
) && (y
<= m_ypos
+m_height
));
239 void wxListHeaderData::GetItem( wxListItem
&item
)
241 item
.m_mask
= m_mask
;
242 item
.m_text
= m_text
;
243 item
.m_image
= m_image
;
244 item
.m_format
= m_format
;
245 item
.m_width
= m_width
;
248 void wxListHeaderData::GetText( wxString
&s
)
253 int wxListHeaderData::GetImage() const
258 int wxListHeaderData::GetWidth() const
263 int wxListHeaderData::GetFormat() const
268 //-----------------------------------------------------------------------------
270 //-----------------------------------------------------------------------------
272 IMPLEMENT_DYNAMIC_CLASS(wxListLineData
,wxObject
);
274 wxListLineData::wxListLineData( wxListMainWindow
*owner
, int mode
, wxBrush
*hilightBrush
)
279 m_hilightBrush
= hilightBrush
;
280 m_items
.DeleteContents( TRUE
);
284 void wxListLineData::CalculateSize( wxDC
*dc
, int spacing
)
291 m_bound_all
.width
= m_spacing
;
292 m_bound_all
.height
= m_spacing
+13;
293 wxNode
*node
= m_items
.First();
296 wxListItemData
*item
= (wxListItemData
*)node
->Data();
297 wxString s
= item
->GetText();
299 dc
->GetTextExtent( s
, &lw
, &lh
);
300 if (lw
> m_spacing
) m_bound_all
.width
= lw
;
306 wxNode
*node
= m_items
.First();
309 wxListItemData
*item
= (wxListItemData
*)node
->Data();
310 wxString s
= item
->GetText();
312 dc
->GetTextExtent( s
, &lw
, &lh
);
313 m_bound_all
.width
= lw
;
314 m_bound_all
.height
= lh
;
315 if (item
->HasImage())
319 m_owner
->GetImageSize( item
->GetImage(), w
, h
);
320 m_bound_all
.width
+= 4 + w
;
321 if (h
> m_bound_all
.height
) m_bound_all
.height
= h
;
328 m_bound_all
.width
= 0;
329 m_bound_all
.height
= 0;
330 wxNode
*node
= m_items
.First();
333 wxListItemData
*item
= (wxListItemData
*)node
->Data();
336 if (s
.IsNull()) s
= "H";
338 dc
->GetTextExtent( s
, &lw
, &lh
);
339 item
->SetSize( item
->GetWidth(), lh
);
340 m_bound_all
.width
+= lw
;
341 m_bound_all
.height
= lh
;
349 void wxListLineData::SetPosition( wxDC
*dc
, int x
, int y
, int window_width
)
357 AssignRect( m_bound_icon
, 0, 0, 0, 0 );
358 AssignRect( m_bound_label
, 0, 0, 0, 0 );
359 AssignRect( m_bound_hilight
, m_bound_all
);
360 wxNode
*node
= m_items
.First();
363 wxListItemData
*item
= (wxListItemData
*)node
->Data();
364 if (item
->HasImage())
366 wxListItemData
*item
= (wxListItemData
*)node
->Data();
369 m_owner
->GetImageSize( item
->GetImage(), w
, h
);
370 m_bound_icon
.x
= m_bound_all
.x
+ (m_spacing
/2) - (w
/2);
371 m_bound_icon
.y
= m_bound_all
.y
+ m_spacing
- h
- 5;
372 m_bound_icon
.width
= w
;
373 m_bound_icon
.height
= h
;
374 if (!item
->HasText())
376 AssignRect( m_bound_hilight
, m_bound_icon
);
377 m_bound_hilight
.x
-= 5;
378 m_bound_hilight
.y
-= 5;
379 m_bound_hilight
.width
+= 9;
380 m_bound_hilight
.height
+= 9;
388 dc
->GetTextExtent( s
, &lw
, &lh
);
389 if (m_bound_all
.width
> m_spacing
)
390 m_bound_label
.x
= m_bound_all
.x
;
392 m_bound_label
.x
= m_bound_all
.x
+ (m_spacing
/2) - lw
/2;
393 m_bound_label
.y
= m_bound_all
.y
+ m_bound_all
.height
- lh
;
394 m_bound_label
.width
= lw
;
395 m_bound_label
.height
= lh
;
396 AssignRect( m_bound_hilight
, m_bound_label
);
397 m_bound_hilight
.x
-= 2;
398 m_bound_hilight
.y
-= 2;
399 m_bound_hilight
.width
+= 4;
400 m_bound_hilight
.height
+= 4;
407 AssignRect( m_bound_label
, m_bound_all
);
410 m_bound_all
.width
+= 4;
411 m_bound_all
.height
+= 3;
412 AssignRect( m_bound_hilight
, m_bound_all
);
413 AssignRect( m_bound_icon
, 0, 0, 0, 0 );
414 wxNode
*node
= m_items
.First();
417 wxListItemData
*item
= (wxListItemData
*)node
->Data();
418 if (item
->HasImage())
420 m_bound_icon
.x
= m_bound_all
.x
+ 2;
421 m_bound_icon
.y
= m_bound_all
.y
+ 2;
424 m_owner
->GetImageSize( item
->GetImage(), w
, h
);
425 m_bound_icon
.width
= w
;
426 m_bound_icon
.height
= h
;
427 m_bound_label
.x
+= 4 + w
;
428 m_bound_label
.width
-= 4 + w
;
436 dc
->GetTextExtent( "H", &lw
, &lh
);
439 m_bound_all
.height
= lh
+3;
440 m_bound_all
.width
= window_width
;
441 AssignRect( m_bound_hilight
, m_bound_all
);
442 AssignRect( m_bound_label
, m_bound_all
);
443 AssignRect( m_bound_icon
, 0, 0, 0, 0 );
444 wxNode
*node
= m_items
.First();
447 wxListItemData
*item
= (wxListItemData
*)node
->Data();
450 if (s
.IsEmpty()) s
= wxT("H");
452 dc
->GetTextExtent( s
, &lw
, &lh
);
453 m_bound_label
.width
= lw
;
454 m_bound_label
.height
= lh
;
455 if (item
->HasImage())
457 m_bound_icon
.x
= m_bound_all
.x
+ 2;
458 m_bound_icon
.y
= m_bound_all
.y
+ 2;
461 m_owner
->GetImageSize( item
->GetImage(), w
, h
);
462 m_bound_icon
.width
= w
;
463 m_bound_icon
.height
= h
;
464 m_bound_label
.x
+= 4 + w
;
472 void wxListLineData::SetColumnPosition( int index
, int x
)
474 wxNode
*node
= m_items
.Nth( (size_t)index
);
477 wxListItemData
*item
= (wxListItemData
*)node
->Data();
478 item
->SetPosition( x
, m_bound_all
.y
+1 );
482 void wxListLineData::GetSize( int &width
, int &height
)
484 width
= m_bound_all
.width
;
485 height
= m_bound_all
.height
;
488 void wxListLineData::GetExtent( int &x
, int &y
, int &width
, int &height
)
492 width
= m_bound_all
.width
;
493 height
= m_bound_all
.height
;
496 void wxListLineData::GetLabelExtent( int &x
, int &y
, int &width
, int &height
)
500 width
= m_bound_label
.width
;
501 height
= m_bound_label
.height
;
504 void wxListLineData::GetRect( wxRect
&rect
)
506 AssignRect( rect
, m_bound_all
);
509 long wxListLineData::IsHit( int x
, int y
)
511 wxNode
*node
= m_items
.First();
514 wxListItemData
*item
= (wxListItemData
*)node
->Data();
515 if (item
->HasImage() && IsInRect( x
, y
, m_bound_icon
)) return wxLIST_HITTEST_ONITEMICON
;
516 if (item
->HasText() && IsInRect( x
, y
, m_bound_label
)) return wxLIST_HITTEST_ONITEMLABEL
;
517 // if (!(item->HasImage() || item->HasText())) return 0;
519 // if there is no icon or text = empty
520 if (IsInRect( x
, y
, m_bound_all
)) return wxLIST_HITTEST_ONITEMICON
;
524 void wxListLineData::InitItems( int num
)
526 for (int i
= 0; i
< num
; i
++) m_items
.Append( new wxListItemData() );
529 void wxListLineData::SetItem( int index
, const wxListItem
&info
)
531 wxNode
*node
= m_items
.Nth( index
);
534 wxListItemData
*item
= (wxListItemData
*)node
->Data();
535 item
->SetItem( info
);
539 void wxListLineData::GetItem( int index
, wxListItem
&info
)
542 wxNode
*node
= m_items
.Nth( i
);
545 wxListItemData
*item
= (wxListItemData
*)node
->Data();
546 item
->GetItem( info
);
550 void wxListLineData::GetText( int index
, wxString
&s
)
553 wxNode
*node
= m_items
.Nth( i
);
557 wxListItemData
*item
= (wxListItemData
*)node
->Data();
562 void wxListLineData::SetText( int index
, const wxString s
)
565 wxNode
*node
= m_items
.Nth( i
);
568 wxListItemData
*item
= (wxListItemData
*)node
->Data();
573 int wxListLineData::GetImage( int index
)
576 wxNode
*node
= m_items
.Nth( i
);
579 wxListItemData
*item
= (wxListItemData
*)node
->Data();
580 return item
->GetImage();
585 void wxListLineData::SetAttributes(wxDC
*dc
,
586 const wxListItemAttr
*attr
,
587 const wxColour
& colText
,
591 // don't use foregroud colour for drawing highlighted items - this might
592 // make them completely invisible (and there is no way to do bit
593 // arithmetics on wxColour, unfortunately)
594 if ( !hilight
&& attr
&& attr
->HasTextColour() )
596 dc
->SetTextForeground(attr
->GetTextColour());
600 dc
->SetTextForeground(colText
);
603 if ( attr
&& attr
->HasFont() )
605 dc
->SetFont(attr
->GetFont());
613 void wxListLineData::DoDraw( wxDC
*dc
, bool hilight
, bool paintBG
)
615 wxCoord dev_x
= dc
->LogicalToDeviceX( m_bound_all
.x
-2 );
616 wxCoord dev_y
= dc
->LogicalToDeviceY( m_bound_all
.y
-2 );
617 wxCoord dev_w
= dc
->LogicalToDeviceXRel( m_bound_all
.width
+4 );
618 wxCoord dev_h
= dc
->LogicalToDeviceYRel( m_bound_all
.height
+4 );
620 if (!m_owner
->IsExposed( dev_x
, dev_y
, dev_w
, dev_h
))
625 wxWindow
*listctrl
= m_owner
->GetParent();
627 // default foreground colour
631 colText
= wxSystemSettings::GetSystemColour( wxSYS_COLOUR_HIGHLIGHTTEXT
);
635 colText
= listctrl
->GetForegroundColour();
639 wxFont font
= listctrl
->GetFont();
641 // VZ: currently we set the colours/fonts only once, but like this (i.e.
642 // using SetAttributes() inside the loop), it will be trivial to
643 // customize the subitems (in report mode) too.
644 wxListItemData
*item
= (wxListItemData
*)m_items
.First()->Data();
645 wxListItemAttr
*attr
= item
->GetAttributes();
646 SetAttributes(dc
, attr
, colText
, font
, hilight
);
648 bool hasBgCol
= attr
&& attr
->HasBackgroundColour();
649 if ( paintBG
|| hasBgCol
)
653 dc
->SetBrush( * m_hilightBrush
);
658 dc
->SetBrush(wxBrush(attr
->GetBackgroundColour(), wxSOLID
));
660 dc
->SetBrush( * wxWHITE_BRUSH
);
663 dc
->SetPen( * wxTRANSPARENT_PEN
);
664 dc
->DrawRectangle( m_bound_hilight
.x
, m_bound_hilight
.y
,
665 m_bound_hilight
.width
, m_bound_hilight
.height
);
668 if (m_mode
== wxLC_REPORT
)
670 wxNode
*node
= m_items
.First();
673 wxListItemData
*item
= (wxListItemData
*)node
->Data();
674 dc
->SetClippingRegion( item
->GetX(), item
->GetY(), item
->GetWidth()-3, item
->GetHeight() );
675 int x
= item
->GetX();
676 if (item
->HasImage())
679 m_owner
->DrawImage( item
->GetImage(), dc
, x
, item
->GetY() );
680 m_owner
->GetImageSize( item
->GetImage(), x
, y
);
681 x
+= item
->GetX() + 5;
685 dc
->DrawText( item
->GetText(), x
, item
->GetY() );
687 dc
->DestroyClippingRegion();
693 wxNode
*node
= m_items
.First();
696 wxListItemData
*item
= (wxListItemData
*)node
->Data();
697 if (item
->HasImage())
699 m_owner
->DrawImage( item
->GetImage(), dc
, m_bound_icon
.x
, m_bound_icon
.y
);
703 dc
->DrawText( item
->GetText(), m_bound_label
.x
, m_bound_label
.y
);
709 void wxListLineData::Hilight( bool on
)
711 if (on
== m_hilighted
) return;
714 m_owner
->SelectLine( this );
716 m_owner
->DeselectLine( this );
719 void wxListLineData::ReverseHilight( void )
721 m_hilighted
= !m_hilighted
;
723 m_owner
->SelectLine( this );
725 m_owner
->DeselectLine( this );
728 void wxListLineData::DrawRubberBand( wxDC
*dc
, bool on
)
732 dc
->SetPen( * wxBLACK_PEN
);
733 dc
->SetBrush( * wxTRANSPARENT_BRUSH
);
734 dc
->DrawRectangle( m_bound_hilight
.x
, m_bound_hilight
.y
,
735 m_bound_hilight
.width
, m_bound_hilight
.height
);
739 void wxListLineData::Draw( wxDC
*dc
)
741 DoDraw( dc
, m_hilighted
, m_hilighted
);
744 bool wxListLineData::IsInRect( int x
, int y
, const wxRect
&rect
)
746 return ((x
>= rect
.x
) && (x
<= rect
.x
+rect
.width
) &&
747 (y
>= rect
.y
) && (y
<= rect
.y
+rect
.height
));
750 bool wxListLineData::IsHilighted( void )
755 void wxListLineData::AssignRect( wxRect
&dest
, int x
, int y
, int width
, int height
)
760 dest
.height
= height
;
763 void wxListLineData::AssignRect( wxRect
&dest
, const wxRect
&source
)
767 dest
.width
= source
.width
;
768 dest
.height
= source
.height
;
771 //-----------------------------------------------------------------------------
772 // wxListHeaderWindow
773 //-----------------------------------------------------------------------------
775 IMPLEMENT_DYNAMIC_CLASS(wxListHeaderWindow
,wxWindow
);
777 BEGIN_EVENT_TABLE(wxListHeaderWindow
,wxWindow
)
778 EVT_PAINT (wxListHeaderWindow::OnPaint
)
779 EVT_MOUSE_EVENTS (wxListHeaderWindow::OnMouse
)
780 EVT_SET_FOCUS (wxListHeaderWindow::OnSetFocus
)
783 wxListHeaderWindow::wxListHeaderWindow( void )
785 m_owner
= (wxListMainWindow
*) NULL
;
786 m_currentCursor
= (wxCursor
*) NULL
;
787 m_resizeCursor
= (wxCursor
*) NULL
;
788 m_isDragging
= FALSE
;
791 wxListHeaderWindow::wxListHeaderWindow( wxWindow
*win
, wxWindowID id
, wxListMainWindow
*owner
,
792 const wxPoint
&pos
, const wxSize
&size
,
793 long style
, const wxString
&name
) :
794 wxWindow( win
, id
, pos
, size
, style
, name
)
797 // m_currentCursor = wxSTANDARD_CURSOR;
798 m_currentCursor
= (wxCursor
*) NULL
;
799 m_resizeCursor
= new wxCursor( wxCURSOR_SIZEWE
);
800 m_isDragging
= FALSE
;
801 SetBackgroundColour( wxSystemSettings::GetSystemColour( wxSYS_COLOUR_BTNFACE
) );
804 wxListHeaderWindow::~wxListHeaderWindow( void )
806 delete m_resizeCursor
;
809 void wxListHeaderWindow::DoDrawRect( wxDC
*dc
, int x
, int y
, int w
, int h
)
811 const int m_corner
= 1;
813 dc
->SetBrush( *wxTRANSPARENT_BRUSH
);
815 dc
->SetPen( *wxBLACK_PEN
);
816 dc
->DrawLine( x
+w
-m_corner
+1, y
, x
+w
, y
+h
); // right (outer)
817 dc
->DrawRectangle( x
, y
+h
, w
+1, 1 ); // bottom (outer)
819 wxPen
pen( wxSystemSettings::GetSystemColour( wxSYS_COLOUR_BTNSHADOW
), 1, wxSOLID
);
822 dc
->DrawLine( x
+w
-m_corner
, y
, x
+w
-1, y
+h
); // right (inner)
823 dc
->DrawRectangle( x
+1, y
+h
-1, w
-2, 1 ); // bottom (inner)
825 dc
->SetPen( *wxWHITE_PEN
);
826 dc
->DrawRectangle( x
, y
, w
-m_corner
+1, 1 ); // top (outer)
827 dc
->DrawRectangle( x
, y
, 1, h
); // left (outer)
828 dc
->DrawLine( x
, y
+h
-1, x
+1, y
+h
-1 );
829 dc
->DrawLine( x
+w
-1, y
, x
+w
-1, y
+1 );
832 void wxListHeaderWindow::OnPaint( wxPaintEvent
&WXUNUSED(event
) )
834 wxPaintDC
dc( this );
836 #if wxUSE_GENERIC_LIST_EXTENSIONS
837 if ( m_owner
->GetMode() & wxLC_REPORT
)
842 m_owner
->GetScrollPixelsPerUnit( &xpix
, &ypix
) ;
843 m_owner
->ViewStart( &x
, &y
) ;
844 dc
.SetDeviceOrigin( -x
* xpix
, 0 );
849 dc
.SetFont( GetFont() );
855 GetClientSize( &w
, &h
);
857 dc
.SetBackgroundMode(wxTRANSPARENT
);
858 dc
.SetTextForeground( *wxBLACK
);
860 // do *not* use the listctrl colour for headers - one day we will have a
861 // function to set it separately
865 int numColumns
= m_owner
->GetColumnCount();
867 for (int i
= 0; i
< numColumns
; i
++)
869 m_owner
->GetColumn( i
, item
);
870 int cw
= item
.m_width
-2;
871 #if wxUSE_GENERIC_LIST_EXTENSIONS
872 if ((i
+1 == numColumns
) || ( dc
.LogicalToDeviceX(x
+item
.m_width
) > w
-5))
873 cw
= dc
.DeviceToLogicalX(w
)-x
-1;
875 if ((i
+1 == numColumns
) || (x
+item
.m_width
> w
-5))
878 dc
.SetPen( *wxWHITE_PEN
);
880 DoDrawRect( &dc
, x
, y
, cw
, h
-2 );
881 dc
.SetClippingRegion( x
, y
, cw
-5, h
-4 );
882 dc
.DrawText( item
.m_text
, x
+4, y
+3 );
883 dc
.DestroyClippingRegion();
885 #if wxUSE_GENERIC_LIST_EXTENSIONS
886 if (dc
.LogicalToDeviceX(x
) > w
+5) break;
894 void wxListHeaderWindow::DrawCurrent()
898 int x2
= m_currentX
-1;
901 m_owner
->GetClientSize( &dummy
, &y2
);
902 ClientToScreen( &x1
, &y1
);
903 m_owner
->ClientToScreen( &x2
, &y2
);
906 dc
.SetLogicalFunction( wxINVERT
);
907 dc
.SetPen( wxPen( *wxBLACK
, 2, wxSOLID
) );
908 dc
.SetBrush( *wxTRANSPARENT_BRUSH
);
910 dc
.DrawLine( x1
, y1
, x2
, y2
);
912 dc
.SetLogicalFunction( wxCOPY
);
914 dc
.SetPen( wxNullPen
);
915 dc
.SetBrush( wxNullBrush
);
918 void wxListHeaderWindow::OnMouse( wxMouseEvent
&event
)
920 wxCoord x
= (wxCoord
)event
.GetX();
921 wxCoord y
= (wxCoord
)event
.GetY();
925 if (event
.ButtonUp())
928 m_isDragging
= FALSE
;
929 m_owner
->SetColumnWidth( m_column
, m_currentX
-m_minX
);
935 GetClientSize( &size_x
, & dummy
);
939 m_currentX
= m_minX
+7;
940 if (m_currentX
> size_x
-7) m_currentX
= size_x
-7;
947 bool hit_border
= FALSE
;
949 for (int j
= 0; j
< m_owner
->GetColumnCount()-1; j
++)
951 xpos
+= m_owner
->GetColumnWidth( j
);
953 if ((abs(x
-xpos
) < 3) && (y
< 22))
965 if (event
.LeftDown())
977 wxListEvent
le( wxEVT_COMMAND_LIST_COL_CLICK
, GetParent()->GetId() );
978 le
.SetEventObject( GetParent() );
980 GetParent()->GetEventHandler()->ProcessEvent( le
);
989 if (m_currentCursor
== wxSTANDARD_CURSOR
) SetCursor( * m_resizeCursor
);
990 m_currentCursor
= m_resizeCursor
;
994 if (m_currentCursor
!= wxSTANDARD_CURSOR
) SetCursor( * wxSTANDARD_CURSOR
);
995 m_currentCursor
= wxSTANDARD_CURSOR
;
1000 void wxListHeaderWindow::OnSetFocus( wxFocusEvent
&WXUNUSED(event
) )
1002 m_owner
->SetFocus();
1005 //-----------------------------------------------------------------------------
1006 // wxListRenameTimer (internal)
1007 //-----------------------------------------------------------------------------
1009 wxListRenameTimer::wxListRenameTimer( wxListMainWindow
*owner
)
1014 void wxListRenameTimer::Notify()
1016 m_owner
->OnRenameTimer();
1019 //-----------------------------------------------------------------------------
1020 // wxListTextCtrl (internal)
1021 //-----------------------------------------------------------------------------
1023 IMPLEMENT_DYNAMIC_CLASS(wxListTextCtrl
,wxTextCtrl
);
1025 BEGIN_EVENT_TABLE(wxListTextCtrl
,wxTextCtrl
)
1026 EVT_CHAR (wxListTextCtrl::OnChar
)
1027 EVT_KILL_FOCUS (wxListTextCtrl::OnKillFocus
)
1030 wxListTextCtrl::wxListTextCtrl( wxWindow
*parent
, const wxWindowID id
,
1031 bool *accept
, wxString
*res
, wxListMainWindow
*owner
,
1032 const wxString
&value
, const wxPoint
&pos
, const wxSize
&size
,
1033 #if wxUSE_VALIDATORS
1034 int style
, const wxValidator
& validator
, const wxString
&name
) :
1036 wxTextCtrl( parent
, id
, value
, pos
, size
, style
, validator
, name
)
1041 (*m_accept
) = FALSE
;
1043 m_startValue
= value
;
1046 void wxListTextCtrl::OnChar( wxKeyEvent
&event
)
1048 if (event
.m_keyCode
== WXK_RETURN
)
1051 (*m_res
) = GetValue();
1052 m_owner
->SetFocus();
1055 if (event
.m_keyCode
== WXK_ESCAPE
)
1057 (*m_accept
) = FALSE
;
1059 m_owner
->SetFocus();
1065 void wxListTextCtrl::OnKillFocus( wxFocusEvent
&WXUNUSED(event
) )
1067 if (wxPendingDelete
.Member(this)) return;
1069 wxPendingDelete
.Append(this);
1071 if ((*m_accept
) && ((*m_res
) != m_startValue
))
1072 m_owner
->OnRenameAccept();
1075 //-----------------------------------------------------------------------------
1077 //-----------------------------------------------------------------------------
1079 IMPLEMENT_DYNAMIC_CLASS(wxListMainWindow
,wxScrolledWindow
);
1081 BEGIN_EVENT_TABLE(wxListMainWindow
,wxScrolledWindow
)
1082 EVT_PAINT (wxListMainWindow::OnPaint
)
1083 EVT_SIZE (wxListMainWindow::OnSize
)
1084 EVT_MOUSE_EVENTS (wxListMainWindow::OnMouse
)
1085 EVT_CHAR (wxListMainWindow::OnChar
)
1086 EVT_KEY_DOWN (wxListMainWindow::OnKeyDown
)
1087 EVT_SET_FOCUS (wxListMainWindow::OnSetFocus
)
1088 EVT_KILL_FOCUS (wxListMainWindow::OnKillFocus
)
1089 EVT_SCROLLWIN (wxListMainWindow::OnScroll
)
1092 wxListMainWindow::wxListMainWindow()
1095 m_lines
.DeleteContents( TRUE
);
1096 m_columns
.DeleteContents( TRUE
);
1097 m_current
= (wxListLineData
*) NULL
;
1099 m_hilightBrush
= (wxBrush
*) NULL
;
1103 m_small_image_list
= (wxImageList
*) NULL
;
1104 m_normal_image_list
= (wxImageList
*) NULL
;
1105 m_small_spacing
= 30;
1106 m_normal_spacing
= 40;
1109 m_lastOnSame
= FALSE
;
1110 m_renameTimer
= new wxListRenameTimer( this );
1111 m_isCreated
= FALSE
;
1115 wxListMainWindow::wxListMainWindow( wxWindow
*parent
, wxWindowID id
,
1116 const wxPoint
&pos
, const wxSize
&size
,
1117 long style
, const wxString
&name
) :
1118 wxScrolledWindow( parent
, id
, pos
, size
, style
|wxHSCROLL
|wxVSCROLL
, name
)
1121 m_lines
.DeleteContents( TRUE
);
1122 m_columns
.DeleteContents( TRUE
);
1123 m_current
= (wxListLineData
*) NULL
;
1126 m_hilightBrush
= new wxBrush( wxSystemSettings::GetSystemColour(wxSYS_COLOUR_HIGHLIGHT
), wxSOLID
);
1127 m_small_image_list
= (wxImageList
*) NULL
;
1128 m_normal_image_list
= (wxImageList
*) NULL
;
1129 m_small_spacing
= 30;
1130 m_normal_spacing
= 40;
1133 m_isCreated
= FALSE
;
1137 if (m_mode
& wxLC_REPORT
)
1139 #if wxUSE_GENERIC_LIST_EXTENSIONS
1151 SetScrollbars( m_xScroll
, m_yScroll
, 0, 0, 0, 0 );
1154 m_lastOnSame
= FALSE
;
1155 m_renameTimer
= new wxListRenameTimer( this );
1156 m_renameAccept
= FALSE
;
1158 SetBackgroundColour( *wxWHITE
);
1161 wxListMainWindow::~wxListMainWindow()
1163 if (m_hilightBrush
) delete m_hilightBrush
;
1165 delete m_renameTimer
;
1168 void wxListMainWindow::RefreshLine( wxListLineData
*line
)
1170 if (m_dirty
) return;
1178 wxClientDC
dc(this);
1180 line
->GetExtent( x
, y
, w
, h
);
1182 dc
.LogicalToDeviceX(x
-3),
1183 dc
.LogicalToDeviceY(y
-3),
1184 dc
.LogicalToDeviceXRel(w
+6),
1185 dc
.LogicalToDeviceXRel(h
+6) );
1186 Refresh( TRUE
, &rect
);
1190 void wxListMainWindow::OnPaint( wxPaintEvent
&WXUNUSED(event
) )
1192 // Note: a wxPaintDC must be constructed even if no drawing is
1193 // done (a Windows requirement).
1194 wxPaintDC
dc( this );
1197 if (m_dirty
) return;
1199 if (m_lines
.GetCount() == 0) return;
1203 dc
.SetFont( GetFont() );
1205 if (m_mode
& wxLC_REPORT
)
1207 int lineSpacing
= 0;
1208 wxListLineData
*line
= (wxListLineData
*)m_lines
.First()->Data();
1210 line
->GetSize( dummy
, lineSpacing
);
1213 int y_s
= m_yScroll
*GetScrollPos( wxVERTICAL
);
1215 wxNode
*node
= m_lines
.Nth( y_s
/ lineSpacing
);
1216 for (int i
= 0; i
< m_visibleLines
+2; i
++)
1220 line
= (wxListLineData
*)node
->Data();
1222 node
= node
->Next();
1227 wxNode
*node
= m_lines
.First();
1230 wxListLineData
*line
= (wxListLineData
*)node
->Data();
1232 node
= node
->Next();
1236 if (m_current
) m_current
->DrawRubberBand( &dc
, m_hasFocus
);
1241 void wxListMainWindow::HilightAll( bool on
)
1243 wxNode
*node
= m_lines
.First();
1246 wxListLineData
*line
= (wxListLineData
*)node
->Data();
1247 if (line
->IsHilighted() != on
)
1249 line
->Hilight( on
);
1250 RefreshLine( line
);
1252 node
= node
->Next();
1256 void wxListMainWindow::SendNotify( wxListLineData
*line
, wxEventType command
)
1258 wxListEvent
le( command
, GetParent()->GetId() );
1259 le
.SetEventObject( GetParent() );
1260 le
.m_itemIndex
= GetIndexOfLine( line
);
1261 line
->GetItem( 0, le
.m_item
);
1262 GetParent()->GetEventHandler()->ProcessEvent( le
);
1263 // GetParent()->GetEventHandler()->AddPendingEvent( le );
1266 void wxListMainWindow::FocusLine( wxListLineData
*WXUNUSED(line
) )
1268 // SendNotify( line, wxEVT_COMMAND_LIST_ITEM_FOCUSSED );
1271 void wxListMainWindow::UnfocusLine( wxListLineData
*WXUNUSED(line
) )
1273 // SendNotify( line, wxEVT_COMMAND_LIST_ITEM_UNFOCUSSED );
1276 void wxListMainWindow::SelectLine( wxListLineData
*line
)
1278 SendNotify( line
, wxEVT_COMMAND_LIST_ITEM_SELECTED
);
1281 void wxListMainWindow::DeselectLine( wxListLineData
*line
)
1283 SendNotify( line
, wxEVT_COMMAND_LIST_ITEM_DESELECTED
);
1286 void wxListMainWindow::DeleteLine( wxListLineData
*line
)
1288 SendNotify( line
, wxEVT_COMMAND_LIST_DELETE_ITEM
);
1293 void wxListMainWindow::EditLabel( long item
)
1295 wxNode
*node
= m_lines
.Nth( (size_t)item
);
1296 wxCHECK_RET( node
, wxT("wrong index in wxListCtrl::Edit()") );
1298 m_currentEdit
= (wxListLineData
*) node
->Data();
1300 wxListEvent
le( wxEVT_COMMAND_LIST_BEGIN_LABEL_EDIT
, GetParent()->GetId() );
1301 le
.SetEventObject( GetParent() );
1302 le
.m_itemIndex
= GetIndexOfLine( m_currentEdit
);
1303 m_currentEdit
->GetItem( 0, le
.m_item
);
1304 GetParent()->GetEventHandler()->ProcessEvent( le
);
1306 if (!le
.IsAllowed())
1309 // We have to call this here because the label in
1310 // question might just have been added and no screen
1311 // update taken place.
1312 if (m_dirty
) wxYield();
1315 m_currentEdit
->GetText( 0, s
);
1320 m_currentEdit
->GetLabelExtent( x
, y
, w
, h
);
1322 wxClientDC
dc(this);
1324 x
= dc
.LogicalToDeviceX( x
);
1325 y
= dc
.LogicalToDeviceY( y
);
1327 wxListTextCtrl
*text
= new wxListTextCtrl(
1328 this, -1, &m_renameAccept
, &m_renameRes
, this, s
, wxPoint(x
-4,y
-4), wxSize(w
+11,h
+8) );
1332 void wxListMainWindow::OnRenameTimer()
1334 wxCHECK_RET( m_current
, wxT("invalid m_current") );
1336 Edit( m_lines
.IndexOf( m_current
) );
1339 void wxListMainWindow::OnRenameAccept()
1341 wxListEvent
le( wxEVT_COMMAND_LIST_END_LABEL_EDIT
, GetParent()->GetId() );
1342 le
.SetEventObject( GetParent() );
1343 le
.m_itemIndex
= GetIndexOfLine( m_currentEdit
);
1344 m_currentEdit
->GetItem( 0, le
.m_item
);
1345 le
.m_item
.m_text
= m_renameRes
;
1346 GetParent()->GetEventHandler()->ProcessEvent( le
);
1348 if (!le
.IsAllowed()) return;
1351 info
.m_mask
= wxLIST_MASK_TEXT
;
1352 info
.m_itemId
= le
.m_itemIndex
;
1353 info
.m_text
= m_renameRes
;
1354 info
.SetTextColour(le
.m_item
.GetTextColour());
1358 void wxListMainWindow::OnMouse( wxMouseEvent
&event
)
1360 if (GetParent()->GetEventHandler()->ProcessEvent( event
)) return;
1362 if (!m_current
) return;
1363 if (m_dirty
) return;
1364 if ( !(event
.Dragging() || event
.ButtonDown() || event
.LeftUp() || event
.ButtonDClick()) ) return;
1366 wxClientDC
dc(this);
1368 wxCoord x
= dc
.DeviceToLogicalX( (wxCoord
)event
.GetX() );
1369 wxCoord y
= dc
.DeviceToLogicalY( (wxCoord
)event
.GetY() );
1371 /* Did we actually hit an item ? */
1373 wxNode
*node
= m_lines
.First();
1374 wxListLineData
*line
= (wxListLineData
*) NULL
;
1377 line
= (wxListLineData
*)node
->Data();
1378 hitResult
= line
->IsHit( x
, y
);
1379 if (hitResult
) break;
1380 line
= (wxListLineData
*) NULL
;
1381 node
= node
->Next();
1384 if (event
.Dragging())
1386 if (m_dragCount
== 0)
1387 m_dragStart
= wxPoint(x
,y
);
1391 if (m_dragCount
!= 3) return;
1393 int command
= wxEVT_COMMAND_LIST_BEGIN_DRAG
;
1394 if (event
.RightIsDown()) command
= wxEVT_COMMAND_LIST_BEGIN_RDRAG
;
1396 wxListEvent
le( command
, GetParent()->GetId() );
1397 le
.SetEventObject( GetParent() );
1398 le
.m_pointDrag
= m_dragStart
;
1399 GetParent()->GetEventHandler()->ProcessEvent( le
);
1410 if (event
.ButtonDClick())
1413 m_lastOnSame
= FALSE
;
1414 m_renameTimer
->Stop();
1416 SendNotify( line
, wxEVT_COMMAND_LIST_ITEM_ACTIVATED
);
1421 if (event
.LeftUp() && m_lastOnSame
)
1424 if ((line
== m_current
) &&
1425 (hitResult
== wxLIST_HITTEST_ONITEMLABEL
) &&
1426 (m_mode
& wxLC_EDIT_LABELS
) )
1428 m_renameTimer
->Start( 100, TRUE
);
1430 m_lastOnSame
= FALSE
;
1434 if (event
.RightDown())
1436 SendNotify( line
, wxEVT_COMMAND_LIST_ITEM_RIGHT_CLICK
);
1440 if (event
.MiddleDown())
1442 SendNotify( line
, wxEVT_COMMAND_LIST_ITEM_MIDDLE_CLICK
);
1446 if (event
.LeftDown())
1449 wxListLineData
*oldCurrent
= m_current
;
1450 if (m_mode
& wxLC_SINGLE_SEL
)
1453 HilightAll( FALSE
);
1454 m_current
->ReverseHilight();
1455 RefreshLine( m_current
);
1459 if (event
.ShiftDown())
1462 m_current
->ReverseHilight();
1463 RefreshLine( m_current
);
1465 else if (event
.ControlDown())
1469 int numOfCurrent
= -1;
1470 node
= m_lines
.First();
1473 wxListLineData
*test_line
= (wxListLineData
*)node
->Data();
1475 if (test_line
== oldCurrent
) break;
1476 node
= node
->Next();
1480 node
= m_lines
.First();
1483 wxListLineData
*test_line
= (wxListLineData
*)node
->Data();
1485 if (test_line
== line
) break;
1486 node
= node
->Next();
1489 if (numOfLine
< numOfCurrent
)
1492 numOfLine
= numOfCurrent
;
1496 wxNode
*node
= m_lines
.Nth( numOfCurrent
);
1497 for (int i
= 0; i
<= numOfLine
-numOfCurrent
; i
++)
1499 wxListLineData
*test_line
= (wxListLineData
*)node
->Data();
1500 test_line
->Hilight(TRUE
);
1501 RefreshLine( test_line
);
1502 node
= node
->Next();
1508 HilightAll( FALSE
);
1509 m_current
->ReverseHilight();
1510 RefreshLine( m_current
);
1513 if (m_current
!= oldCurrent
)
1515 RefreshLine( oldCurrent
);
1516 UnfocusLine( oldCurrent
);
1517 FocusLine( m_current
);
1519 m_lastOnSame
= (m_current
== oldCurrent
);
1524 void wxListMainWindow::MoveToFocus()
1526 if (!m_current
) return;
1532 m_current
->GetExtent( x
, y
, w
, h
);
1536 GetClientSize( &w_p
, &h_p
);
1538 if (m_mode
& wxLC_REPORT
)
1540 int y_s
= m_yScroll
*GetScrollPos( wxVERTICAL
);
1541 if ((y
> y_s
) && (y
+h
< y_s
+h_p
)) return;
1542 if (y
-y_s
< 5) { Scroll( -1, (y
-5-h_p
/2)/m_yScroll
); }
1543 if (y
+h
+5 > y_s
+h_p
) { Scroll( -1, (y
+h
-h_p
/2+h
+15)/m_yScroll
); }
1547 int x_s
= m_xScroll
*GetScrollPos( wxHORIZONTAL
);
1548 if ((x
> x_s
) && (x
+w
< x_s
+w_p
)) return;
1549 if (x
-x_s
< 5) { Scroll( (x
-5)/m_xScroll
, -1 ); }
1550 if (x
+w
-5 > x_s
+w_p
) { Scroll( (x
+w
-w_p
+15)/m_xScroll
, -1 ); }
1554 void wxListMainWindow::OnArrowChar( wxListLineData
*newCurrent
, bool shiftDown
)
1556 if ((m_mode
& wxLC_SINGLE_SEL
) || (m_usedKeys
== FALSE
)) m_current
->Hilight( FALSE
);
1557 wxListLineData
*oldCurrent
= m_current
;
1558 m_current
= newCurrent
;
1560 if (shiftDown
|| (m_mode
& wxLC_SINGLE_SEL
)) m_current
->Hilight( TRUE
);
1561 RefreshLine( m_current
);
1562 RefreshLine( oldCurrent
);
1563 FocusLine( m_current
);
1564 UnfocusLine( oldCurrent
);
1567 void wxListMainWindow::OnKeyDown( wxKeyEvent
&event
)
1569 wxWindow
*parent
= GetParent();
1571 /* we propagate the key event up */
1572 wxKeyEvent
ke( wxEVT_KEY_DOWN
);
1573 ke
.m_shiftDown
= event
.m_shiftDown
;
1574 ke
.m_controlDown
= event
.m_controlDown
;
1575 ke
.m_altDown
= event
.m_altDown
;
1576 ke
.m_metaDown
= event
.m_metaDown
;
1577 ke
.m_keyCode
= event
.m_keyCode
;
1580 ke
.SetEventObject( parent
);
1581 if (parent
->GetEventHandler()->ProcessEvent( ke
)) return;
1586 void wxListMainWindow::OnChar( wxKeyEvent
&event
)
1588 wxWindow
*parent
= GetParent();
1590 /* we send a list_key event up */
1591 wxListEvent
le( wxEVT_COMMAND_LIST_KEY_DOWN
, GetParent()->GetId() );
1592 le
.m_code
= (int)event
.KeyCode();
1593 le
.SetEventObject( parent
);
1594 parent
->GetEventHandler()->ProcessEvent( le
);
1596 /* we propagate the char event up */
1597 wxKeyEvent
ke( wxEVT_CHAR
);
1598 ke
.m_shiftDown
= event
.m_shiftDown
;
1599 ke
.m_controlDown
= event
.m_controlDown
;
1600 ke
.m_altDown
= event
.m_altDown
;
1601 ke
.m_metaDown
= event
.m_metaDown
;
1602 ke
.m_keyCode
= event
.m_keyCode
;
1605 ke
.SetEventObject( parent
);
1606 if (parent
->GetEventHandler()->ProcessEvent( ke
)) return;
1608 if (event
.KeyCode() == WXK_TAB
)
1610 wxNavigationKeyEvent nevent
;
1611 nevent
.SetDirection( !event
.ShiftDown() );
1612 nevent
.SetCurrentFocus( m_parent
);
1613 if (m_parent
->GetEventHandler()->ProcessEvent( nevent
)) return;
1616 /* no item -> nothing to do */
1623 switch (event
.KeyCode())
1627 wxNode
*node
= m_lines
.Member( m_current
)->Previous();
1628 if (node
) OnArrowChar( (wxListLineData
*)node
->Data(), event
.ShiftDown() );
1633 wxNode
*node
= m_lines
.Member( m_current
)->Next();
1634 if (node
) OnArrowChar( (wxListLineData
*)node
->Data(), event
.ShiftDown() );
1639 wxNode
*node
= m_lines
.Last();
1640 OnArrowChar( (wxListLineData
*)node
->Data(), event
.ShiftDown() );
1645 wxNode
*node
= m_lines
.First();
1646 OnArrowChar( (wxListLineData
*)node
->Data(), event
.ShiftDown() );
1652 if (m_mode
& wxLC_REPORT
)
1654 steps
= m_visibleLines
-1;
1659 wxNode
*node
= m_lines
.First();
1660 for (;;) { if (m_current
== (wxListLineData
*)node
->Data()) break; pos
++; node
= node
->Next(); }
1661 steps
= pos
% m_visibleLines
;
1663 wxNode
*node
= m_lines
.Member( m_current
);
1664 for (int i
= 0; i
< steps
; i
++) if (node
->Previous()) node
= node
->Previous();
1665 if (node
) OnArrowChar( (wxListLineData
*)node
->Data(), event
.ShiftDown() );
1671 if (m_mode
& wxLC_REPORT
)
1673 steps
= m_visibleLines
-1;
1677 int pos
= 0; wxNode
*node
= m_lines
.First();
1678 for (;;) { if (m_current
== (wxListLineData
*)node
->Data()) break; pos
++; node
= node
->Next(); }
1679 steps
= m_visibleLines
-(pos
% m_visibleLines
)-1;
1681 wxNode
*node
= m_lines
.Member( m_current
);
1682 for (int i
= 0; i
< steps
; i
++) if (node
->Next()) node
= node
->Next();
1683 if (node
) OnArrowChar( (wxListLineData
*)node
->Data(), event
.ShiftDown() );
1688 if (!(m_mode
& wxLC_REPORT
))
1690 wxNode
*node
= m_lines
.Member( m_current
);
1691 for (int i
= 0; i
<m_visibleLines
; i
++) if (node
->Previous()) node
= node
->Previous();
1692 if (node
) OnArrowChar( (wxListLineData
*)node
->Data(), event
.ShiftDown() );
1698 if (!(m_mode
& wxLC_REPORT
))
1700 wxNode
*node
= m_lines
.Member( m_current
);
1701 for (int i
= 0; i
<m_visibleLines
; i
++) if (node
->Next()) node
= node
->Next();
1702 if (node
) OnArrowChar( (wxListLineData
*)node
->Data(), event
.ShiftDown() );
1708 m_current
->ReverseHilight();
1709 RefreshLine( m_current
);
1714 if (!(m_mode
& wxLC_SINGLE_SEL
))
1716 wxListLineData
*oldCurrent
= m_current
;
1717 m_current
->ReverseHilight();
1718 wxNode
*node
= m_lines
.Member( m_current
)->Next();
1719 if (node
) m_current
= (wxListLineData
*)node
->Data();
1721 RefreshLine( oldCurrent
);
1722 RefreshLine( m_current
);
1723 UnfocusLine( oldCurrent
);
1724 FocusLine( m_current
);
1731 wxListEvent
le( wxEVT_COMMAND_LIST_ITEM_ACTIVATED
, GetParent()->GetId() );
1732 le
.SetEventObject( GetParent() );
1733 le
.m_itemIndex
= GetIndexOfLine( m_current
);
1734 m_current
->GetItem( 0, le
.m_item
);
1735 GetParent()->GetEventHandler()->ProcessEvent( le
);
1748 extern wxWindow
*g_focusWindow
;
1751 void wxListMainWindow::OnSetFocus( wxFocusEvent
&WXUNUSED(event
) )
1754 RefreshLine( m_current
);
1756 if (!GetParent()) return;
1759 g_focusWindow
= GetParent();
1762 wxFocusEvent
event( wxEVT_SET_FOCUS
, GetParent()->GetId() );
1763 event
.SetEventObject( GetParent() );
1764 GetParent()->GetEventHandler()->ProcessEvent( event
);
1767 void wxListMainWindow::OnKillFocus( wxFocusEvent
&WXUNUSED(event
) )
1770 RefreshLine( m_current
);
1773 void wxListMainWindow::OnSize( wxSizeEvent
&WXUNUSED(event
) )
1776 We don't even allow the wxScrolledWindow::AdjustScrollbars() call
1781 void wxListMainWindow::DrawImage( int index
, wxDC
*dc
, int x
, int y
)
1783 if ((m_mode
& wxLC_ICON
) && (m_normal_image_list
))
1785 m_normal_image_list
->Draw( index
, *dc
, x
, y
, wxIMAGELIST_DRAW_TRANSPARENT
);
1788 if ((m_mode
& wxLC_SMALL_ICON
) && (m_small_image_list
))
1790 m_small_image_list
->Draw( index
, *dc
, x
, y
, wxIMAGELIST_DRAW_TRANSPARENT
);
1792 if ((m_mode
& wxLC_LIST
) && (m_small_image_list
))
1794 m_small_image_list
->Draw( index
, *dc
, x
, y
, wxIMAGELIST_DRAW_TRANSPARENT
);
1796 if ((m_mode
& wxLC_REPORT
) && (m_small_image_list
))
1798 m_small_image_list
->Draw( index
, *dc
, x
, y
, wxIMAGELIST_DRAW_TRANSPARENT
);
1803 void wxListMainWindow::GetImageSize( int index
, int &width
, int &height
)
1805 if ((m_mode
& wxLC_ICON
) && (m_normal_image_list
))
1807 m_normal_image_list
->GetSize( index
, width
, height
);
1810 if ((m_mode
& wxLC_SMALL_ICON
) && (m_small_image_list
))
1812 m_small_image_list
->GetSize( index
, width
, height
);
1815 if ((m_mode
& wxLC_LIST
) && (m_small_image_list
))
1817 m_small_image_list
->GetSize( index
, width
, height
);
1820 if ((m_mode
& wxLC_REPORT
) && (m_small_image_list
))
1822 m_small_image_list
->GetSize( index
, width
, height
);
1829 int wxListMainWindow::GetTextLength( wxString
&s
)
1831 wxClientDC
dc( this );
1834 dc
.GetTextExtent( s
, &lw
, &lh
);
1838 int wxListMainWindow::GetIndexOfLine( const wxListLineData
*line
)
1841 wxNode
*node
= m_lines
.First();
1844 if (line
== (wxListLineData
*)node
->Data()) return i
;
1846 node
= node
->Next();
1851 void wxListMainWindow::SetImageList( wxImageList
*imageList
, int which
)
1854 if (which
== wxIMAGE_LIST_NORMAL
) m_normal_image_list
= imageList
;
1855 if (which
== wxIMAGE_LIST_SMALL
) m_small_image_list
= imageList
;
1858 void wxListMainWindow::SetItemSpacing( int spacing
, bool isSmall
)
1863 m_small_spacing
= spacing
;
1867 m_normal_spacing
= spacing
;
1871 int wxListMainWindow::GetItemSpacing( bool isSmall
)
1873 if (isSmall
) return m_small_spacing
; else return m_normal_spacing
;
1876 void wxListMainWindow::SetColumn( int col
, wxListItem
&item
)
1879 wxNode
*node
= m_columns
.Nth( col
);
1882 if (item
.m_width
== wxLIST_AUTOSIZE_USEHEADER
) item
.m_width
= GetTextLength( item
.m_text
)+7;
1883 wxListHeaderData
*column
= (wxListHeaderData
*)node
->Data();
1884 column
->SetItem( item
);
1886 wxListCtrl
*lc
= (wxListCtrl
*) GetParent();
1887 if (lc
->m_headerWin
) lc
->m_headerWin
->Refresh();
1890 void wxListMainWindow::SetColumnWidth( int col
, int width
)
1892 if (!(m_mode
& wxLC_REPORT
)) return;
1896 wxNode
*node
= (wxNode
*) NULL
;
1898 if (width
== wxLIST_AUTOSIZE_USEHEADER
) width
= 80;
1899 if (width
== wxLIST_AUTOSIZE
)
1901 wxClientDC
dc(this);
1902 dc
.SetFont( GetFont() );
1904 node
= m_lines
.First();
1907 wxListLineData
*line
= (wxListLineData
*)node
->Data();
1908 wxNode
*n
= line
->m_items
.Nth( col
);
1911 wxListItemData
*item
= (wxListItemData
*)n
->Data();
1912 int current
= 0, ix
= 0, iy
= 0;
1913 wxCoord lx
= 0, ly
= 0;
1914 if (item
->HasImage())
1916 GetImageSize( item
->GetImage(), ix
, iy
);
1919 if (item
->HasText())
1922 item
->GetText( str
);
1923 dc
.GetTextExtent( str
, &lx
, &ly
);
1926 if (current
> max
) max
= current
;
1928 node
= node
->Next();
1933 node
= m_columns
.Nth( col
);
1936 wxListHeaderData
*column
= (wxListHeaderData
*)node
->Data();
1937 column
->SetWidth( width
);
1940 node
= m_lines
.First();
1943 wxListLineData
*line
= (wxListLineData
*)node
->Data();
1944 wxNode
*n
= line
->m_items
.Nth( col
);
1947 wxListItemData
*item
= (wxListItemData
*)n
->Data();
1948 item
->SetSize( width
, -1 );
1950 node
= node
->Next();
1953 wxListCtrl
*lc
= (wxListCtrl
*) GetParent();
1954 if (lc
->m_headerWin
) lc
->m_headerWin
->Refresh();
1957 void wxListMainWindow::GetColumn( int col
, wxListItem
&item
)
1959 wxNode
*node
= m_columns
.Nth( col
);
1962 wxListHeaderData
*column
= (wxListHeaderData
*)node
->Data();
1963 column
->GetItem( item
);
1975 int wxListMainWindow::GetColumnWidth( int col
)
1977 wxNode
*node
= m_columns
.Nth( col
);
1980 wxListHeaderData
*column
= (wxListHeaderData
*)node
->Data();
1981 return column
->GetWidth();
1989 int wxListMainWindow::GetColumnCount()
1991 return m_columns
.Number();
1994 int wxListMainWindow::GetCountPerPage()
1996 return m_visibleLines
;
1999 void wxListMainWindow::SetItem( wxListItem
&item
)
2002 wxNode
*node
= m_lines
.Nth( (size_t)item
.m_itemId
);
2005 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2006 if (m_mode
& wxLC_REPORT
) item
.m_width
= GetColumnWidth( item
.m_col
)-3;
2007 line
->SetItem( item
.m_col
, item
);
2011 void wxListMainWindow::SetItemState( long item
, long state
, long stateMask
)
2013 // m_dirty = TRUE; no recalcs needed
2015 wxListLineData
*oldCurrent
= m_current
;
2017 if (stateMask
& wxLIST_STATE_FOCUSED
)
2019 wxNode
*node
= m_lines
.Nth( (size_t)item
);
2022 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2023 UnfocusLine( m_current
);
2025 FocusLine( m_current
);
2026 RefreshLine( m_current
);
2027 if (oldCurrent
) RefreshLine( oldCurrent
);
2031 if (stateMask
& wxLIST_STATE_SELECTED
)
2033 bool on
= (state
& wxLIST_STATE_SELECTED
) != 0;
2034 if (!on
&& (m_mode
& wxLC_SINGLE_SEL
)) return;
2036 wxNode
*node
= m_lines
.Nth( (size_t)item
);
2039 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2040 if (m_mode
& wxLC_SINGLE_SEL
)
2042 UnfocusLine( m_current
);
2044 FocusLine( m_current
);
2045 if (oldCurrent
) oldCurrent
->Hilight( FALSE
);
2046 RefreshLine( m_current
);
2047 if (oldCurrent
) RefreshLine( oldCurrent
);
2049 bool on
= (state
& wxLIST_STATE_SELECTED
) != 0;
2050 if (on
!= line
->IsHilighted())
2052 line
->Hilight( on
);
2053 RefreshLine( line
);
2059 int wxListMainWindow::GetItemState( long item
, long stateMask
)
2061 int ret
= wxLIST_STATE_DONTCARE
;
2062 if (stateMask
& wxLIST_STATE_FOCUSED
)
2064 wxNode
*node
= m_lines
.Nth( (size_t)item
);
2067 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2068 if (line
== m_current
) ret
|= wxLIST_STATE_FOCUSED
;
2071 if (stateMask
& wxLIST_STATE_SELECTED
)
2073 wxNode
*node
= m_lines
.Nth( (size_t)item
);
2076 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2077 if (line
->IsHilighted()) ret
|= wxLIST_STATE_FOCUSED
;
2083 void wxListMainWindow::GetItem( wxListItem
&item
)
2085 wxNode
*node
= m_lines
.Nth( (size_t)item
.m_itemId
);
2088 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2089 line
->GetItem( item
.m_col
, item
);
2100 int wxListMainWindow::GetItemCount()
2102 return m_lines
.Number();
2105 void wxListMainWindow::GetItemRect( long index
, wxRect
&rect
)
2107 wxNode
*node
= m_lines
.Nth( (size_t)index
);
2110 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2111 line
->GetRect( rect
);
2122 bool wxListMainWindow::GetItemPosition(long item
, wxPoint
& pos
)
2124 wxNode
*node
= m_lines
.Nth( (size_t)item
);
2128 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2129 line
->GetRect( rect
);
2141 int wxListMainWindow::GetSelectedItemCount()
2144 wxNode
*node
= m_lines
.First();
2147 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2148 if (line
->IsHilighted()) ret
++;
2149 node
= node
->Next();
2154 void wxListMainWindow::SetMode( long mode
)
2161 if (m_mode
& wxLC_REPORT
)
2163 #if wxUSE_GENERIC_LIST_EXTENSIONS
2177 long wxListMainWindow::GetMode() const
2182 void wxListMainWindow::CalculatePositions()
2184 if (!m_lines
.First()) return;
2186 wxClientDC
dc( this );
2187 dc
.SetFont( GetFont() );
2189 int iconSpacing
= 0;
2190 if (m_mode
& wxLC_ICON
) iconSpacing
= m_normal_spacing
;
2191 if (m_mode
& wxLC_SMALL_ICON
) iconSpacing
= m_small_spacing
;
2193 // we take the first line (which also can be an icon or
2194 // an a text item in wxLC_ICON and wxLC_LIST modes) to
2195 // measure the size of the line
2199 int lineSpacing
= 0;
2201 wxListLineData
*line
= (wxListLineData
*)m_lines
.First()->Data();
2202 line
->CalculateSize( &dc
, iconSpacing
);
2204 line
->GetSize( dummy
, lineSpacing
);
2207 int clientWidth
= 0;
2208 int clientHeight
= 0;
2210 if (m_mode
& wxLC_REPORT
)
2214 int entireHeight
= m_lines
.Number() * lineSpacing
+ 2;
2215 int scroll_pos
= GetScrollPos( wxVERTICAL
);
2216 #if wxUSE_GENERIC_LIST_EXTENSIONS
2217 int x_scroll_pos
= GetScrollPos( wxHORIZONTAL
);
2219 SetScrollbars( m_xScroll
, m_yScroll
, 0, (entireHeight
+15) / m_yScroll
, 0, scroll_pos
, TRUE
);
2221 GetClientSize( &clientWidth
, &clientHeight
);
2223 wxNode
* node
= m_lines
.First();
2224 int entireWidth
= 0 ;
2227 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2228 line
->CalculateSize( &dc
, iconSpacing
);
2229 line
->SetPosition( &dc
, x
, y
, clientWidth
);
2231 for (int i
= 0; i
< GetColumnCount(); i
++)
2233 line
->SetColumnPosition( i
, col_x
);
2234 col_x
+= GetColumnWidth( i
);
2236 entireWidth
= wxMax( entireWidth
, col_x
) ;
2237 #if wxUSE_GENERIC_LIST_EXTENSIONS
2238 line
->SetPosition( &dc
, x
, y
, col_x
);
2240 y
+= lineSpacing
; // one pixel blank line between items
2241 node
= node
->Next();
2243 m_visibleLines
= clientHeight
/ lineSpacing
;
2244 #if wxUSE_GENERIC_LIST_EXTENSIONS
2245 SetScrollbars( m_xScroll
, m_yScroll
, entireWidth
/ m_xScroll
, (entireHeight
+15) / m_yScroll
, x_scroll_pos
, scroll_pos
, TRUE
);
2250 // at first we try without any scrollbar. if the items don't
2251 // fit into the window, we recalculate after subtracting an
2252 // approximated 15 pt for the horizontal scrollbar
2254 GetSize( &clientWidth
, &clientHeight
);
2255 clientHeight
-= 4; // sunken frame
2257 int entireWidth
= 0;
2259 for (int tries
= 0; tries
< 2; tries
++)
2262 int x
= 5; // painting is done at x-2
2263 int y
= 5; // painting is done at y-2
2266 int m_currentVisibleLines
= 0;
2267 wxNode
*node
= m_lines
.First();
2270 m_currentVisibleLines
++;
2271 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2272 line
->CalculateSize( &dc
, iconSpacing
);
2273 line
->SetPosition( &dc
, x
, y
, clientWidth
);
2274 line
->GetSize( lineWidth
, lineHeight
);
2275 if (lineWidth
> maxWidth
) maxWidth
= lineWidth
;
2277 if (m_currentVisibleLines
> m_visibleLines
)
2278 m_visibleLines
= m_currentVisibleLines
;
2279 if (y
+lineSpacing
-6 >= clientHeight
) // -6 for earlier "line breaking"
2281 m_currentVisibleLines
= 0;
2284 entireWidth
+= maxWidth
+6;
2287 node
= node
->Next();
2288 if (!node
) entireWidth
+= maxWidth
;
2289 if ((tries
== 0) && (entireWidth
> clientWidth
))
2291 clientHeight
-= 15; // scrollbar height
2293 m_currentVisibleLines
= 0;
2296 if (!node
) tries
= 1; // everything fits, no second try required
2300 int scroll_pos
= GetScrollPos( wxHORIZONTAL
);
2301 SetScrollbars( m_xScroll
, m_yScroll
, (entireWidth
+15) / m_xScroll
, 0, scroll_pos
, 0, TRUE
);
2305 void wxListMainWindow::RealizeChanges( void )
2309 wxNode
*node
= m_lines
.First();
2310 if (node
) m_current
= (wxListLineData
*)node
->Data();
2314 FocusLine( m_current
);
2315 if (m_mode
& wxLC_SINGLE_SEL
) m_current
->Hilight( TRUE
);
2319 long wxListMainWindow::GetNextItem( long item
, int WXUNUSED(geometry
), int state
)
2322 if (item
> 0) ret
= item
;
2323 if(ret
>= GetItemCount()) return -1;
2324 wxNode
*node
= m_lines
.Nth( (size_t)ret
);
2327 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2328 if ((state
& wxLIST_STATE_FOCUSED
) && (line
== m_current
)) return ret
;
2329 if ((state
& wxLIST_STATE_SELECTED
) && (line
->IsHilighted())) return ret
;
2330 if (!state
) return ret
;
2332 node
= node
->Next();
2337 void wxListMainWindow::DeleteItem( long index
)
2340 wxNode
*node
= m_lines
.Nth( (size_t)index
);
2343 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2344 if (m_current
== line
) m_current
= (wxListLineData
*) NULL
;
2346 m_lines
.DeleteNode( node
);
2350 void wxListMainWindow::DeleteColumn( int col
)
2352 wxCHECK_RET( col
< (int)m_columns
.GetCount(),
2353 wxT("attempting to delete inexistent column in wxListView") );
2356 wxNode
*node
= m_columns
.Nth( col
);
2357 if (node
) m_columns
.DeleteNode( node
);
2360 void wxListMainWindow::DeleteAllItems( void )
2363 m_current
= (wxListLineData
*) NULL
;
2365 // to make the deletion of all items faster, we don't send the
2366 // notifications in this case: this is compatible with wxMSW and
2367 // documented in DeleteAllItems() description
2369 wxNode
*node
= m_lines
.First();
2372 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2376 node
= node
->Next();
2383 void wxListMainWindow::DeleteEverything( void )
2386 m_current
= (wxListLineData
*) NULL
;
2387 wxNode
*node
= m_lines
.First();
2390 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2392 node
= node
->Next();
2395 m_current
= (wxListLineData
*) NULL
;
2399 void wxListMainWindow::EnsureVisible( long index
)
2401 // We have to call this here because the label in
2402 // question might just have been added and no screen
2403 // update taken place.
2404 if (m_dirty
) wxYield();
2406 wxListLineData
*oldCurrent
= m_current
;
2407 m_current
= (wxListLineData
*) NULL
;
2408 wxNode
*node
= m_lines
.Nth( (size_t)index
);
2409 if (node
) m_current
= (wxListLineData
*)node
->Data();
2410 if (m_current
) MoveToFocus();
2411 m_current
= oldCurrent
;
2414 long wxListMainWindow::FindItem(long start
, const wxString
& str
, bool WXUNUSED(partial
) )
2418 if (pos
< 0) pos
= 0;
2419 wxNode
*node
= m_lines
.Nth( (size_t)pos
);
2422 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2424 line
->GetText( 0, s
);
2425 if (s
== tmp
) return pos
;
2426 node
= node
->Next();
2432 long wxListMainWindow::FindItem(long start
, long data
)
2435 if (pos
< 0) pos
= 0;
2436 wxNode
*node
= m_lines
.Nth( (size_t)pos
);
2439 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2441 line
->GetItem( 0, item
);
2442 if (item
.m_data
== data
) return pos
;
2443 node
= node
->Next();
2449 long wxListMainWindow::HitTest( int x
, int y
, int &flags
)
2451 wxNode
*node
= m_lines
.First();
2455 wxListLineData
*line
= (wxListLineData
*)node
->Data();
2456 long ret
= line
->IsHit( x
, y
);
2462 node
= node
->Next();
2468 void wxListMainWindow::InsertItem( wxListItem
&item
)
2472 if (m_mode
& wxLC_REPORT
) mode
= wxLC_REPORT
;
2473 else if (m_mode
& wxLC_LIST
) mode
= wxLC_LIST
;
2474 else if (m_mode
& wxLC_ICON
) mode
= wxLC_ICON
;
2475 else if (m_mode
& wxLC_SMALL_ICON
) mode
= wxLC_ICON
; // no typo
2477 wxListLineData
*line
= new wxListLineData( this, mode
, m_hilightBrush
);
2479 if (m_mode
& wxLC_REPORT
)
2481 line
->InitItems( GetColumnCount() );
2482 item
.m_width
= GetColumnWidth( 0 )-3;
2486 line
->InitItems( 1 );
2489 line
->SetItem( 0, item
);
2490 if ((item
.m_itemId
>= 0) && (item
.m_itemId
< (int)m_lines
.GetCount()))
2492 wxNode
*node
= m_lines
.Nth( (size_t)item
.m_itemId
);
2493 if (node
) m_lines
.Insert( node
, line
);
2497 m_lines
.Append( line
);
2501 void wxListMainWindow::InsertColumn( long col
, wxListItem
&item
)
2504 if (m_mode
& wxLC_REPORT
)
2506 if (item
.m_width
== wxLIST_AUTOSIZE_USEHEADER
) item
.m_width
= GetTextLength( item
.m_text
);
2507 wxListHeaderData
*column
= new wxListHeaderData( item
);
2508 if ((col
>= 0) && (col
< (int)m_columns
.GetCount()))
2510 wxNode
*node
= m_columns
.Nth( (size_t)col
);
2512 m_columns
.Insert( node
, column
);
2516 m_columns
.Append( column
);
2521 wxListCtrlCompare list_ctrl_compare_func_2
;
2522 long list_ctrl_compare_data
;
2524 int LINKAGEMODE
list_ctrl_compare_func_1( const void *arg1
, const void *arg2
)
2526 wxListLineData
*line1
= *((wxListLineData
**)arg1
);
2527 wxListLineData
*line2
= *((wxListLineData
**)arg2
);
2529 line1
->GetItem( 0, item
);
2530 long data1
= item
.m_data
;
2531 line2
->GetItem( 0, item
);
2532 long data2
= item
.m_data
;
2533 return list_ctrl_compare_func_2( data1
, data2
, list_ctrl_compare_data
);
2536 void wxListMainWindow::SortItems( wxListCtrlCompare fn
, long data
)
2538 list_ctrl_compare_func_2
= fn
;
2539 list_ctrl_compare_data
= data
;
2540 m_lines
.Sort( list_ctrl_compare_func_1
);
2543 void wxListMainWindow::OnScroll(wxScrollWinEvent
& event
)
2545 wxScrolledWindow::OnScroll( event
) ;
2546 #if wxUSE_GENERIC_LIST_EXTENSIONS
2548 if (event
.GetOrientation() == wxHORIZONTAL
&& ( m_mode
& wxLC_REPORT
))
2550 wxListCtrl
* lc
= wxDynamicCast( GetParent() , wxListCtrl
) ;
2553 lc
->m_headerWin
->Refresh() ;
2555 lc
->m_headerWin
->MacUpdateImmediately() ;
2562 // -------------------------------------------------------------------------------------
2564 // -------------------------------------------------------------------------------------
2566 IMPLEMENT_DYNAMIC_CLASS(wxListItem
, wxObject
)
2568 wxListItem::wxListItem()
2577 m_format
= wxLIST_FORMAT_CENTRE
;
2583 void wxListItem::Clear()
2592 m_format
= wxLIST_FORMAT_CENTRE
;
2594 m_text
= wxEmptyString
;
2596 if (m_attr
) delete m_attr
;
2600 void wxListItem::ClearAttributes()
2602 if (m_attr
) delete m_attr
;
2606 // -------------------------------------------------------------------------------------
2608 // -------------------------------------------------------------------------------------
2610 IMPLEMENT_DYNAMIC_CLASS(wxListEvent
, wxNotifyEvent
)
2612 wxListEvent::wxListEvent( wxEventType commandType
, int id
):
2613 wxNotifyEvent( commandType
, id
)
2619 m_cancelled
= FALSE
;
2624 void wxListEvent::CopyObject(wxObject
& object_dest
) const
2626 wxListEvent
*obj
= (wxListEvent
*)&object_dest
;
2628 wxNotifyEvent::CopyObject(object_dest
);
2630 obj
->m_code
= m_code
;
2631 obj
->m_itemIndex
= m_itemIndex
;
2632 obj
->m_oldItemIndex
= m_oldItemIndex
;
2634 obj
->m_cancelled
= m_cancelled
;
2635 obj
->m_pointDrag
= m_pointDrag
;
2636 obj
->m_item
.m_mask
= m_item
.m_mask
;
2637 obj
->m_item
.m_itemId
= m_item
.m_itemId
;
2638 obj
->m_item
.m_col
= m_item
.m_col
;
2639 obj
->m_item
.m_state
= m_item
.m_state
;
2640 obj
->m_item
.m_stateMask
= m_item
.m_stateMask
;
2641 obj
->m_item
.m_text
= m_item
.m_text
;
2642 obj
->m_item
.m_image
= m_item
.m_image
;
2643 obj
->m_item
.m_data
= m_item
.m_data
;
2644 obj
->m_item
.m_format
= m_item
.m_format
;
2645 obj
->m_item
.m_width
= m_item
.m_width
;
2647 if ( m_item
.HasAttributes() )
2649 obj
->m_item
.SetTextColour(m_item
.GetTextColour());
2653 // -------------------------------------------------------------------------------------
2655 // -------------------------------------------------------------------------------------
2657 IMPLEMENT_DYNAMIC_CLASS(wxListCtrl
, wxControl
)
2659 BEGIN_EVENT_TABLE(wxListCtrl
,wxControl
)
2660 EVT_SIZE (wxListCtrl::OnSize
)
2661 EVT_IDLE (wxListCtrl::OnIdle
)
2664 wxListCtrl::wxListCtrl()
2666 m_imageListNormal
= (wxImageList
*) NULL
;
2667 m_imageListSmall
= (wxImageList
*) NULL
;
2668 m_imageListState
= (wxImageList
*) NULL
;
2669 m_mainWin
= (wxListMainWindow
*) NULL
;
2670 m_headerWin
= (wxListHeaderWindow
*) NULL
;
2673 wxListCtrl::~wxListCtrl()
2677 bool wxListCtrl::Create(wxWindow
*parent
,
2682 #if wxUSE_VALIDATORS
2683 const wxValidator
&validator
,
2685 const wxString
&name
)
2687 m_imageListNormal
= (wxImageList
*) NULL
;
2688 m_imageListSmall
= (wxImageList
*) NULL
;
2689 m_imageListState
= (wxImageList
*) NULL
;
2690 m_mainWin
= (wxListMainWindow
*) NULL
;
2691 m_headerWin
= (wxListHeaderWindow
*) NULL
;
2693 if ( !(style
& (wxLC_REPORT
| wxLC_LIST
| wxLC_ICON
)) )
2695 style
= style
| wxLC_LIST
;
2698 bool ret
= wxControl::Create( parent
,
2703 #if wxUSE_VALIDATORS
2708 #if wxUSE_VALIDATORS
2709 SetValidator( validator
);
2712 if (style
& wxSUNKEN_BORDER
)
2713 style
-= wxSUNKEN_BORDER
;
2715 m_mainWin
= new wxListMainWindow( this, -1, wxPoint(0,0), size
, style
);
2717 if (HasFlag(wxLC_REPORT
))
2718 m_headerWin
= new wxListHeaderWindow( this, -1, m_mainWin
, wxPoint(0,0), wxSize(size
.x
,23), wxTAB_TRAVERSAL
);
2720 m_headerWin
= (wxListHeaderWindow
*) NULL
;
2722 SetBackgroundColour( *wxWHITE
);
2727 void wxListCtrl::OnSize( wxSizeEvent
&WXUNUSED(event
) )
2729 /* handled in OnIdle */
2731 if (m_mainWin
) m_mainWin
->m_dirty
= TRUE
;
2734 void wxListCtrl::SetSingleStyle( long style
, bool add
)
2736 long flag
= GetWindowStyle();
2740 if (style
& wxLC_MASK_TYPE
) flag
= flag
& ~wxLC_MASK_TYPE
;
2741 if (style
& wxLC_MASK_ALIGN
) flag
= flag
& ~wxLC_MASK_ALIGN
;
2742 if (style
& wxLC_MASK_SORT
) flag
= flag
& ~wxLC_MASK_SORT
;
2751 if (flag
& style
) flag
-= style
;
2754 SetWindowStyleFlag( flag
);
2757 void wxListCtrl::SetWindowStyleFlag( long flag
)
2761 m_mainWin
->DeleteEverything();
2765 GetClientSize( &width
, &height
);
2767 m_mainWin
->SetMode( flag
);
2769 if (flag
& wxLC_REPORT
)
2771 if (!HasFlag(wxLC_REPORT
))
2775 m_headerWin
= new wxListHeaderWindow( this, -1, m_mainWin
,
2776 wxPoint(0,0), wxSize(width
,23), wxTAB_TRAVERSAL
);
2777 if (HasFlag(wxLC_NO_HEADER
))
2778 m_headerWin
->Show( FALSE
);
2782 if (flag
& wxLC_NO_HEADER
)
2783 m_headerWin
->Show( FALSE
);
2785 m_headerWin
->Show( TRUE
);
2791 if (HasFlag(wxLC_REPORT
) && !(HasFlag(wxLC_NO_HEADER
)))
2793 m_headerWin
->Show( FALSE
);
2798 wxWindow::SetWindowStyleFlag( flag
);
2801 bool wxListCtrl::GetColumn(int col
, wxListItem
&item
) const
2803 m_mainWin
->GetColumn( col
, item
);
2807 bool wxListCtrl::SetColumn( int col
, wxListItem
& item
)
2809 m_mainWin
->SetColumn( col
, item
);
2813 int wxListCtrl::GetColumnWidth( int col
) const
2815 return m_mainWin
->GetColumnWidth( col
);
2818 bool wxListCtrl::SetColumnWidth( int col
, int width
)
2820 m_mainWin
->SetColumnWidth( col
, width
);
2824 int wxListCtrl::GetCountPerPage() const
2826 return m_mainWin
->GetCountPerPage(); // different from Windows ?
2829 bool wxListCtrl::GetItem( wxListItem
&info
) const
2831 m_mainWin
->GetItem( info
);
2835 bool wxListCtrl::SetItem( wxListItem
&info
)
2837 m_mainWin
->SetItem( info
);
2841 long wxListCtrl::SetItem( long index
, int col
, const wxString
& label
, int imageId
)
2844 info
.m_text
= label
;
2845 info
.m_mask
= wxLIST_MASK_TEXT
;
2846 info
.m_itemId
= index
;
2850 info
.m_image
= imageId
;
2851 info
.m_mask
|= wxLIST_MASK_IMAGE
;
2853 m_mainWin
->SetItem(info
);
2857 int wxListCtrl::GetItemState( long item
, long stateMask
) const
2859 return m_mainWin
->GetItemState( item
, stateMask
);
2862 bool wxListCtrl::SetItemState( long item
, long state
, long stateMask
)
2864 m_mainWin
->SetItemState( item
, state
, stateMask
);
2868 bool wxListCtrl::SetItemImage( long item
, int image
, int WXUNUSED(selImage
) )
2871 info
.m_image
= image
;
2872 info
.m_mask
= wxLIST_MASK_IMAGE
;
2873 info
.m_itemId
= item
;
2874 m_mainWin
->SetItem( info
);
2878 wxString
wxListCtrl::GetItemText( long item
) const
2881 info
.m_itemId
= item
;
2882 m_mainWin
->GetItem( info
);
2886 void wxListCtrl::SetItemText( long item
, const wxString
&str
)
2889 info
.m_mask
= wxLIST_MASK_TEXT
;
2890 info
.m_itemId
= item
;
2892 m_mainWin
->SetItem( info
);
2895 long wxListCtrl::GetItemData( long item
) const
2898 info
.m_itemId
= item
;
2899 m_mainWin
->GetItem( info
);
2903 bool wxListCtrl::SetItemData( long item
, long data
)
2906 info
.m_mask
= wxLIST_MASK_DATA
;
2907 info
.m_itemId
= item
;
2909 m_mainWin
->SetItem( info
);
2913 bool wxListCtrl::GetItemRect( long item
, wxRect
&rect
, int WXUNUSED(code
) ) const
2915 m_mainWin
->GetItemRect( item
, rect
);
2919 bool wxListCtrl::GetItemPosition( long item
, wxPoint
& pos
) const
2921 m_mainWin
->GetItemPosition( item
, pos
);
2925 bool wxListCtrl::SetItemPosition( long WXUNUSED(item
), const wxPoint
& WXUNUSED(pos
) )
2930 int wxListCtrl::GetItemCount() const
2932 return m_mainWin
->GetItemCount();
2935 int wxListCtrl::GetColumnCount() const
2937 return m_mainWin
->GetColumnCount();
2940 void wxListCtrl::SetItemSpacing( int spacing
, bool isSmall
)
2942 m_mainWin
->SetItemSpacing( spacing
, isSmall
);
2945 int wxListCtrl::GetItemSpacing( bool isSmall
) const
2947 return m_mainWin
->GetItemSpacing( isSmall
);
2950 int wxListCtrl::GetSelectedItemCount() const
2952 return m_mainWin
->GetSelectedItemCount();
2955 wxColour
wxListCtrl::GetTextColour() const
2957 return GetForegroundColour();
2960 void wxListCtrl::SetTextColour(const wxColour
& col
)
2962 SetForegroundColour(col
);
2965 long wxListCtrl::GetTopItem() const
2970 long wxListCtrl::GetNextItem( long item
, int geom
, int state
) const
2972 return m_mainWin
->GetNextItem( item
, geom
, state
);
2975 wxImageList
*wxListCtrl::GetImageList(int which
) const
2977 if (which
== wxIMAGE_LIST_NORMAL
)
2979 return m_imageListNormal
;
2981 else if (which
== wxIMAGE_LIST_SMALL
)
2983 return m_imageListSmall
;
2985 else if (which
== wxIMAGE_LIST_STATE
)
2987 return m_imageListState
;
2989 return (wxImageList
*) NULL
;
2992 void wxListCtrl::SetImageList( wxImageList
*imageList
, int which
)
2994 m_mainWin
->SetImageList( imageList
, which
);
2997 bool wxListCtrl::Arrange( int WXUNUSED(flag
) )
3002 bool wxListCtrl::DeleteItem( long item
)
3004 m_mainWin
->DeleteItem( item
);
3008 bool wxListCtrl::DeleteAllItems()
3010 m_mainWin
->DeleteAllItems();
3014 bool wxListCtrl::DeleteAllColumns()
3016 for ( size_t n
= 0; n
< m_mainWin
->m_columns
.GetCount(); n
++ )
3022 void wxListCtrl::ClearAll()
3024 m_mainWin
->DeleteEverything();
3027 bool wxListCtrl::DeleteColumn( int col
)
3029 m_mainWin
->DeleteColumn( col
);
3033 void wxListCtrl::Edit( long item
)
3035 m_mainWin
->Edit( item
);
3038 bool wxListCtrl::EnsureVisible( long item
)
3040 m_mainWin
->EnsureVisible( item
);
3044 long wxListCtrl::FindItem( long start
, const wxString
& str
, bool partial
)
3046 return m_mainWin
->FindItem( start
, str
, partial
);
3049 long wxListCtrl::FindItem( long start
, long data
)
3051 return m_mainWin
->FindItem( start
, data
);
3054 long wxListCtrl::FindItem( long WXUNUSED(start
), const wxPoint
& WXUNUSED(pt
),
3055 int WXUNUSED(direction
))
3060 long wxListCtrl::HitTest( const wxPoint
&point
, int &flags
)
3062 return m_mainWin
->HitTest( (int)point
.x
, (int)point
.y
, flags
);
3065 long wxListCtrl::InsertItem( wxListItem
& info
)
3067 m_mainWin
->InsertItem( info
);
3068 return info
.m_itemId
;
3071 long wxListCtrl::InsertItem( long index
, const wxString
&label
)
3074 info
.m_text
= label
;
3075 info
.m_mask
= wxLIST_MASK_TEXT
;
3076 info
.m_itemId
= index
;
3077 return InsertItem( info
);
3080 long wxListCtrl::InsertItem( long index
, int imageIndex
)
3083 info
.m_mask
= wxLIST_MASK_IMAGE
;
3084 info
.m_image
= imageIndex
;
3085 info
.m_itemId
= index
;
3086 return InsertItem( info
);
3089 long wxListCtrl::InsertItem( long index
, const wxString
&label
, int imageIndex
)
3092 info
.m_text
= label
;
3093 info
.m_image
= imageIndex
;
3094 info
.m_mask
= wxLIST_MASK_TEXT
| wxLIST_MASK_IMAGE
;
3095 info
.m_itemId
= index
;
3096 return InsertItem( info
);
3099 long wxListCtrl::InsertColumn( long col
, wxListItem
&item
)
3101 wxASSERT( m_headerWin
);
3102 m_mainWin
->InsertColumn( col
, item
);
3103 m_headerWin
->Refresh();
3108 long wxListCtrl::InsertColumn( long col
, const wxString
&heading
,
3109 int format
, int width
)
3112 item
.m_mask
= wxLIST_MASK_TEXT
| wxLIST_MASK_FORMAT
;
3113 item
.m_text
= heading
;
3116 item
.m_mask
|= wxLIST_MASK_WIDTH
;
3117 item
.m_width
= width
;
3119 item
.m_format
= format
;
3121 return InsertColumn( col
, item
);
3124 bool wxListCtrl::ScrollList( int WXUNUSED(dx
), int WXUNUSED(dy
) )
3130 // fn is a function which takes 3 long arguments: item1, item2, data.
3131 // item1 is the long data associated with a first item (NOT the index).
3132 // item2 is the long data associated with a second item (NOT the index).
3133 // data is the same value as passed to SortItems.
3134 // The return value is a negative number if the first item should precede the second
3135 // item, a positive number of the second item should precede the first,
3136 // or zero if the two items are equivalent.
3137 // data is arbitrary data to be passed to the sort function.
3139 bool wxListCtrl::SortItems( wxListCtrlCompare fn
, long data
)
3141 m_mainWin
->SortItems( fn
, data
);
3145 void wxListCtrl::OnIdle( wxIdleEvent
&WXUNUSED(event
) )
3147 if (!m_mainWin
->m_dirty
) return;
3151 GetClientSize( &cw
, &ch
);
3158 if (HasFlag(wxLC_REPORT
) && !HasFlag(wxLC_NO_HEADER
))
3160 m_headerWin
->GetPosition( &x
, &y
);
3161 m_headerWin
->GetSize( &w
, &h
);
3162 if ((x
!= 0) || (y
!= 0) || (w
!= cw
) || (h
!= 23))
3163 m_headerWin
->SetSize( 0, 0, cw
, 23 );
3165 m_mainWin
->GetPosition( &x
, &y
);
3166 m_mainWin
->GetSize( &w
, &h
);
3167 if ((x
!= 0) || (y
!= 24) || (w
!= cw
) || (h
!= ch
-24))
3168 m_mainWin
->SetSize( 0, 24, cw
, ch
-24 );
3172 m_mainWin
->GetPosition( &x
, &y
);
3173 m_mainWin
->GetSize( &w
, &h
);
3174 if ((x
!= 0) || (y
!= 24) || (w
!= cw
) || (h
!= ch
))
3175 m_mainWin
->SetSize( 0, 0, cw
, ch
);
3178 m_mainWin
->CalculatePositions();
3179 m_mainWin
->RealizeChanges();
3180 m_mainWin
->m_dirty
= FALSE
;
3181 m_mainWin
->Refresh();
3184 bool wxListCtrl::SetBackgroundColour( const wxColour
&colour
)
3186 if ( !wxWindow::SetBackgroundColour( colour
) )
3191 m_mainWin
->SetBackgroundColour( colour
);
3192 m_mainWin
->m_dirty
= TRUE
;
3197 // m_headerWin->SetBackgroundColour( colour );
3203 bool wxListCtrl::SetForegroundColour( const wxColour
&colour
)
3205 if ( !wxWindow::SetForegroundColour( colour
) )
3210 m_mainWin
->SetForegroundColour( colour
);
3211 m_mainWin
->m_dirty
= TRUE
;
3216 m_headerWin
->SetForegroundColour( colour
);
3222 bool wxListCtrl::SetFont( const wxFont
&font
)
3224 if ( !wxWindow::SetFont( font
) )
3229 m_mainWin
->SetFont( font
);
3230 m_mainWin
->m_dirty
= TRUE
;
3235 m_headerWin
->SetFont( font
);