1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: wxGrid and related classes
4 // Author: Michael Bedward (based on code by Julian Smart, Robin Dunn)
8 // Copyright: (c) Michael Bedward (mbedward@ozemail.com.au)
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
13 #pragma implementation "grid.h"
16 // For compilers that support precompilation, includes "wx/wx.h".
17 #include "wx/wxprec.h"
25 #if !defined(wxUSE_NEW_GRID) || !(wxUSE_NEW_GRID)
31 #include "wx/dcclient.h"
32 #include "wx/settings.h"
37 #include "wx/generic/grid.h"
39 //////////////////////////////////////////////////////////////////////
41 wxGridCellCoords
wxGridNoCellCoords( -1, -1 );
42 wxRect
wxGridNoCellRect( -1, -1, -1, -1 );
44 // this is a magic incantation which must be done!
45 #include "wx/arrimpl.cpp"
47 WX_DEFINE_OBJARRAY(wxGridCellCoordsArray
)
50 // TODO: fixed so far - make configurable later (and also different for x/y)
51 static const size_t GRID_SCROLL_LINE
= 10;
53 //////////////////////////////////////////////////////////////////////
55 // Abstract base class for grid data (the model)
57 IMPLEMENT_ABSTRACT_CLASS( wxGridTableBase
, wxObject
)
60 wxGridTableBase::wxGridTableBase()
63 m_view
= (wxGrid
*) NULL
;
66 wxGridTableBase::~wxGridTableBase()
71 bool wxGridTableBase::InsertRows( size_t pos
, size_t numRows
)
73 wxLogWarning( wxT("Called grid table class function InsertRows(pos=%d, N=%d)\n"
74 "but your derived table class does not override this function"),
80 bool wxGridTableBase::AppendRows( size_t numRows
)
82 wxLogWarning( wxT("Called grid table class function AppendRows(N=%d)\n"
83 "but your derived table class does not override this function"),
89 bool wxGridTableBase::DeleteRows( size_t pos
, size_t numRows
)
91 wxLogWarning( wxT("Called grid table class function DeleteRows(pos=%d, N=%d)\n"
92 "but your derived table class does not override this function"),
98 bool wxGridTableBase::InsertCols( size_t pos
, size_t numCols
)
100 wxLogWarning( wxT("Called grid table class function InsertCols(pos=%d, N=%d)\n"
101 "but your derived table class does not override this function"),
107 bool wxGridTableBase::AppendCols( size_t numCols
)
109 wxLogWarning( wxT("Called grid table class function AppendCols(N=%d)\n"
110 "but your derived table class does not override this function"),
116 bool wxGridTableBase::DeleteCols( size_t pos
, size_t numCols
)
118 wxLogWarning( wxT("Called grid table class function DeleteCols(pos=%d, N=%d)\n"
119 "but your derived table class does not override this function"),
126 wxString
wxGridTableBase::GetRowLabelValue( int row
)
133 wxString
wxGridTableBase::GetColLabelValue( int col
)
135 // default col labels are:
136 // cols 0 to 25 : A-Z
137 // cols 26 to 675 : AA-ZZ
144 s
+= (_T('A') + (wxChar
)( col%26
));
146 if ( col
< 0 ) break;
149 // reverse the string...
151 for ( i
= 0; i
< n
; i
++ )
161 //////////////////////////////////////////////////////////////////////
163 // Message class for the grid table to send requests and notifications
167 wxGridTableMessage::wxGridTableMessage()
169 m_table
= (wxGridTableBase
*) NULL
;
175 wxGridTableMessage::wxGridTableMessage( wxGridTableBase
*table
, int id
,
176 int commandInt1
, int commandInt2
)
180 m_comInt1
= commandInt1
;
181 m_comInt2
= commandInt2
;
186 //////////////////////////////////////////////////////////////////////
188 // A basic grid table for string data. An object of this class will
189 // created by wxGrid if you don't specify an alternative table class.
192 WX_DEFINE_OBJARRAY(wxGridStringArray
)
194 IMPLEMENT_DYNAMIC_CLASS( wxGridStringTable
, wxGridTableBase
)
196 wxGridStringTable::wxGridStringTable()
201 wxGridStringTable::wxGridStringTable( int numRows
, int numCols
)
206 m_data
.Alloc( numRows
);
210 for ( col
= 0; col
< numCols
; col
++ )
212 sa
.Add( wxEmptyString
);
215 for ( row
= 0; row
< numRows
; row
++ )
221 wxGridStringTable::~wxGridStringTable()
225 long wxGridStringTable::GetNumberRows()
227 return m_data
.GetCount();
230 long wxGridStringTable::GetNumberCols()
232 if ( m_data
.GetCount() > 0 )
233 return m_data
[0].GetCount();
238 wxString
wxGridStringTable::GetValue( int row
, int col
)
240 // TODO: bounds checking
242 return m_data
[row
][col
];
245 void wxGridStringTable::SetValue( int row
, int col
, const wxString
& s
)
247 // TODO: bounds checking
249 m_data
[row
][col
] = s
;
252 bool wxGridStringTable::IsEmptyCell( int row
, int col
)
254 // TODO: bounds checking
256 return (m_data
[row
][col
] == wxEmptyString
);
260 void wxGridStringTable::Clear()
263 int numRows
, numCols
;
265 numRows
= m_data
.GetCount();
268 numCols
= m_data
[0].GetCount();
270 for ( row
= 0; row
< numRows
; row
++ )
272 for ( col
= 0; col
< numCols
; col
++ )
274 m_data
[row
][col
] = wxEmptyString
;
281 bool wxGridStringTable::InsertRows( size_t pos
, size_t numRows
)
285 size_t curNumRows
= m_data
.GetCount();
286 size_t curNumCols
= ( curNumRows
> 0 ? m_data
[0].GetCount() : 0 );
288 if ( pos
>= curNumRows
)
290 return AppendRows( numRows
);
294 sa
.Alloc( curNumCols
);
295 for ( col
= 0; col
< curNumCols
; col
++ )
297 sa
.Add( wxEmptyString
);
300 for ( row
= pos
; row
< pos
+ numRows
; row
++ )
302 m_data
.Insert( sa
, row
);
307 wxGridTableMessage
msg( this,
308 wxGRIDTABLE_NOTIFY_ROWS_INSERTED
,
312 GetView()->ProcessTableMessage( msg
);
318 bool wxGridStringTable::AppendRows( size_t numRows
)
322 size_t curNumRows
= m_data
.GetCount();
323 size_t curNumCols
= ( curNumRows
> 0 ? m_data
[0].GetCount() : 0 );
326 if ( curNumCols
> 0 )
328 sa
.Alloc( curNumCols
);
329 for ( col
= 0; col
< curNumCols
; col
++ )
331 sa
.Add( wxEmptyString
);
335 for ( row
= 0; row
< numRows
; row
++ )
342 wxGridTableMessage
msg( this,
343 wxGRIDTABLE_NOTIFY_ROWS_APPENDED
,
346 GetView()->ProcessTableMessage( msg
);
352 bool wxGridStringTable::DeleteRows( size_t pos
, size_t numRows
)
356 size_t curNumRows
= m_data
.GetCount();
358 if ( pos
>= curNumRows
)
360 wxLogError( wxT("Called wxGridStringTable::DeleteRows(pos=%d, N=%d)...\n"
361 "Pos value is invalid for present table with %d rows"),
362 pos
, numRows
, curNumRows
);
366 if ( numRows
> curNumRows
- pos
)
368 numRows
= curNumRows
- pos
;
371 if ( numRows
>= curNumRows
)
373 m_data
.Empty(); // don't release memory just yet
377 for ( n
= 0; n
< numRows
; n
++ )
379 m_data
.Remove( pos
);
385 wxGridTableMessage
msg( this,
386 wxGRIDTABLE_NOTIFY_ROWS_DELETED
,
390 GetView()->ProcessTableMessage( msg
);
396 bool wxGridStringTable::InsertCols( size_t pos
, size_t numCols
)
400 size_t curNumRows
= m_data
.GetCount();
401 size_t curNumCols
= ( curNumRows
> 0 ? m_data
[0].GetCount() : 0 );
403 if ( pos
>= curNumCols
)
405 return AppendCols( numCols
);
408 for ( row
= 0; row
< curNumRows
; row
++ )
410 for ( col
= pos
; col
< pos
+ numCols
; col
++ )
412 m_data
[row
].Insert( wxEmptyString
, col
);
418 wxGridTableMessage
msg( this,
419 wxGRIDTABLE_NOTIFY_COLS_INSERTED
,
423 GetView()->ProcessTableMessage( msg
);
429 bool wxGridStringTable::AppendCols( size_t numCols
)
433 size_t curNumRows
= m_data
.GetCount();
436 // TODO: something better than this ?
438 wxLogError( wxT("Unable to append cols to a grid table with no rows.\n"
439 "Call AppendRows() first") );
443 for ( row
= 0; row
< curNumRows
; row
++ )
445 for ( n
= 0; n
< numCols
; n
++ )
447 m_data
[row
].Add( wxEmptyString
);
453 wxGridTableMessage
msg( this,
454 wxGRIDTABLE_NOTIFY_COLS_APPENDED
,
457 GetView()->ProcessTableMessage( msg
);
463 bool wxGridStringTable::DeleteCols( size_t pos
, size_t numCols
)
467 size_t curNumRows
= m_data
.GetCount();
468 size_t curNumCols
= ( curNumRows
> 0 ? m_data
[0].GetCount() : 0 );
470 if ( pos
>= curNumCols
)
472 wxLogError( wxT("Called wxGridStringTable::DeleteCols(pos=%d, N=%d)...\n"
473 "Pos value is invalid for present table with %d cols"),
474 pos
, numCols
, curNumCols
);
478 if ( numCols
> curNumCols
- pos
)
480 numCols
= curNumCols
- pos
;
483 for ( row
= 0; row
< curNumRows
; row
++ )
485 if ( numCols
>= curNumCols
)
491 for ( n
= 0; n
< numCols
; n
++ )
493 m_data
[row
].Remove( pos
);
500 wxGridTableMessage
msg( this,
501 wxGRIDTABLE_NOTIFY_COLS_DELETED
,
505 GetView()->ProcessTableMessage( msg
);
511 wxString
wxGridStringTable::GetRowLabelValue( int row
)
513 if ( row
> (int)(m_rowLabels
.GetCount()) - 1 )
515 // using default label
517 return wxGridTableBase::GetRowLabelValue( row
);
521 return m_rowLabels
[ row
];
525 wxString
wxGridStringTable::GetColLabelValue( int col
)
527 if ( col
> (int)(m_colLabels
.GetCount()) - 1 )
529 // using default label
531 return wxGridTableBase::GetColLabelValue( col
);
535 return m_colLabels
[ col
];
539 void wxGridStringTable::SetRowLabelValue( int row
, const wxString
& value
)
541 if ( row
> (int)(m_rowLabels
.GetCount()) - 1 )
543 int n
= m_rowLabels
.GetCount();
545 for ( i
= n
; i
<= row
; i
++ )
547 m_rowLabels
.Add( wxGridTableBase::GetRowLabelValue(i
) );
551 m_rowLabels
[row
] = value
;
554 void wxGridStringTable::SetColLabelValue( int col
, const wxString
& value
)
556 if ( col
> (int)(m_colLabels
.GetCount()) - 1 )
558 int n
= m_colLabels
.GetCount();
560 for ( i
= n
; i
<= col
; i
++ )
562 m_colLabels
.Add( wxGridTableBase::GetColLabelValue(i
) );
566 m_colLabels
[col
] = value
;
572 //////////////////////////////////////////////////////////////////////
574 IMPLEMENT_DYNAMIC_CLASS( wxGridTextCtrl
, wxTextCtrl
)
576 BEGIN_EVENT_TABLE( wxGridTextCtrl
, wxTextCtrl
)
577 EVT_KEY_DOWN( wxGridTextCtrl::OnKeyDown
)
581 wxGridTextCtrl::wxGridTextCtrl( wxWindow
*par
,
585 const wxString
& value
,
589 : wxTextCtrl( par
, id
, value
, pos
, size
, style
)
592 m_isCellControl
= isCellControl
;
596 void wxGridTextCtrl::OnKeyDown( wxKeyEvent
& event
)
598 switch ( event
.KeyCode() )
601 m_grid
->SetEditControlValue( startValue
);
602 SetInsertionPointEnd();
611 if ( m_isCellControl
)
613 // send the event to the parent grid, skipping the
614 // event if nothing happens
616 event
.Skip( m_grid
->ProcessEvent( event
) );
620 // default text control response within the top edit
628 if ( m_isCellControl
)
630 if ( !m_grid
->ProcessEvent( event
) )
633 // wxMotif needs a little extra help...
635 int pos
= GetInsertionPoint();
636 wxString
s( GetValue() );
637 s
= s
.Left(pos
) + "\n" + s
.Mid(pos
);
639 SetInsertionPoint( pos
);
641 // the other ports can handle a Return key press
651 if ( m_isCellControl
)
653 // send the event to the parent grid, skipping the
654 // event if nothing happens
656 event
.Skip( m_grid
->ProcessEvent( event
) );
660 // default text control response within the top edit
672 void wxGridTextCtrl::SetStartValue( const wxString
& s
)
675 wxTextCtrl::SetValue(s
);
680 //////////////////////////////////////////////////////////////////////
682 IMPLEMENT_DYNAMIC_CLASS( wxGridRowLabelWindow
, wxWindow
)
684 BEGIN_EVENT_TABLE( wxGridRowLabelWindow
, wxWindow
)
685 EVT_PAINT( wxGridRowLabelWindow::OnPaint
)
686 EVT_MOUSE_EVENTS( wxGridRowLabelWindow::OnMouseEvent
)
687 EVT_KEY_DOWN( wxGridRowLabelWindow::OnKeyDown
)
690 wxGridRowLabelWindow::wxGridRowLabelWindow( wxGrid
*parent
,
692 const wxPoint
&pos
, const wxSize
&size
)
693 : wxWindow( parent
, id
, pos
, size
)
698 void wxGridRowLabelWindow::OnPaint( wxPaintEvent
&event
)
702 // NO - don't do this because it will set both the x and y origin
703 // coords to match the parent scrolled window and we just want to
704 // set the y coord - MB
706 // m_owner->PrepareDC( dc );
709 m_owner
->CalcUnscrolledPosition( 0, 0, &x
, &y
);
710 dc
.SetDeviceOrigin( 0, -y
);
712 m_owner
->CalcRowLabelsExposed( GetUpdateRegion() );
713 m_owner
->DrawRowLabels( dc
);
717 void wxGridRowLabelWindow::OnMouseEvent( wxMouseEvent
& event
)
719 m_owner
->ProcessRowLabelMouseEvent( event
);
723 // This seems to be required for wxMotif otherwise the mouse
724 // cursor must be in the cell edit control to get key events
726 void wxGridRowLabelWindow::OnKeyDown( wxKeyEvent
& event
)
728 if ( !m_owner
->ProcessEvent( event
) ) event
.Skip();
733 //////////////////////////////////////////////////////////////////////
735 IMPLEMENT_DYNAMIC_CLASS( wxGridColLabelWindow
, wxWindow
)
737 BEGIN_EVENT_TABLE( wxGridColLabelWindow
, wxWindow
)
738 EVT_PAINT( wxGridColLabelWindow::OnPaint
)
739 EVT_MOUSE_EVENTS( wxGridColLabelWindow::OnMouseEvent
)
740 EVT_KEY_DOWN( wxGridColLabelWindow::OnKeyDown
)
743 wxGridColLabelWindow::wxGridColLabelWindow( wxGrid
*parent
,
745 const wxPoint
&pos
, const wxSize
&size
)
746 : wxWindow( parent
, id
, pos
, size
)
751 void wxGridColLabelWindow::OnPaint( wxPaintEvent
&event
)
755 // NO - don't do this because it will set both the x and y origin
756 // coords to match the parent scrolled window and we just want to
757 // set the x coord - MB
759 // m_owner->PrepareDC( dc );
762 m_owner
->CalcUnscrolledPosition( 0, 0, &x
, &y
);
763 dc
.SetDeviceOrigin( -x
, 0 );
765 m_owner
->CalcColLabelsExposed( GetUpdateRegion() );
766 m_owner
->DrawColLabels( dc
);
770 void wxGridColLabelWindow::OnMouseEvent( wxMouseEvent
& event
)
772 m_owner
->ProcessColLabelMouseEvent( event
);
776 // This seems to be required for wxMotif otherwise the mouse
777 // cursor must be in the cell edit control to get key events
779 void wxGridColLabelWindow::OnKeyDown( wxKeyEvent
& event
)
781 if ( !m_owner
->ProcessEvent( event
) ) event
.Skip();
786 //////////////////////////////////////////////////////////////////////
788 IMPLEMENT_DYNAMIC_CLASS( wxGridCornerLabelWindow
, wxWindow
)
790 BEGIN_EVENT_TABLE( wxGridCornerLabelWindow
, wxWindow
)
791 EVT_MOUSE_EVENTS( wxGridCornerLabelWindow::OnMouseEvent
)
792 EVT_PAINT( wxGridCornerLabelWindow::OnPaint
)
793 EVT_KEY_DOWN( wxGridCornerLabelWindow::OnKeyDown
)
796 wxGridCornerLabelWindow::wxGridCornerLabelWindow( wxGrid
*parent
,
798 const wxPoint
&pos
, const wxSize
&size
)
799 : wxWindow( parent
, id
, pos
, size
)
804 void wxGridCornerLabelWindow::OnPaint( wxPaintEvent
& WXUNUSED(event
) )
808 int client_height
= 0;
809 int client_width
= 0;
810 GetClientSize( &client_width
, &client_height
);
812 dc
.SetPen( *wxBLACK_PEN
);
813 dc
.DrawLine( client_width
-1, client_height
-1, client_width
-1, 0 );
814 dc
.DrawLine( client_width
-1, client_height
-1, 0, client_height
-1 );
816 dc
.SetPen( *wxWHITE_PEN
);
817 dc
.DrawLine( 0, 0, client_width
, 0 );
818 dc
.DrawLine( 0, 0, 0, client_height
);
822 void wxGridCornerLabelWindow::OnMouseEvent( wxMouseEvent
& event
)
824 m_owner
->ProcessCornerLabelMouseEvent( event
);
828 // This seems to be required for wxMotif otherwise the mouse
829 // cursor must be in the cell edit control to get key events
831 void wxGridCornerLabelWindow::OnKeyDown( wxKeyEvent
& event
)
833 if ( !m_owner
->ProcessEvent( event
) ) event
.Skip();
838 //////////////////////////////////////////////////////////////////////
840 IMPLEMENT_DYNAMIC_CLASS( wxGridWindow
, wxPanel
)
842 BEGIN_EVENT_TABLE( wxGridWindow
, wxPanel
)
843 EVT_PAINT( wxGridWindow::OnPaint
)
844 EVT_MOUSE_EVENTS( wxGridWindow::OnMouseEvent
)
845 EVT_KEY_DOWN( wxGridWindow::OnKeyDown
)
848 wxGridWindow::wxGridWindow( wxGrid
*parent
,
849 wxGridRowLabelWindow
*rowLblWin
,
850 wxGridColLabelWindow
*colLblWin
,
851 wxWindowID id
, const wxPoint
&pos
, const wxSize
&size
)
852 : wxPanel( parent
, id
, pos
, size
, wxSUNKEN_BORDER
, "grid window" )
855 m_rowLabelWin
= rowLblWin
;
856 m_colLabelWin
= colLblWin
;
858 SetBackgroundColour( "WHITE" );
862 wxGridWindow::~wxGridWindow()
867 void wxGridWindow::OnPaint( wxPaintEvent
&WXUNUSED(event
) )
869 wxPaintDC
dc( this );
870 m_owner
->PrepareDC( dc
);
872 m_owner
->CalcCellsExposed( GetUpdateRegion() );
873 m_owner
->DrawGridCellArea( dc
);
877 void wxGridWindow::ScrollWindow( int dx
, int dy
, const wxRect
*rect
)
879 wxPanel::ScrollWindow( dx
, dy
, rect
);
880 m_rowLabelWin
->ScrollWindow( 0, dy
, rect
);
881 m_colLabelWin
->ScrollWindow( dx
, 0, rect
);
885 void wxGridWindow::OnMouseEvent( wxMouseEvent
& event
)
887 m_owner
->ProcessGridCellMouseEvent( event
);
891 // This seems to be required for wxMotif otherwise the mouse
892 // cursor must be in the cell edit control to get key events
894 void wxGridWindow::OnKeyDown( wxKeyEvent
& event
)
896 if ( !m_owner
->ProcessEvent( event
) ) event
.Skip();
901 //////////////////////////////////////////////////////////////////////
903 IMPLEMENT_DYNAMIC_CLASS( wxGrid
, wxScrolledWindow
)
905 BEGIN_EVENT_TABLE( wxGrid
, wxScrolledWindow
)
906 EVT_PAINT( wxGrid::OnPaint
)
907 EVT_SIZE( wxGrid::OnSize
)
908 EVT_KEY_DOWN( wxGrid::OnKeyDown
)
911 wxGrid::wxGrid( wxWindow
*parent
,
916 const wxString
& name
)
917 : wxScrolledWindow( parent
, id
, pos
, size
, style
, name
)
930 // ----- internal init and update functions
933 void wxGrid::Create()
935 int colLblH
= WXGRID_DEFAULT_COL_LABEL_HEIGHT
;
936 int rowLblW
= WXGRID_DEFAULT_ROW_LABEL_WIDTH
;
938 m_rowLabelWin
= new wxGridRowLabelWindow( this,
941 wxSize(rowLblW
,-1) );
943 m_colLabelWin
= new wxGridColLabelWindow( this,
946 wxSize(-1, colLblH
) );
948 m_cornerLabelWin
= new wxGridCornerLabelWindow( this,
951 wxSize(rowLblW
, colLblH
) );
953 m_gridWin
= new wxGridWindow( this,
960 SetTargetWindow( m_gridWin
);
962 m_mainSizer
= new wxBoxSizer( wxVERTICAL
);
964 m_topSizer
= new wxBoxSizer( wxHORIZONTAL
);
965 m_topSizer
->Add( m_cornerLabelWin
, 0 );
966 m_topSizer
->Add( m_colLabelWin
, 1 );
968 m_mainSizer
->Add( m_topSizer
, 0, wxEXPAND
);
970 m_middleSizer
= new wxBoxSizer( wxHORIZONTAL
);
971 m_middleSizer
->Add( m_rowLabelWin
, 0, wxEXPAND
);
972 m_middleSizer
->Add( m_gridWin
, 1, wxEXPAND
);
974 m_mainSizer
->Add( m_middleSizer
, 1, wxEXPAND
);
976 SetAutoLayout( TRUE
);
977 SetSizer( m_mainSizer
);
981 bool wxGrid::CreateGrid( int numRows
, int numCols
)
985 wxLogError( wxT("wxGrid::CreateGrid(numRows, numCols) called more than once") );
993 m_table
= new wxGridStringTable( m_numRows
, m_numCols
);
994 m_table
->SetView( this );
1007 if ( m_numRows
<= 0 )
1008 m_numRows
= WXGRID_DEFAULT_NUMBER_ROWS
;
1010 if ( m_numCols
<= 0 )
1011 m_numCols
= WXGRID_DEFAULT_NUMBER_COLS
;
1013 m_rowLabelWidth
= WXGRID_DEFAULT_ROW_LABEL_WIDTH
;
1014 m_colLabelHeight
= WXGRID_DEFAULT_COL_LABEL_HEIGHT
;
1016 if ( m_rowLabelWin
)
1018 m_labelBackgroundColour
= m_rowLabelWin
->GetBackgroundColour();
1022 m_labelBackgroundColour
= wxColour( _T("WHITE") );
1025 m_labelTextColour
= wxColour( _T("BLACK") );
1027 // TODO: something better than this ?
1029 m_labelFont
= this->GetFont();
1030 m_labelFont
.SetWeight( m_labelFont
.GetWeight() + 2 );
1032 m_rowLabelHorizAlign
= wxLEFT
;
1033 m_rowLabelVertAlign
= wxCENTRE
;
1035 m_colLabelHorizAlign
= wxCENTRE
;
1036 m_colLabelVertAlign
= wxTOP
;
1038 m_defaultColWidth
= WXGRID_DEFAULT_COL_WIDTH
;
1039 m_defaultRowHeight
= m_gridWin
->GetCharHeight();
1041 #if defined(__WXMOTIF__) || defined(__WXGTK__) // see also text ctrl sizing in ShowCellEditControl()
1042 m_defaultRowHeight
+= 8;
1044 m_defaultRowHeight
+= 4;
1047 m_rowHeights
.Alloc( m_numRows
);
1048 m_rowBottoms
.Alloc( m_numRows
);
1050 for ( i
= 0; i
< m_numRows
; i
++ )
1052 m_rowHeights
.Add( m_defaultRowHeight
);
1053 rowBottom
+= m_defaultRowHeight
;
1054 m_rowBottoms
.Add( rowBottom
);
1057 m_colWidths
.Alloc( m_numCols
);
1058 m_colRights
.Alloc( m_numCols
);
1060 for ( i
= 0; i
< m_numCols
; i
++ )
1062 m_colWidths
.Add( m_defaultColWidth
);
1063 colRight
+= m_defaultColWidth
;
1064 m_colRights
.Add( colRight
);
1067 // TODO: improve this ?
1069 m_defaultCellFont
= this->GetFont();
1071 m_gridLineColour
= wxColour( 128, 128, 255 );
1072 m_gridLinesEnabled
= TRUE
;
1074 m_cursorMode
= WXGRID_CURSOR_SELECT_CELL
;
1076 m_dragRowOrCol
= -1;
1077 m_isDragging
= FALSE
;
1079 m_rowResizeCursor
= wxCursor( wxCURSOR_SIZENS
);
1080 m_colResizeCursor
= wxCursor( wxCURSOR_SIZEWE
);
1082 m_currentCellCoords
= wxGridNoCellCoords
;
1084 m_selectedTopLeft
= wxGridNoCellCoords
;
1085 m_selectedBottomRight
= wxGridNoCellCoords
;
1087 m_editable
= TRUE
; // default for whole grid
1089 m_inOnKeyDown
= FALSE
;
1092 // TODO: extend this to other types of controls
1094 m_cellEditCtrl
= new wxGridTextCtrl( m_gridWin
,
1101 #if defined(__WXMSW__)
1102 , wxTE_MULTILINE
| wxTE_NO_VSCROLL
1106 m_cellEditCtrl
->Show( FALSE
);
1107 m_cellEditCtrlEnabled
= TRUE
;
1108 m_editCtrlType
= wxGRID_TEXTCTRL
;
1112 void wxGrid::CalcDimensions()
1115 GetClientSize( &cw
, &ch
);
1117 if ( m_numRows
> 0 && m_numCols
> 0 )
1119 int right
= m_colRights
[ m_numCols
-1 ] + 20;
1120 int bottom
= m_rowBottoms
[ m_numRows
-1 ] + 20;
1122 // TODO: restore the scroll position that we had before sizing
1125 GetViewStart( &x
, &y
);
1126 SetScrollbars( GRID_SCROLL_LINE
, GRID_SCROLL_LINE
,
1127 right
/GRID_SCROLL_LINE
, bottom
/GRID_SCROLL_LINE
,
1133 // this is called when the grid table sends a message to say that it
1134 // has been redimensioned
1136 bool wxGrid::Redimension( wxGridTableMessage
& msg
)
1140 switch ( msg
.GetId() )
1142 case wxGRIDTABLE_NOTIFY_ROWS_INSERTED
:
1144 size_t pos
= msg
.GetCommandInt();
1145 int numRows
= msg
.GetCommandInt2();
1146 for ( i
= 0; i
< numRows
; i
++ )
1148 m_rowHeights
.Insert( m_defaultRowHeight
, pos
);
1149 m_rowBottoms
.Insert( 0, pos
);
1151 m_numRows
+= numRows
;
1154 if ( pos
> 0 ) bottom
= m_rowBottoms
[pos
-1];
1156 for ( i
= pos
; i
< m_numRows
; i
++ )
1158 bottom
+= m_rowHeights
[i
];
1159 m_rowBottoms
[i
] = bottom
;
1165 case wxGRIDTABLE_NOTIFY_ROWS_APPENDED
:
1167 int numRows
= msg
.GetCommandInt();
1168 for ( i
= 0; i
< numRows
; i
++ )
1170 m_rowHeights
.Add( m_defaultRowHeight
);
1171 m_rowBottoms
.Add( 0 );
1174 int oldNumRows
= m_numRows
;
1175 m_numRows
+= numRows
;
1178 if ( oldNumRows
> 0 ) bottom
= m_rowBottoms
[oldNumRows
-1];
1180 for ( i
= oldNumRows
; i
< m_numRows
; i
++ )
1182 bottom
+= m_rowHeights
[i
];
1183 m_rowBottoms
[i
] = bottom
;
1189 case wxGRIDTABLE_NOTIFY_ROWS_DELETED
:
1191 size_t pos
= msg
.GetCommandInt();
1192 int numRows
= msg
.GetCommandInt2();
1193 for ( i
= 0; i
< numRows
; i
++ )
1195 m_rowHeights
.Remove( pos
);
1196 m_rowBottoms
.Remove( pos
);
1198 m_numRows
-= numRows
;
1203 m_colWidths
.Clear();
1204 m_colRights
.Clear();
1205 m_currentCellCoords
= wxGridNoCellCoords
;
1209 if ( m_currentCellCoords
.GetRow() >= m_numRows
)
1210 m_currentCellCoords
.Set( 0, 0 );
1213 for ( i
= 0; i
< m_numRows
; i
++ )
1215 h
+= m_rowHeights
[i
];
1216 m_rowBottoms
[i
] = h
;
1224 case wxGRIDTABLE_NOTIFY_COLS_INSERTED
:
1226 size_t pos
= msg
.GetCommandInt();
1227 int numCols
= msg
.GetCommandInt2();
1228 for ( i
= 0; i
< numCols
; i
++ )
1230 m_colWidths
.Insert( m_defaultColWidth
, pos
);
1231 m_colRights
.Insert( 0, pos
);
1233 m_numCols
+= numCols
;
1236 if ( pos
> 0 ) right
= m_colRights
[pos
-1];
1238 for ( i
= pos
; i
< m_numCols
; i
++ )
1240 right
+= m_colWidths
[i
];
1241 m_colRights
[i
] = right
;
1247 case wxGRIDTABLE_NOTIFY_COLS_APPENDED
:
1249 int numCols
= msg
.GetCommandInt();
1250 for ( i
= 0; i
< numCols
; i
++ )
1252 m_colWidths
.Add( m_defaultColWidth
);
1253 m_colRights
.Add( 0 );
1256 int oldNumCols
= m_numCols
;
1257 m_numCols
+= numCols
;
1260 if ( oldNumCols
> 0 ) right
= m_colRights
[oldNumCols
-1];
1262 for ( i
= oldNumCols
; i
< m_numCols
; i
++ )
1264 right
+= m_colWidths
[i
];
1265 m_colRights
[i
] = right
;
1271 case wxGRIDTABLE_NOTIFY_COLS_DELETED
:
1273 size_t pos
= msg
.GetCommandInt();
1274 int numCols
= msg
.GetCommandInt2();
1275 for ( i
= 0; i
< numCols
; i
++ )
1277 m_colWidths
.Remove( pos
);
1278 m_colRights
.Remove( pos
);
1280 m_numCols
-= numCols
;
1284 #if 0 // leave the row alone here so that AppendCols will work subsequently
1286 m_rowHeights
.Clear();
1287 m_rowBottoms
.Clear();
1289 m_currentCellCoords
= wxGridNoCellCoords
;
1293 if ( m_currentCellCoords
.GetCol() >= m_numCols
)
1294 m_currentCellCoords
.Set( 0, 0 );
1297 for ( i
= 0; i
< m_numCols
; i
++ )
1299 w
+= m_colWidths
[i
];
1312 void wxGrid::CalcRowLabelsExposed( wxRegion
& reg
)
1314 wxRegionIterator
iter( reg
);
1317 m_rowLabelsExposed
.Empty();
1324 // TODO: remove this when we can...
1325 // There is a bug in wxMotif that gives garbage update
1326 // rectangles if you jump-scroll a long way by clicking the
1327 // scrollbar with middle button. This is a work-around
1329 #if defined(__WXMOTIF__)
1331 m_gridWin
->GetClientSize( &cw
, &ch
);
1332 if ( r
.GetTop() > ch
) r
.SetTop( 0 );
1333 r
.SetBottom( wxMin( r
.GetBottom(), ch
) );
1336 // logical bounds of update region
1339 CalcUnscrolledPosition( 0, r
.GetTop(), &dummy
, &top
);
1340 CalcUnscrolledPosition( 0, r
.GetBottom(), &dummy
, &bottom
);
1342 // find the row labels within these bounds
1346 for ( row
= 0; row
< m_numRows
; row
++ )
1348 if ( m_rowBottoms
[row
] < top
) continue;
1350 rowTop
= m_rowBottoms
[row
] - m_rowHeights
[row
];
1351 if ( rowTop
> bottom
) break;
1353 m_rowLabelsExposed
.Add( row
);
1361 void wxGrid::CalcColLabelsExposed( wxRegion
& reg
)
1363 wxRegionIterator
iter( reg
);
1366 m_colLabelsExposed
.Empty();
1373 // TODO: remove this when we can...
1374 // There is a bug in wxMotif that gives garbage update
1375 // rectangles if you jump-scroll a long way by clicking the
1376 // scrollbar with middle button. This is a work-around
1378 #if defined(__WXMOTIF__)
1380 m_gridWin
->GetClientSize( &cw
, &ch
);
1381 if ( r
.GetLeft() > cw
) r
.SetLeft( 0 );
1382 r
.SetRight( wxMin( r
.GetRight(), cw
) );
1385 // logical bounds of update region
1388 CalcUnscrolledPosition( r
.GetLeft(), 0, &left
, &dummy
);
1389 CalcUnscrolledPosition( r
.GetRight(), 0, &right
, &dummy
);
1391 // find the cells within these bounds
1395 for ( col
= 0; col
< m_numCols
; col
++ )
1397 if ( m_colRights
[col
] < left
) continue;
1399 colLeft
= m_colRights
[col
] - m_colWidths
[col
];
1400 if ( colLeft
> right
) break;
1402 m_colLabelsExposed
.Add( col
);
1410 void wxGrid::CalcCellsExposed( wxRegion
& reg
)
1412 wxRegionIterator
iter( reg
);
1415 m_cellsExposed
.Empty();
1416 m_rowsExposed
.Empty();
1417 m_colsExposed
.Empty();
1419 int left
, top
, right
, bottom
;
1424 // TODO: remove this when we can...
1425 // There is a bug in wxMotif that gives garbage update
1426 // rectangles if you jump-scroll a long way by clicking the
1427 // scrollbar with middle button. This is a work-around
1429 #if defined(__WXMOTIF__)
1431 m_gridWin
->GetClientSize( &cw
, &ch
);
1432 if ( r
.GetTop() > ch
) r
.SetTop( 0 );
1433 if ( r
.GetLeft() > cw
) r
.SetLeft( 0 );
1434 r
.SetRight( wxMin( r
.GetRight(), cw
) );
1435 r
.SetBottom( wxMin( r
.GetBottom(), ch
) );
1438 // logical bounds of update region
1440 CalcUnscrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top
);
1441 CalcUnscrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom
);
1443 // find the cells within these bounds
1446 int colLeft
, rowTop
;
1447 for ( row
= 0; row
< m_numRows
; row
++ )
1449 if ( m_rowBottoms
[row
] < top
) continue;
1451 rowTop
= m_rowBottoms
[row
] - m_rowHeights
[row
];
1452 if ( rowTop
> bottom
) break;
1454 m_rowsExposed
.Add( row
);
1456 for ( col
= 0; col
< m_numCols
; col
++ )
1458 if ( m_colRights
[col
] < left
) continue;
1460 colLeft
= m_colRights
[col
] - m_colWidths
[col
];
1461 if ( colLeft
> right
) break;
1463 if ( m_colsExposed
.Index( col
) == wxNOT_FOUND
) m_colsExposed
.Add( col
);
1464 m_cellsExposed
.Add( wxGridCellCoords( row
, col
) );
1473 void wxGrid::ProcessRowLabelMouseEvent( wxMouseEvent
& event
)
1476 wxPoint
pos( event
.GetPosition() );
1477 CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y
);
1479 if ( event
.Dragging() )
1481 m_isDragging
= TRUE
;
1483 if ( event
.LeftIsDown() )
1485 switch( m_cursorMode
)
1487 case WXGRID_CURSOR_RESIZE_ROW
:
1489 int cw
, ch
, left
, dummy
;
1490 m_gridWin
->GetClientSize( &cw
, &ch
);
1491 CalcUnscrolledPosition( 0, 0, &left
, &dummy
);
1493 wxClientDC
dc( m_gridWin
);
1495 dc
.SetLogicalFunction(wxINVERT
);
1496 if ( m_dragLastPos
>= 0 )
1498 dc
.DrawLine( left
, m_dragLastPos
, left
+cw
, m_dragLastPos
);
1500 dc
.DrawLine( left
, y
, left
+cw
, y
);
1505 case WXGRID_CURSOR_SELECT_ROW
:
1507 if ( (row
= YToRow( y
)) >= 0 &&
1508 !IsInSelection( row
, 0 ) )
1510 SelectRow( row
, TRUE
);
1519 m_isDragging
= FALSE
;
1522 // ------------ Left button pressed
1524 if ( event
.LeftDown() )
1526 // don't send a label click event for a hit on the
1527 // edge of the row label - this is probably the user
1528 // wanting to resize the row
1530 if ( YToEdgeOfRow(y
) < 0 )
1534 !SendEvent( EVT_GRID_LABEL_LEFT_CLICK
, row
, -1, event
) )
1536 SelectRow( row
, event
.ShiftDown() );
1537 m_cursorMode
= WXGRID_CURSOR_SELECT_ROW
;
1542 // starting to drag-resize a row
1544 m_rowLabelWin
->CaptureMouse();
1549 // ------------ Left double click
1551 else if (event
.LeftDClick() )
1553 if ( YToEdgeOfRow(y
) < 0 )
1556 SendEvent( EVT_GRID_LABEL_LEFT_DCLICK
, row
, -1, event
);
1561 // ------------ Left button released
1563 else if ( event
.LeftUp() )
1565 if ( m_cursorMode
== WXGRID_CURSOR_RESIZE_ROW
)
1567 m_rowLabelWin
->ReleaseMouse();
1569 if ( m_dragLastPos
>= 0 )
1571 // erase the last line and resize the row
1573 int cw
, ch
, left
, dummy
;
1574 m_gridWin
->GetClientSize( &cw
, &ch
);
1575 CalcUnscrolledPosition( 0, 0, &left
, &dummy
);
1577 wxClientDC
dc( m_gridWin
);
1579 dc
.SetLogicalFunction( wxINVERT
);
1580 dc
.DrawLine( left
, m_dragLastPos
, left
+cw
, m_dragLastPos
);
1581 HideCellEditControl();
1583 int rowTop
= m_rowBottoms
[m_dragRowOrCol
] - m_rowHeights
[m_dragRowOrCol
];
1584 SetRowSize( m_dragRowOrCol
, wxMax( y
- rowTop
, WXGRID_MIN_ROW_HEIGHT
) );
1585 if ( !GetBatchCount() )
1587 // TODO: optimize this
1588 m_rowLabelWin
->Refresh();
1589 m_gridWin
->Refresh();
1592 ShowCellEditControl();
1594 // Note: we are ending the event *after* doing
1595 // default processing in this case
1597 SendEvent( EVT_GRID_ROW_SIZE
, m_dragRowOrCol
, -1, event
);
1605 // ------------ Right button down
1607 else if ( event
.RightDown() )
1610 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_CLICK
, row
, -1, event
) )
1612 // no default action at the moment
1617 // ------------ Right double click
1619 else if ( event
.RightDClick() )
1622 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_DCLICK
, row
, -1, event
) )
1624 // no default action at the moment
1629 // ------------ No buttons down and mouse moving
1631 else if ( event
.Moving() )
1633 m_dragRowOrCol
= YToEdgeOfRow( y
);
1634 if ( m_dragRowOrCol
>= 0 )
1636 if ( m_cursorMode
== WXGRID_CURSOR_SELECT_CELL
)
1638 m_cursorMode
= WXGRID_CURSOR_RESIZE_ROW
;
1639 m_rowLabelWin
->SetCursor( m_rowResizeCursor
);
1644 if ( m_cursorMode
!= WXGRID_CURSOR_SELECT_CELL
)
1646 m_cursorMode
= WXGRID_CURSOR_SELECT_CELL
;
1647 m_rowLabelWin
->SetCursor( *wxSTANDARD_CURSOR
);
1654 void wxGrid::ProcessColLabelMouseEvent( wxMouseEvent
& event
)
1657 wxPoint
pos( event
.GetPosition() );
1658 CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y
);
1660 if ( event
.Dragging() )
1662 m_isDragging
= TRUE
;
1664 if ( event
.LeftIsDown() )
1666 switch( m_cursorMode
)
1668 case WXGRID_CURSOR_RESIZE_COL
:
1670 int cw
, ch
, dummy
, top
;
1671 m_gridWin
->GetClientSize( &cw
, &ch
);
1672 CalcUnscrolledPosition( 0, 0, &dummy
, &top
);
1674 wxClientDC
dc( m_gridWin
);
1676 dc
.SetLogicalFunction(wxINVERT
);
1677 if ( m_dragLastPos
>= 0 )
1679 dc
.DrawLine( m_dragLastPos
, top
, m_dragLastPos
, top
+ch
);
1681 dc
.DrawLine( x
, top
, x
, top
+ch
);
1686 case WXGRID_CURSOR_SELECT_COL
:
1688 if ( (col
= XToCol( x
)) >= 0 &&
1689 !IsInSelection( 0, col
) )
1691 SelectCol( col
, TRUE
);
1700 m_isDragging
= FALSE
;
1703 // ------------ Left button pressed
1705 if ( event
.LeftDown() )
1707 // don't send a label click event for a hit on the
1708 // edge of the col label - this is probably the user
1709 // wanting to resize the col
1711 if ( XToEdgeOfCol(x
) < 0 )
1715 !SendEvent( EVT_GRID_LABEL_LEFT_CLICK
, -1, col
, event
) )
1717 SelectCol( col
, event
.ShiftDown() );
1718 m_cursorMode
= WXGRID_CURSOR_SELECT_COL
;
1723 // starting to drag-resize a col
1725 m_colLabelWin
->CaptureMouse();
1730 // ------------ Left double click
1732 if ( event
.LeftDClick() )
1734 if ( XToEdgeOfCol(x
) < 0 )
1737 SendEvent( EVT_GRID_LABEL_LEFT_DCLICK
, -1, col
, event
);
1742 // ------------ Left button released
1744 else if ( event
.LeftUp() )
1746 if ( m_cursorMode
== WXGRID_CURSOR_RESIZE_COL
)
1748 m_colLabelWin
->ReleaseMouse();
1750 if ( m_dragLastPos
>= 0 )
1752 // erase the last line and resize the col
1754 int cw
, ch
, dummy
, top
;
1755 m_gridWin
->GetClientSize( &cw
, &ch
);
1756 CalcUnscrolledPosition( 0, 0, &dummy
, &top
);
1758 wxClientDC
dc( m_gridWin
);
1760 dc
.SetLogicalFunction( wxINVERT
);
1761 dc
.DrawLine( m_dragLastPos
, top
, m_dragLastPos
, top
+ch
);
1762 HideCellEditControl();
1764 int colLeft
= m_colRights
[m_dragRowOrCol
] - m_colWidths
[m_dragRowOrCol
];
1765 SetColSize( m_dragRowOrCol
, wxMax( x
- colLeft
, WXGRID_MIN_COL_WIDTH
) );
1767 if ( !GetBatchCount() )
1769 // TODO: optimize this
1770 m_colLabelWin
->Refresh();
1771 m_gridWin
->Refresh();
1774 ShowCellEditControl();
1776 // Note: we are ending the event *after* doing
1777 // default processing in this case
1779 SendEvent( EVT_GRID_COL_SIZE
, -1, m_dragRowOrCol
, event
);
1787 // ------------ Right button down
1789 else if ( event
.RightDown() )
1792 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_CLICK
, -1, col
, event
) )
1794 // no default action at the moment
1799 // ------------ Right double click
1801 else if ( event
.RightDClick() )
1804 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_DCLICK
, -1, col
, event
) )
1806 // no default action at the moment
1811 // ------------ No buttons down and mouse moving
1813 else if ( event
.Moving() )
1815 m_dragRowOrCol
= XToEdgeOfCol( x
);
1816 if ( m_dragRowOrCol
>= 0 )
1818 if ( m_cursorMode
== WXGRID_CURSOR_SELECT_CELL
)
1820 m_cursorMode
= WXGRID_CURSOR_RESIZE_COL
;
1821 m_colLabelWin
->SetCursor( m_colResizeCursor
);
1826 if ( m_cursorMode
!= WXGRID_CURSOR_SELECT_CELL
)
1828 m_cursorMode
= WXGRID_CURSOR_SELECT_CELL
;
1829 m_colLabelWin
->SetCursor( *wxSTANDARD_CURSOR
);
1836 void wxGrid::ProcessCornerLabelMouseEvent( wxMouseEvent
& event
)
1838 if ( event
.LeftDown() )
1840 // indicate corner label by having both row and
1843 if ( !SendEvent( EVT_GRID_LABEL_LEFT_CLICK
, -1, -1, event
) )
1849 else if ( event
.LeftDClick() )
1851 SendEvent( EVT_GRID_LABEL_LEFT_DCLICK
, -1, -1, event
);
1854 else if ( event
.RightDown() )
1856 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_CLICK
, -1, -1, event
) )
1858 // no default action at the moment
1862 else if ( event
.RightDClick() )
1864 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_DCLICK
, -1, -1, event
) )
1866 // no default action at the moment
1872 void wxGrid::ProcessGridCellMouseEvent( wxMouseEvent
& event
)
1875 wxPoint
pos( event
.GetPosition() );
1876 CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y
);
1878 wxGridCellCoords coords
;
1879 XYToCell( x
, y
, coords
);
1881 if ( event
.Dragging() )
1883 m_isDragging
= TRUE
;
1884 if ( m_cursorMode
== WXGRID_CURSOR_SELECT_CELL
)
1886 // Hide the edit control, so it
1887 // won't interfer with drag-shrinking.
1888 if ( IsCellEditControlEnabled() )
1889 HideCellEditControl();
1890 if ( coords
!= wxGridNoCellCoords
)
1892 if ( !IsSelection() )
1894 SelectBlock( coords
, coords
);
1898 SelectBlock( m_currentCellCoords
, coords
);
1906 m_isDragging
= FALSE
;
1908 if ( coords
!= wxGridNoCellCoords
)
1910 if ( event
.LeftDown() )
1912 if ( event
.ShiftDown() )
1914 SelectBlock( m_currentCellCoords
, coords
);
1918 if ( !SendEvent( EVT_GRID_CELL_LEFT_CLICK
,
1923 MakeCellVisible( coords
);
1924 SetCurrentCell( coords
);
1930 // ------------ Left double click
1932 else if ( event
.LeftDClick() )
1934 SendEvent( EVT_GRID_CELL_LEFT_DCLICK
,
1941 // ------------ Left button released
1943 else if ( event
.LeftUp() )
1945 if ( m_cursorMode
== WXGRID_CURSOR_SELECT_CELL
)
1947 if ( IsSelection() )
1949 SendEvent( EVT_GRID_RANGE_SELECT
, -1, -1, event
);
1953 // Show the edit control, if it has
1954 // been hidden for drag-shrinking.
1955 if ( IsCellEditControlEnabled() )
1956 ShowCellEditControl();
1962 // ------------ Right button down
1964 else if ( event
.RightDown() )
1966 if ( !SendEvent( EVT_GRID_CELL_RIGHT_CLICK
,
1971 // no default action at the moment
1976 // ------------ Right double click
1978 else if ( event
.RightDClick() )
1980 if ( !SendEvent( EVT_GRID_CELL_RIGHT_DCLICK
,
1985 // no default action at the moment
1993 // ------ interaction with data model
1995 bool wxGrid::ProcessTableMessage( wxGridTableMessage
& msg
)
1997 switch ( msg
.GetId() )
1999 case wxGRIDTABLE_REQUEST_VIEW_GET_VALUES
:
2000 return GetModelValues();
2002 case wxGRIDTABLE_REQUEST_VIEW_SEND_VALUES
:
2003 return SetModelValues();
2005 case wxGRIDTABLE_NOTIFY_ROWS_INSERTED
:
2006 case wxGRIDTABLE_NOTIFY_ROWS_APPENDED
:
2007 case wxGRIDTABLE_NOTIFY_ROWS_DELETED
:
2008 case wxGRIDTABLE_NOTIFY_COLS_INSERTED
:
2009 case wxGRIDTABLE_NOTIFY_COLS_APPENDED
:
2010 case wxGRIDTABLE_NOTIFY_COLS_DELETED
:
2011 return Redimension( msg
);
2020 // The behaviour of this function depends on the grid table class
2021 // Clear() function. For the default wxGridStringTable class the
2022 // behavious is to replace all cell contents with wxEmptyString but
2023 // not to change the number of rows or cols.
2025 void wxGrid::ClearGrid()
2030 SetEditControlValue();
2031 if ( !GetBatchCount() ) m_gridWin
->Refresh();
2036 bool wxGrid::InsertRows( int pos
, int numRows
, bool WXUNUSED(updateLabels
) )
2038 // TODO: something with updateLabels flag
2042 wxLogError( wxT("Called wxGrid::InsertRows() before calling CreateGrid()") );
2048 bool ok
= m_table
->InsertRows( pos
, numRows
);
2050 // the table will have sent the results of the insert row
2051 // operation to this view object as a grid table message
2055 if ( m_numCols
== 0 )
2057 m_table
->AppendCols( WXGRID_DEFAULT_NUMBER_COLS
);
2059 // TODO: perhaps instead of appending the default number of cols
2060 // we should remember what the last non-zero number of cols was ?
2064 if ( m_currentCellCoords
== wxGridNoCellCoords
)
2066 // if we have just inserted cols into an empty grid the current
2067 // cell will be undefined...
2069 SetCurrentCell( 0, 0 );
2073 if ( !GetBatchCount() ) Refresh();
2076 SetEditControlValue();
2086 bool wxGrid::AppendRows( int numRows
, bool WXUNUSED(updateLabels
) )
2088 // TODO: something with updateLabels flag
2092 wxLogError( wxT("Called wxGrid::AppendRows() before calling CreateGrid()") );
2096 if ( m_table
&& m_table
->AppendRows( numRows
) )
2098 if ( m_currentCellCoords
== wxGridNoCellCoords
)
2100 // if we have just inserted cols into an empty grid the current
2101 // cell will be undefined...
2103 SetCurrentCell( 0, 0 );
2106 // the table will have sent the results of the append row
2107 // operation to this view object as a grid table message
2110 if ( !GetBatchCount() ) Refresh();
2120 bool wxGrid::DeleteRows( int pos
, int numRows
, bool WXUNUSED(updateLabels
) )
2122 // TODO: something with updateLabels flag
2126 wxLogError( wxT("Called wxGrid::DeleteRows() before calling CreateGrid()") );
2130 if ( m_table
&& m_table
->DeleteRows( pos
, numRows
) )
2132 // the table will have sent the results of the delete row
2133 // operation to this view object as a grid table message
2135 if ( m_numRows
> 0 )
2136 SetEditControlValue();
2138 HideCellEditControl();
2141 if ( !GetBatchCount() ) Refresh();
2151 bool wxGrid::InsertCols( int pos
, int numCols
, bool WXUNUSED(updateLabels
) )
2153 // TODO: something with updateLabels flag
2157 wxLogError( wxT("Called wxGrid::InsertCols() before calling CreateGrid()") );
2163 HideCellEditControl();
2164 bool ok
= m_table
->InsertCols( pos
, numCols
);
2166 // the table will have sent the results of the insert col
2167 // operation to this view object as a grid table message
2171 if ( m_currentCellCoords
== wxGridNoCellCoords
)
2173 // if we have just inserted cols into an empty grid the current
2174 // cell will be undefined...
2176 SetCurrentCell( 0, 0 );
2180 if ( !GetBatchCount() ) Refresh();
2183 SetEditControlValue();
2193 bool wxGrid::AppendCols( int numCols
, bool WXUNUSED(updateLabels
) )
2195 // TODO: something with updateLabels flag
2199 wxLogError( wxT("Called wxGrid::AppendCols() before calling CreateGrid()") );
2203 if ( m_table
&& m_table
->AppendCols( numCols
) )
2205 // the table will have sent the results of the append col
2206 // operation to this view object as a grid table message
2208 if ( m_currentCellCoords
== wxGridNoCellCoords
)
2210 // if we have just inserted cols into an empty grid the current
2211 // cell will be undefined...
2213 SetCurrentCell( 0, 0 );
2217 if ( !GetBatchCount() ) Refresh();
2227 bool wxGrid::DeleteCols( int pos
, int numCols
, bool WXUNUSED(updateLabels
) )
2229 // TODO: something with updateLabels flag
2233 wxLogError( wxT("Called wxGrid::DeleteCols() before calling CreateGrid()") );
2237 if ( m_table
&& m_table
->DeleteCols( pos
, numCols
) )
2239 // the table will have sent the results of the delete col
2240 // operation to this view object as a grid table message
2242 if ( m_numCols
> 0 )
2243 SetEditControlValue();
2245 HideCellEditControl();
2248 if ( !GetBatchCount() ) Refresh();
2260 // ----- event handlers
2263 // Generate a grid event based on a mouse event and
2264 // return the result of ProcessEvent()
2266 bool wxGrid::SendEvent( const wxEventType type
,
2268 wxMouseEvent
& mouseEv
)
2270 if ( type
== EVT_GRID_ROW_SIZE
||
2271 type
== EVT_GRID_COL_SIZE
)
2273 int rowOrCol
= (row
== -1 ? col
: row
);
2275 wxGridSizeEvent
gridEvt( GetId(),
2279 mouseEv
.GetX(), mouseEv
.GetY(),
2280 mouseEv
.ControlDown(),
2281 mouseEv
.ShiftDown(),
2283 mouseEv
.MetaDown() );
2285 return GetEventHandler()->ProcessEvent(gridEvt
);
2287 else if ( type
== EVT_GRID_RANGE_SELECT
)
2289 wxGridRangeSelectEvent
gridEvt( GetId(),
2293 m_selectedBottomRight
,
2294 mouseEv
.ControlDown(),
2295 mouseEv
.ShiftDown(),
2297 mouseEv
.MetaDown() );
2299 return GetEventHandler()->ProcessEvent(gridEvt
);
2303 wxGridEvent
gridEvt( GetId(),
2307 mouseEv
.GetX(), mouseEv
.GetY(),
2308 mouseEv
.ControlDown(),
2309 mouseEv
.ShiftDown(),
2311 mouseEv
.MetaDown() );
2313 return GetEventHandler()->ProcessEvent(gridEvt
);
2318 // Generate a grid event of specified type and return the result
2319 // of ProcessEvent().
2321 bool wxGrid::SendEvent( const wxEventType type
,
2324 if ( type
== EVT_GRID_ROW_SIZE
||
2325 type
== EVT_GRID_COL_SIZE
)
2327 int rowOrCol
= (row
== -1 ? col
: row
);
2329 wxGridSizeEvent
gridEvt( GetId(),
2334 return GetEventHandler()->ProcessEvent(gridEvt
);
2338 wxGridEvent
gridEvt( GetId(),
2343 return GetEventHandler()->ProcessEvent(gridEvt
);
2348 void wxGrid::OnPaint( wxPaintEvent
& WXUNUSED(event
) )
2350 wxPaintDC
dc( this );
2352 if ( m_currentCellCoords
== wxGridNoCellCoords
&&
2353 m_numRows
&& m_numCols
)
2355 m_currentCellCoords
.Set(0, 0);
2356 SetEditControlValue();
2357 ShowCellEditControl();
2362 // This is just here to make sure that CalcDimensions gets called when
2363 // the grid view is resized... then the size event is skipped to allow
2364 // the box sizers to handle everything
2366 void wxGrid::OnSize( wxSizeEvent
& event
)
2373 void wxGrid::OnKeyDown( wxKeyEvent
& event
)
2375 if ( m_inOnKeyDown
)
2377 // shouldn't be here - we are going round in circles...
2379 wxLogFatalError( wxT("wxGrid::OnKeyDown called while alread active") );
2382 m_inOnKeyDown
= TRUE
;
2384 // propagate the event up and see if it gets processed
2386 wxWindow
*parent
= GetParent();
2387 wxKeyEvent
keyEvt( event
);
2388 keyEvt
.SetEventObject( parent
);
2390 if ( !parent
->GetEventHandler()->ProcessEvent( keyEvt
) )
2392 // try local handlers
2394 switch ( event
.KeyCode() )
2397 if ( event
.ControlDown() )
2399 MoveCursorUpBlock();
2408 if ( event
.ControlDown() )
2410 MoveCursorDownBlock();
2419 if ( event
.ControlDown() )
2421 MoveCursorLeftBlock();
2430 if ( event
.ControlDown() )
2432 MoveCursorRightBlock();
2441 if ( event
.ControlDown() )
2443 event
.Skip(); // to let the edit control have the return
2452 if ( event
.ControlDown() )
2454 MakeCellVisible( 0, 0 );
2455 SetCurrentCell( 0, 0 );
2464 if ( event
.ControlDown() )
2466 MakeCellVisible( m_numRows
-1, m_numCols
-1 );
2467 SetCurrentCell( m_numRows
-1, m_numCols
-1 );
2484 // now try the cell edit control
2486 if ( IsCellEditControlEnabled() )
2488 event
.SetEventObject( m_cellEditCtrl
);
2489 m_cellEditCtrl
->GetEventHandler()->ProcessEvent( event
);
2495 m_inOnKeyDown
= FALSE
;
2499 void wxGrid::SetCurrentCell( const wxGridCellCoords
& coords
)
2501 if ( SendEvent( EVT_GRID_SELECT_CELL
, coords
.GetRow(), coords
.GetCol() ) )
2503 // the event has been intercepted - do nothing
2507 wxClientDC
dc( m_gridWin
);
2510 if ( m_currentCellCoords
!= wxGridNoCellCoords
)
2512 HideCellEditControl();
2513 SaveEditControlValue();
2516 m_currentCellCoords
= coords
;
2518 SetEditControlValue();
2519 ShowCellEditControl();
2521 if ( IsSelection() )
2523 wxRect
r( SelectionToDeviceRect() );
2525 if ( !GetBatchCount() ) m_gridWin
->Refresh( TRUE
, &r
);
2531 // ------ functions to get/send data (see also public functions)
2534 bool wxGrid::GetModelValues()
2538 // all we need to do is repaint the grid
2540 m_gridWin
->Refresh();
2548 bool wxGrid::SetModelValues()
2554 for ( row
= 0; row
< m_numRows
; row
++ )
2556 for ( col
= 0; col
< m_numCols
; col
++ )
2558 m_table
->SetValue( row
, col
, GetCellValue(row
, col
) );
2570 // Note - this function only draws cells that are in the list of
2571 // exposed cells (usually set from the update region by
2572 // CalcExposedCells)
2574 void wxGrid::DrawGridCellArea( wxDC
& dc
)
2576 if ( !m_numRows
|| !m_numCols
) return;
2579 size_t numCells
= m_cellsExposed
.GetCount();
2581 for ( i
= 0; i
< numCells
; i
++ )
2583 DrawCell( dc
, m_cellsExposed
[i
] );
2588 void wxGrid::DrawCell( wxDC
& dc
, const wxGridCellCoords
& coords
)
2590 if ( m_colWidths
[coords
.GetCol()] <=0 ||
2591 m_rowHeights
[coords
.GetRow()] <= 0 ) return;
2593 if ( m_gridLinesEnabled
)
2594 DrawCellBorder( dc
, coords
);
2596 DrawCellBackground( dc
, coords
);
2598 // TODO: separate functions here for different kinds of cells ?
2601 DrawCellValue( dc
, coords
);
2605 void wxGrid::DrawCellBorder( wxDC
& dc
, const wxGridCellCoords
& coords
)
2607 if ( m_colWidths
[coords
.GetCol()] <=0 ||
2608 m_rowHeights
[coords
.GetRow()] <= 0 ) return;
2610 dc
.SetPen( wxPen(GetGridLineColour(), 1, wxSOLID
) );
2611 int row
= coords
.GetRow();
2612 int col
= coords
.GetCol();
2614 // right hand border
2616 dc
.DrawLine( m_colRights
[col
], m_rowBottoms
[row
] - m_rowHeights
[row
],
2617 m_colRights
[col
], m_rowBottoms
[row
] );
2621 dc
.DrawLine( m_colRights
[col
] - m_colWidths
[col
], m_rowBottoms
[row
],
2622 m_colRights
[col
], m_rowBottoms
[row
] );
2626 void wxGrid::DrawCellBackground( wxDC
& dc
, const wxGridCellCoords
& coords
)
2628 if ( m_colWidths
[coords
.GetCol()] <=0 ||
2629 m_rowHeights
[coords
.GetRow()] <= 0 ) return;
2631 int row
= coords
.GetRow();
2632 int col
= coords
.GetCol();
2634 dc
.SetBackgroundMode( wxSOLID
);
2636 if ( IsInSelection( coords
) )
2638 // TODO: improve this
2640 dc
.SetBrush( *wxBLACK_BRUSH
);
2644 dc
.SetBrush( wxBrush(GetCellBackgroundColour(row
, col
), wxSOLID
) );
2647 dc
.SetPen( *wxTRANSPARENT_PEN
);
2649 dc
.DrawRectangle( m_colRights
[col
] - m_colWidths
[col
] + 1,
2650 m_rowBottoms
[row
] - m_rowHeights
[row
] + 1,
2652 m_rowHeights
[row
]-1 );
2656 void wxGrid::DrawCellValue( wxDC
& dc
, const wxGridCellCoords
& coords
)
2658 if ( m_colWidths
[coords
.GetCol()] <=0 ||
2659 m_rowHeights
[coords
.GetRow()] <= 0 ) return;
2661 int row
= coords
.GetRow();
2662 int col
= coords
.GetCol();
2664 dc
.SetBackgroundMode( wxTRANSPARENT
);
2666 if ( IsInSelection( row
, col
) )
2668 // TODO: improve this
2670 dc
.SetTextBackground( wxColour(0, 0, 0) );
2671 dc
.SetTextForeground( wxColour(255, 255, 255) );
2675 dc
.SetTextBackground( GetCellBackgroundColour(row
, col
) );
2676 dc
.SetTextForeground( GetCellTextColour(row
, col
) );
2678 dc
.SetFont( GetCellFont(row
, col
) );
2681 GetCellAlignment( row
, col
, &hAlign
, &vAlign
);
2684 rect
.SetX( m_colRights
[col
] - m_colWidths
[col
] + 2 );
2685 rect
.SetY( m_rowBottoms
[row
] - m_rowHeights
[row
] + 2 );
2686 rect
.SetWidth( m_colWidths
[col
] - 4 );
2687 rect
.SetHeight( m_rowHeights
[row
] - 4 );
2689 DrawTextRectangle( dc
, GetCellValue( row
, col
), rect
, hAlign
, vAlign
);
2694 // TODO: remove this ???
2695 // This is used to redraw all grid lines e.g. when the grid line colour
2698 void wxGrid::DrawAllGridLines( wxDC
& dc
)
2700 if ( !m_gridLinesEnabled
||
2702 !m_numCols
) return;
2705 m_gridWin
->GetClientSize(&cw
, &ch
);
2707 // virtual coords of visible area
2709 int top
, bottom
, left
, right
;
2710 CalcUnscrolledPosition( 0, 0, &left
, &top
);
2711 CalcUnscrolledPosition( cw
, ch
, &right
, &bottom
);
2713 dc
.SetPen( wxPen(GetGridLineColour(), 1, wxSOLID
) );
2715 // horizontal grid lines
2718 for ( i
= 0; i
<= m_numRows
; i
++ )
2720 if ( m_rowBottoms
[i
] > bottom
)
2724 else if ( m_rowBottoms
[i
] >= top
)
2726 dc
.DrawLine( left
, m_rowBottoms
[i
], right
, m_rowBottoms
[i
] );
2731 // vertical grid lines
2733 for ( i
= 0; i
<= m_numCols
; i
++ )
2735 if ( m_colRights
[i
] > right
)
2739 else if ( m_colRights
[i
] >= left
)
2741 dc
.DrawLine( m_colRights
[i
], top
, m_colRights
[i
], bottom
);
2747 void wxGrid::DrawRowLabels( wxDC
& dc
)
2749 if ( !m_numRows
|| !m_numCols
) return;
2752 size_t numLabels
= m_rowLabelsExposed
.GetCount();
2754 for ( i
= 0; i
< numLabels
; i
++ )
2756 DrawRowLabel( dc
, m_rowLabelsExposed
[i
] );
2761 void wxGrid::DrawRowLabel( wxDC
& dc
, int row
)
2763 if ( m_rowHeights
[row
] <= 0 ) return;
2765 // draw the label's horizontal border (the vertical border is
2766 // provided by the cell area window margin)
2768 dc
.SetPen( *wxBLACK_PEN
);
2770 dc
.DrawLine( 0, m_rowBottoms
[row
]+1,
2771 m_rowLabelWidth
, m_rowBottoms
[row
]+1 );
2773 dc
.SetPen( *wxWHITE_PEN
);
2775 dc
.DrawLine( 0, m_rowBottoms
[row
]+2,
2776 m_rowLabelWidth
, m_rowBottoms
[row
]+2 );
2778 dc
.SetBackgroundMode( wxTRANSPARENT
);
2779 dc
.SetTextForeground( GetLabelTextColour() );
2780 dc
.SetFont( GetLabelFont() );
2783 GetRowLabelAlignment( &hAlign
, &vAlign
);
2787 rect
.SetY( m_rowBottoms
[row
] - m_rowHeights
[row
] + 2 );
2788 rect
.SetWidth( m_rowLabelWidth
- 4 );
2789 rect
.SetHeight( m_rowHeights
[row
] - 4 );
2790 DrawTextRectangle( dc
, GetRowLabelValue( row
), rect
, hAlign
, vAlign
);
2794 void wxGrid::DrawColLabels( wxDC
& dc
)
2796 if ( !m_numRows
|| !m_numCols
) return;
2799 size_t numLabels
= m_colLabelsExposed
.GetCount();
2801 for ( i
= 0; i
< numLabels
; i
++ )
2803 DrawColLabel( dc
, m_colLabelsExposed
[i
] );
2808 void wxGrid::DrawColLabel( wxDC
& dc
, int col
)
2810 if ( m_colWidths
[col
] <= 0 ) return;
2812 // draw the label's vertical border (the horizontal border is
2813 // provided by the cell area window margin)
2815 dc
.SetPen( *wxBLACK_PEN
);
2817 dc
.DrawLine( m_colRights
[col
]+1, 0,
2818 m_colRights
[col
]+1, m_colLabelHeight
);
2820 dc
.SetPen( *wxWHITE_PEN
);
2822 dc
.DrawLine( m_colRights
[col
]+2, 0,
2823 m_colRights
[col
]+2, m_colLabelHeight
);
2825 dc
.SetBackgroundMode( wxTRANSPARENT
);
2826 dc
.SetTextForeground( GetLabelTextColour() );
2827 dc
.SetFont( GetLabelFont() );
2830 GetColLabelAlignment( &hAlign
, &vAlign
);
2833 rect
.SetX( m_colRights
[col
] - m_colWidths
[col
] + 2 );
2835 rect
.SetWidth( m_colWidths
[col
] - 4 );
2836 rect
.SetHeight( m_colLabelHeight
- 4 );
2837 DrawTextRectangle( dc
, GetColLabelValue( col
), rect
, hAlign
, vAlign
);
2841 void wxGrid::DrawTextRectangle( wxDC
& dc
,
2842 const wxString
& value
,
2847 long textWidth
, textHeight
;
2848 long lineWidth
, lineHeight
;
2849 wxArrayString lines
;
2851 dc
.SetClippingRegion( rect
);
2852 StringToLines( value
, lines
);
2853 if ( lines
.GetCount() )
2855 GetTextBoxSize( dc
, lines
, &textWidth
, &textHeight
);
2856 dc
.GetTextExtent( lines
[0], &lineWidth
, &lineHeight
);
2859 switch ( horizAlign
)
2862 x
= rect
.x
+ (rect
.width
- textWidth
- 1);
2866 x
= rect
.x
+ ((rect
.width
- textWidth
)/2);
2875 switch ( vertAlign
)
2878 y
= rect
.y
+ (rect
.height
- textHeight
- 1);
2882 y
= rect
.y
+ ((rect
.height
- textHeight
)/2);
2891 for ( size_t i
= 0; i
< lines
.GetCount(); i
++ )
2893 dc
.DrawText( lines
[i
], (long)x
, (long)y
);
2898 dc
.DestroyClippingRegion();
2902 // Split multi line text up into an array of strings. Any existing
2903 // contents of the string array are preserved.
2905 void wxGrid::StringToLines( const wxString
& value
, wxArrayString
& lines
)
2907 // TODO: this won't work for WXMAC ? (lines end with '\r')
2908 // => use wxTextFile functions then (VZ)
2911 while ( startPos
< (int)value
.Length() )
2913 pos
= value
.Mid(startPos
).Find( '\n' );
2918 else if ( pos
== 0 )
2920 lines
.Add( wxEmptyString
);
2924 if ( value
[startPos
+pos
-1] == '\r' )
2926 lines
.Add( value
.Mid(startPos
, pos
-1) );
2930 lines
.Add( value
.Mid(startPos
, pos
) );
2935 if ( startPos
< (int)value
.Length() )
2937 lines
.Add( value
.Mid( startPos
) );
2942 void wxGrid::GetTextBoxSize( wxDC
& dc
,
2943 wxArrayString
& lines
,
2944 long *width
, long *height
)
2951 for ( i
= 0; i
< lines
.GetCount(); i
++ )
2953 dc
.GetTextExtent( lines
[i
], &lineW
, &lineH
);
2954 w
= wxMax( w
, lineW
);
2964 // ------ Edit control functions
2968 void wxGrid::EnableEditing( bool edit
)
2970 // TODO: improve this ?
2972 if ( edit
!= m_editable
)
2976 // TODO: extend this for other edit control types
2978 if ( m_editCtrlType
== wxGRID_TEXTCTRL
)
2980 ((wxTextCtrl
*)m_cellEditCtrl
)->SetEditable( m_editable
);
2986 #if 0 // disabled for the moment - the cell control is always active
2987 void wxGrid::EnableCellEditControl( bool enable
)
2989 if ( m_cellEditCtrl
&&
2990 enable
!= m_cellEditCtrlEnabled
)
2992 m_cellEditCtrlEnabled
= enable
;
2994 if ( m_cellEditCtrlEnabled
)
2996 SetEditControlValue();
2997 ShowCellEditControl();
3001 HideCellEditControl();
3002 SaveEditControlValue();
3009 void wxGrid::ShowCellEditControl()
3013 if ( IsCellEditControlEnabled() )
3015 if ( !IsVisible( m_currentCellCoords
) )
3021 rect
= CellToRect( m_currentCellCoords
);
3023 // convert to scrolled coords
3025 int left
, top
, right
, bottom
;
3026 CalcScrolledPosition( rect
.GetLeft(), rect
.GetTop(), &left
, &top
);
3027 CalcScrolledPosition( rect
.GetRight(), rect
.GetBottom(), &right
, &bottom
);
3030 m_gridWin
->GetClientSize( &cw
, &ch
);
3032 // Make the edit control large enough to allow for internal margins
3033 // TODO: remove this if the text ctrl sizing is improved esp. for unix
3036 #if defined(__WXMOTIF__)
3037 if ( m_currentCellCoords
.GetRow() == 0 ||
3038 m_currentCellCoords
.GetCol() == 0 )
3047 if ( m_currentCellCoords
.GetRow() == 0 ||
3048 m_currentCellCoords
.GetCol() == 0 )
3058 #if defined(__WXGTK__)
3061 if (left
!= 0) left_diff
++;
3062 if (top
!= 0) top_diff
++;
3063 rect
.SetLeft( left
+ left_diff
);
3064 rect
.SetTop( top
+ top_diff
);
3065 rect
.SetRight( rect
.GetRight() - left_diff
);
3066 rect
.SetBottom( rect
.GetBottom() - top_diff
);
3068 rect
.SetLeft( wxMax(0, left
- extra
) );
3069 rect
.SetTop( wxMax(0, top
- extra
) );
3070 rect
.SetRight( rect
.GetRight() + 2*extra
);
3071 rect
.SetBottom( rect
.GetBottom() + 2*extra
);
3074 m_cellEditCtrl
->SetSize( rect
);
3075 m_cellEditCtrl
->Show( TRUE
);
3077 switch ( m_editCtrlType
)
3079 case wxGRID_TEXTCTRL
:
3080 ((wxTextCtrl
*) m_cellEditCtrl
)->SetInsertionPointEnd();
3083 case wxGRID_CHECKBOX
:
3084 // TODO: anything ???
3089 // TODO: anything ???
3093 case wxGRID_COMBOBOX
:
3094 // TODO: anything ???
3099 m_cellEditCtrl
->SetFocus();
3105 void wxGrid::HideCellEditControl()
3107 if ( IsCellEditControlEnabled() )
3109 m_cellEditCtrl
->Show( FALSE
);
3114 void wxGrid::SetEditControlValue( const wxString
& value
)
3120 s
= GetCellValue(m_currentCellCoords
);
3124 if ( IsCellEditControlEnabled() )
3126 switch ( m_editCtrlType
)
3128 case wxGRID_TEXTCTRL
:
3129 ((wxGridTextCtrl
*)m_cellEditCtrl
)->SetStartValue(s
);
3132 case wxGRID_CHECKBOX
:
3133 // TODO: implement this
3138 // TODO: implement this
3142 case wxGRID_COMBOBOX
:
3143 // TODO: implement this
3152 void wxGrid::SaveEditControlValue()
3156 wxWindow
*ctrl
= (wxWindow
*)NULL
;
3158 if ( IsCellEditControlEnabled() )
3160 ctrl
= m_cellEditCtrl
;
3167 bool valueChanged
= FALSE
;
3169 switch ( m_editCtrlType
)
3171 case wxGRID_TEXTCTRL
:
3172 valueChanged
= (((wxGridTextCtrl
*)ctrl
)->GetValue() !=
3173 ((wxGridTextCtrl
*)ctrl
)->GetStartValue());
3174 SetCellValue( m_currentCellCoords
,
3175 ((wxTextCtrl
*) ctrl
)->GetValue() );
3178 case wxGRID_CHECKBOX
:
3179 // TODO: implement this
3184 // TODO: implement this
3188 case wxGRID_COMBOBOX
:
3189 // TODO: implement this
3196 SendEvent( EVT_GRID_CELL_CHANGE
,
3197 m_currentCellCoords
.GetRow(),
3198 m_currentCellCoords
.GetCol() );
3205 // ------ Grid location functions
3206 // Note that all of these functions work with the logical coordinates of
3207 // grid cells and labels so you will need to convert from device
3208 // coordinates for mouse events etc.
3211 void wxGrid::XYToCell( int x
, int y
, wxGridCellCoords
& coords
)
3213 int row
= YToRow(y
);
3214 int col
= XToCol(x
);
3216 if ( row
== -1 || col
== -1 )
3218 coords
= wxGridNoCellCoords
;
3222 coords
.Set( row
, col
);
3227 int wxGrid::YToRow( int y
)
3231 for ( i
= 0; i
< m_numRows
; i
++ )
3233 if ( y
< m_rowBottoms
[i
] ) return i
;
3240 int wxGrid::XToCol( int x
)
3244 for ( i
= 0; i
< m_numCols
; i
++ )
3246 if ( x
< m_colRights
[i
] ) return i
;
3253 // return the row number that that the y coord is near the edge of, or
3254 // -1 if not near an edge
3256 int wxGrid::YToEdgeOfRow( int y
)
3260 for ( i
= 0; i
< m_numRows
; i
++ )
3262 if ( m_rowHeights
[i
] > WXGRID_LABEL_EDGE_ZONE
)
3264 d
= abs( y
- m_rowBottoms
[i
] );
3266 if ( d
< WXGRID_LABEL_EDGE_ZONE
) return i
;
3275 // return the col number that that the x coord is near the edge of, or
3276 // -1 if not near an edge
3278 int wxGrid::XToEdgeOfCol( int x
)
3282 for ( i
= 0; i
< m_numCols
; i
++ )
3284 if ( m_colWidths
[i
] > WXGRID_LABEL_EDGE_ZONE
)
3286 d
= abs( x
- m_colRights
[i
] );
3288 if ( d
< WXGRID_LABEL_EDGE_ZONE
) return i
;
3297 wxRect
wxGrid::CellToRect( int row
, int col
)
3299 wxRect
rect( -1, -1, -1, -1 );
3301 if ( row
>= 0 && row
< m_numRows
&&
3302 col
>= 0 && col
< m_numCols
)
3304 rect
.x
= m_colRights
[col
] - m_colWidths
[col
];
3305 rect
.y
= m_rowBottoms
[row
] - m_rowHeights
[row
];
3306 rect
.width
= m_colWidths
[col
];
3307 rect
.height
= m_rowHeights
[ row
];
3314 bool wxGrid::IsVisible( int row
, int col
, bool wholeCellVisible
)
3316 // get the cell rectangle in logical coords
3318 wxRect
r( CellToRect( row
, col
) );
3320 // convert to device coords
3322 int left
, top
, right
, bottom
;
3323 CalcScrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top
);
3324 CalcScrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom
);
3326 // check against the client area of the grid window
3329 m_gridWin
->GetClientSize( &cw
, &ch
);
3331 if ( wholeCellVisible
)
3333 // is the cell wholly visible ?
3335 return ( left
>= 0 && right
<= cw
&&
3336 top
>= 0 && bottom
<= ch
);
3340 // is the cell partly visible ?
3342 return ( ((left
>=0 && left
< cw
) || (right
> 0 && right
<= cw
)) &&
3343 ((top
>=0 && top
< ch
) || (bottom
> 0 && bottom
<= ch
)) );
3348 // make the specified cell location visible by doing a minimal amount
3351 void wxGrid::MakeCellVisible( int row
, int col
)
3354 int xpos
= -1, ypos
= -1;
3356 if ( row
>= 0 && row
< m_numRows
&&
3357 col
>= 0 && col
< m_numCols
)
3359 // get the cell rectangle in logical coords
3361 wxRect
r( CellToRect( row
, col
) );
3363 // convert to device coords
3365 int left
, top
, right
, bottom
;
3366 CalcScrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top
);
3367 CalcScrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom
);
3370 m_gridWin
->GetClientSize( &cw
, &ch
);
3376 else if ( bottom
> ch
)
3378 int h
= r
.GetHeight();
3380 for ( i
= row
-1; i
>= 0; i
-- )
3382 if ( h
+ m_rowHeights
[i
] > ch
) break;
3384 h
+= m_rowHeights
[i
];
3385 ypos
-= m_rowHeights
[i
];
3388 // we divide it later by GRID_SCROLL_LINE, make sure that we don't
3389 // have rounding errors (this is important, because if we do, we
3390 // might not scroll at all and some cells won't be redrawn)
3391 ypos
+= GRID_SCROLL_LINE
/ 2;
3398 else if ( right
> cw
)
3400 int w
= r
.GetWidth();
3402 for ( i
= col
-1; i
>= 0; i
-- )
3404 if ( w
+ m_colWidths
[i
] > cw
) break;
3406 w
+= m_colWidths
[i
];
3407 xpos
-= m_colWidths
[i
];
3410 // see comment for ypos above
3411 xpos
+= GRID_SCROLL_LINE
/ 2;
3414 if ( xpos
!= -1 || ypos
!= -1 )
3416 if ( xpos
!= -1 ) xpos
/= GRID_SCROLL_LINE
;
3417 if ( ypos
!= -1 ) ypos
/= GRID_SCROLL_LINE
;
3418 Scroll( xpos
, ypos
);
3426 // ------ Grid cursor movement functions
3429 bool wxGrid::MoveCursorUp()
3431 if ( m_currentCellCoords
!= wxGridNoCellCoords
&&
3432 m_currentCellCoords
.GetRow() > 0 )
3434 MakeCellVisible( m_currentCellCoords
.GetRow() - 1,
3435 m_currentCellCoords
.GetCol() );
3437 SetCurrentCell( m_currentCellCoords
.GetRow() - 1,
3438 m_currentCellCoords
.GetCol() );
3447 bool wxGrid::MoveCursorDown()
3449 // TODO: allow for scrolling
3451 if ( m_currentCellCoords
!= wxGridNoCellCoords
&&
3452 m_currentCellCoords
.GetRow() < m_numRows
-1 )
3454 MakeCellVisible( m_currentCellCoords
.GetRow() + 1,
3455 m_currentCellCoords
.GetCol() );
3457 SetCurrentCell( m_currentCellCoords
.GetRow() + 1,
3458 m_currentCellCoords
.GetCol() );
3467 bool wxGrid::MoveCursorLeft()
3469 if ( m_currentCellCoords
!= wxGridNoCellCoords
&&
3470 m_currentCellCoords
.GetCol() > 0 )
3472 MakeCellVisible( m_currentCellCoords
.GetRow(),
3473 m_currentCellCoords
.GetCol() - 1 );
3475 SetCurrentCell( m_currentCellCoords
.GetRow(),
3476 m_currentCellCoords
.GetCol() - 1 );
3485 bool wxGrid::MoveCursorRight()
3487 if ( m_currentCellCoords
!= wxGridNoCellCoords
&&
3488 m_currentCellCoords
.GetCol() < m_numCols
- 1 )
3490 MakeCellVisible( m_currentCellCoords
.GetRow(),
3491 m_currentCellCoords
.GetCol() + 1 );
3493 SetCurrentCell( m_currentCellCoords
.GetRow(),
3494 m_currentCellCoords
.GetCol() + 1 );
3503 bool wxGrid::MovePageUp()
3505 if ( m_currentCellCoords
== wxGridNoCellCoords
) return FALSE
;
3507 int row
= m_currentCellCoords
.GetRow();
3511 m_gridWin
->GetClientSize( &cw
, &ch
);
3513 int y
= m_rowBottoms
[ row
] - m_rowHeights
[ row
];
3514 int newRow
= YToRow( y
- ch
+ 1 );
3519 else if ( newRow
== row
)
3524 MakeCellVisible( newRow
, m_currentCellCoords
.GetCol() );
3525 SetCurrentCell( newRow
, m_currentCellCoords
.GetCol() );
3533 bool wxGrid::MovePageDown()
3535 if ( m_currentCellCoords
== wxGridNoCellCoords
) return FALSE
;
3537 int row
= m_currentCellCoords
.GetRow();
3538 if ( row
< m_numRows
)
3541 m_gridWin
->GetClientSize( &cw
, &ch
);
3543 int y
= m_rowBottoms
[ row
] - m_rowHeights
[ row
];
3544 int newRow
= YToRow( y
+ ch
);
3547 newRow
= m_numRows
- 1;
3549 else if ( newRow
== row
)
3554 MakeCellVisible( newRow
, m_currentCellCoords
.GetCol() );
3555 SetCurrentCell( newRow
, m_currentCellCoords
.GetCol() );
3563 bool wxGrid::MoveCursorUpBlock()
3566 m_currentCellCoords
!= wxGridNoCellCoords
&&
3567 m_currentCellCoords
.GetRow() > 0 )
3569 int row
= m_currentCellCoords
.GetRow();
3570 int col
= m_currentCellCoords
.GetCol();
3572 if ( m_table
->IsEmptyCell(row
, col
) )
3574 // starting in an empty cell: find the next block of
3580 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
3583 else if ( m_table
->IsEmptyCell(row
-1, col
) )
3585 // starting at the top of a block: find the next block
3591 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
3596 // starting within a block: find the top of the block
3601 if ( m_table
->IsEmptyCell(row
, col
) )
3609 MakeCellVisible( row
, col
);
3610 SetCurrentCell( row
, col
);
3618 bool wxGrid::MoveCursorDownBlock()
3621 m_currentCellCoords
!= wxGridNoCellCoords
&&
3622 m_currentCellCoords
.GetRow() < m_numRows
-1 )
3624 int row
= m_currentCellCoords
.GetRow();
3625 int col
= m_currentCellCoords
.GetCol();
3627 if ( m_table
->IsEmptyCell(row
, col
) )
3629 // starting in an empty cell: find the next block of
3632 while ( row
< m_numRows
-1 )
3635 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
3638 else if ( m_table
->IsEmptyCell(row
+1, col
) )
3640 // starting at the bottom of a block: find the next block
3643 while ( row
< m_numRows
-1 )
3646 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
3651 // starting within a block: find the bottom of the block
3653 while ( row
< m_numRows
-1 )
3656 if ( m_table
->IsEmptyCell(row
, col
) )
3664 MakeCellVisible( row
, col
);
3665 SetCurrentCell( row
, col
);
3673 bool wxGrid::MoveCursorLeftBlock()
3676 m_currentCellCoords
!= wxGridNoCellCoords
&&
3677 m_currentCellCoords
.GetCol() > 0 )
3679 int row
= m_currentCellCoords
.GetRow();
3680 int col
= m_currentCellCoords
.GetCol();
3682 if ( m_table
->IsEmptyCell(row
, col
) )
3684 // starting in an empty cell: find the next block of
3690 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
3693 else if ( m_table
->IsEmptyCell(row
, col
-1) )
3695 // starting at the left of a block: find the next block
3701 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
3706 // starting within a block: find the left of the block
3711 if ( m_table
->IsEmptyCell(row
, col
) )
3719 MakeCellVisible( row
, col
);
3720 SetCurrentCell( row
, col
);
3728 bool wxGrid::MoveCursorRightBlock()
3731 m_currentCellCoords
!= wxGridNoCellCoords
&&
3732 m_currentCellCoords
.GetCol() < m_numCols
-1 )
3734 int row
= m_currentCellCoords
.GetRow();
3735 int col
= m_currentCellCoords
.GetCol();
3737 if ( m_table
->IsEmptyCell(row
, col
) )
3739 // starting in an empty cell: find the next block of
3742 while ( col
< m_numCols
-1 )
3745 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
3748 else if ( m_table
->IsEmptyCell(row
, col
+1) )
3750 // starting at the right of a block: find the next block
3753 while ( col
< m_numCols
-1 )
3756 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
3761 // starting within a block: find the right of the block
3763 while ( col
< m_numCols
-1 )
3766 if ( m_table
->IsEmptyCell(row
, col
) )
3774 MakeCellVisible( row
, col
);
3775 SetCurrentCell( row
, col
);
3786 // ------ Label values and formatting
3789 void wxGrid::GetRowLabelAlignment( int *horiz
, int *vert
)
3791 *horiz
= m_rowLabelHorizAlign
;
3792 *vert
= m_rowLabelVertAlign
;
3795 void wxGrid::GetColLabelAlignment( int *horiz
, int *vert
)
3797 *horiz
= m_colLabelHorizAlign
;
3798 *vert
= m_colLabelVertAlign
;
3801 wxString
wxGrid::GetRowLabelValue( int row
)
3805 return m_table
->GetRowLabelValue( row
);
3815 wxString
wxGrid::GetColLabelValue( int col
)
3819 return m_table
->GetColLabelValue( col
);
3829 void wxGrid::SetRowLabelSize( int width
)
3831 // TODO: how to do this with the box sizers ?
3834 void wxGrid::SetColLabelSize( int height
)
3836 // TODO: how to do this with the box sizers ?
3839 void wxGrid::SetLabelBackgroundColour( const wxColour
& colour
)
3841 if ( m_labelBackgroundColour
!= colour
)
3843 m_labelBackgroundColour
= colour
;
3844 m_rowLabelWin
->SetBackgroundColour( colour
);
3845 m_colLabelWin
->SetBackgroundColour( colour
);
3846 m_cornerLabelWin
->SetBackgroundColour( colour
);
3848 if ( !GetBatchCount() )
3850 m_rowLabelWin
->Refresh();
3851 m_colLabelWin
->Refresh();
3852 m_cornerLabelWin
->Refresh();
3857 void wxGrid::SetLabelTextColour( const wxColour
& colour
)
3859 if ( m_labelTextColour
!= colour
)
3861 m_labelTextColour
= colour
;
3862 if ( !GetBatchCount() )
3864 m_rowLabelWin
->Refresh();
3865 m_colLabelWin
->Refresh();
3870 void wxGrid::SetLabelFont( const wxFont
& font
)
3873 if ( !GetBatchCount() )
3875 m_rowLabelWin
->Refresh();
3876 m_colLabelWin
->Refresh();
3880 void wxGrid::SetRowLabelAlignment( int horiz
, int vert
)
3882 if ( horiz
== wxLEFT
|| horiz
== wxCENTRE
|| horiz
== wxRIGHT
)
3884 m_rowLabelHorizAlign
= horiz
;
3887 if ( vert
== wxTOP
|| vert
== wxCENTRE
|| vert
== wxBOTTOM
)
3889 m_rowLabelVertAlign
= vert
;
3892 if ( !GetBatchCount() )
3894 m_rowLabelWin
->Refresh();
3895 m_colLabelWin
->Refresh();
3899 void wxGrid::SetColLabelAlignment( int horiz
, int vert
)
3901 if ( horiz
== wxLEFT
|| horiz
== wxCENTRE
|| horiz
== wxRIGHT
)
3903 m_colLabelHorizAlign
= horiz
;
3906 if ( vert
== wxTOP
|| vert
== wxCENTRE
|| vert
== wxBOTTOM
)
3908 m_colLabelVertAlign
= vert
;
3911 if ( !GetBatchCount() )
3913 m_rowLabelWin
->Refresh();
3914 m_colLabelWin
->Refresh();
3918 void wxGrid::SetRowLabelValue( int row
, const wxString
& s
)
3922 m_table
->SetRowLabelValue( row
, s
);
3923 if ( !GetBatchCount() )
3925 // TODO: Optimize this
3927 m_rowLabelWin
->Refresh();
3932 void wxGrid::SetColLabelValue( int col
, const wxString
& s
)
3936 m_table
->SetColLabelValue( col
, s
);
3937 if ( !GetBatchCount() )
3939 // TODO: optimize this
3941 m_colLabelWin
->Refresh();
3946 void wxGrid::SetGridLineColour( const wxColour
& colour
)
3948 if ( m_gridLineColour
!= colour
)
3950 m_gridLineColour
= colour
;
3952 wxClientDC
dc( m_gridWin
);
3954 DrawAllGridLines( dc
);
3958 void wxGrid::EnableGridLines( bool enable
)
3960 if ( enable
!= m_gridLinesEnabled
)
3962 m_gridLinesEnabled
= enable
;
3964 if ( !GetBatchCount() )
3968 wxClientDC
dc( m_gridWin
);
3970 DrawAllGridLines( dc
);
3974 m_gridWin
->Refresh();
3981 int wxGrid::GetDefaultRowSize()
3983 return m_defaultRowHeight
;
3986 int wxGrid::GetRowSize( int row
)
3988 if ( row
>= 0 && row
< m_numRows
)
3989 return m_rowHeights
[row
];
3991 return 0; // TODO: log an error here
3994 int wxGrid::GetDefaultColSize()
3996 return m_defaultColWidth
;
3999 int wxGrid::GetColSize( int col
)
4001 if ( col
>= 0 && col
< m_numCols
)
4002 return m_colWidths
[col
];
4004 return 0; // TODO: log an error here
4007 wxColour
wxGrid::GetDefaultCellBackgroundColour()
4009 // TODO: replace this temp test code
4011 return wxColour( 255, 255, 255 );
4014 wxColour
wxGrid::GetCellBackgroundColour( int WXUNUSED(row
), int WXUNUSED(col
) )
4016 // TODO: replace this temp test code
4018 return wxColour( 255, 255, 255 );
4021 wxColour
wxGrid::GetDefaultCellTextColour()
4023 // TODO: replace this temp test code
4025 return wxColour( 0, 0, 0 );
4028 wxColour
wxGrid::GetCellTextColour( int WXUNUSED(row
), int WXUNUSED(col
) )
4030 // TODO: replace this temp test code
4032 return wxColour( 0, 0, 0 );
4036 wxFont
wxGrid::GetDefaultCellFont()
4038 return m_defaultCellFont
;
4041 wxFont
wxGrid::GetCellFont( int WXUNUSED(row
), int WXUNUSED(col
) )
4043 // TODO: replace this temp test code
4045 return m_defaultCellFont
;
4048 void wxGrid::GetDefaultCellAlignment( int *horiz
, int *vert
)
4050 // TODO: replace this temp test code
4056 void wxGrid::GetCellAlignment( int WXUNUSED(row
), int WXUNUSED(col
), int *horiz
, int *vert
)
4058 // TODO: replace this temp test code
4064 void wxGrid::SetDefaultRowSize( int height
, bool resizeExistingRows
)
4066 m_defaultRowHeight
= wxMax( height
, WXGRID_MIN_ROW_HEIGHT
);
4068 if ( resizeExistingRows
)
4072 for ( row
= 0; row
< m_numRows
; row
++ )
4074 m_rowHeights
[row
] = m_defaultRowHeight
;
4075 bottom
+= m_defaultRowHeight
;
4076 m_rowBottoms
[row
] = bottom
;
4082 void wxGrid::SetRowSize( int row
, int height
)
4086 if ( row
>= 0 && row
< m_numRows
)
4088 int h
= wxMax( 0, height
);
4089 int diff
= h
- m_rowHeights
[row
];
4091 m_rowHeights
[row
] = h
;
4092 for ( i
= row
; i
< m_numRows
; i
++ )
4094 m_rowBottoms
[i
] += diff
;
4098 // Note: we are ending the event *after* doing
4099 // default processing in this case
4101 SendEvent( EVT_GRID_ROW_SIZE
,
4106 // TODO: log an error here
4110 void wxGrid::SetDefaultColSize( int width
, bool resizeExistingCols
)
4112 m_defaultColWidth
= wxMax( width
, WXGRID_MIN_COL_WIDTH
);
4114 if ( resizeExistingCols
)
4118 for ( col
= 0; col
< m_numCols
; col
++ )
4120 m_colWidths
[col
] = m_defaultColWidth
;
4121 right
+= m_defaultColWidth
;
4122 m_colRights
[col
] = right
;
4128 void wxGrid::SetColSize( int col
, int width
)
4132 if ( col
>= 0 && col
< m_numCols
)
4134 int w
= wxMax( 0, width
);
4135 int diff
= w
- m_colWidths
[col
];
4136 m_colWidths
[col
] = w
;
4138 for ( i
= col
; i
< m_numCols
; i
++ )
4140 m_colRights
[i
] += diff
;
4144 // Note: we are ending the event *after* doing
4145 // default processing in this case
4147 SendEvent( EVT_GRID_COL_SIZE
,
4152 // TODO: log an error here
4156 void wxGrid::SetDefaultCellBackgroundColour( const wxColour
& )
4158 // TODO: everything !!!
4162 void wxGrid::SetCellBackgroundColour( int WXUNUSED(row
), int WXUNUSED(col
), const wxColour
& )
4164 // TODO: everything !!!
4168 void wxGrid::SetDefaultCellTextColour( const wxColour
& )
4170 // TODO: everything !!!
4174 void wxGrid::SetCellTextColour( int WXUNUSED(row
), int WXUNUSED(col
), const wxColour
& )
4176 // TODO: everything !!!
4180 void wxGrid::SetDefaultCellFont( const wxFont
& )
4182 // TODO: everything !!!
4186 void wxGrid::SetCellFont( int WXUNUSED(row
), int WXUNUSED(col
), const wxFont
& )
4188 // TODO: everything !!!
4192 void wxGrid::SetDefaultCellAlignment( int WXUNUSED(horiz
), int WXUNUSED(vert
) )
4194 // TODO: everything !!!
4198 void wxGrid::SetCellAlignment( int WXUNUSED(row
), int WXUNUSED(col
), int WXUNUSED(horiz
), int WXUNUSED(vert
) )
4200 // TODO: everything !!!
4207 // ------ cell value accessor functions
4210 void wxGrid::SetCellValue( int row
, int col
, const wxString
& s
)
4214 m_table
->SetValue( row
, col
, s
.c_str() );
4215 if ( !GetBatchCount() )
4217 wxClientDC
dc( m_gridWin
);
4219 DrawCell( dc
, wxGridCellCoords(row
, col
) );
4222 #if 0 // TODO: edit in place
4224 if ( m_currentCellCoords
.GetRow() == row
&&
4225 m_currentCellCoords
.GetCol() == col
)
4227 SetEditControlValue( s
);
4236 // ------ Block, row and col selection
4239 void wxGrid::SelectRow( int row
, bool addToSelected
)
4243 if ( IsSelection() && addToSelected
)
4245 if ( m_selectedTopLeft
.GetRow() > row
)
4246 m_selectedTopLeft
.SetRow( row
);
4248 m_selectedTopLeft
.SetCol( 0 );
4250 if ( m_selectedBottomRight
.GetRow() < row
)
4251 m_selectedBottomRight
.SetRow( row
);
4253 m_selectedBottomRight
.SetCol( m_numCols
- 1 );
4255 // TODO: optimize this so that we only refresh the newly
4258 r
= SelectionToDeviceRect();
4259 if ( r
!= wxGridNoCellRect
) m_gridWin
->Refresh( TRUE
, &r
);
4263 r
= SelectionToDeviceRect();
4265 if ( r
!= wxGridNoCellRect
) m_gridWin
->Refresh( TRUE
, &r
);
4267 m_selectedTopLeft
.Set( row
, 0 );
4268 m_selectedBottomRight
.Set( row
, m_numCols
-1 );
4269 r
= SelectionToDeviceRect();
4270 m_gridWin
->Refresh( TRUE
, &r
);
4273 wxGridRangeSelectEvent
gridEvt( GetId(),
4274 EVT_GRID_RANGE_SELECT
,
4277 m_selectedBottomRight
);
4279 GetEventHandler()->ProcessEvent(gridEvt
);
4283 void wxGrid::SelectCol( int col
, bool addToSelected
)
4287 if ( IsSelection() && addToSelected
)
4289 if ( m_selectedTopLeft
.GetCol() > col
)
4290 m_selectedTopLeft
.SetCol( col
);
4292 m_selectedTopLeft
.SetRow( 0 );
4294 if ( m_selectedBottomRight
.GetCol() < col
)
4295 m_selectedBottomRight
.SetCol( col
);
4297 m_selectedBottomRight
.SetRow( m_numRows
- 1 );
4299 // TODO: optimize this so that we only refresh the newly
4302 r
= SelectionToDeviceRect();
4303 if ( r
!= wxGridNoCellRect
) m_gridWin
->Refresh( TRUE
, &r
);
4307 r
= SelectionToDeviceRect();
4309 if ( r
!= wxGridNoCellRect
) m_gridWin
->Refresh( TRUE
, &r
);
4311 m_selectedTopLeft
.Set( 0, col
);
4312 m_selectedBottomRight
.Set( m_numRows
-1, col
);
4313 r
= SelectionToDeviceRect();
4314 m_gridWin
->Refresh( TRUE
, &r
);
4317 wxGridRangeSelectEvent
gridEvt( GetId(),
4318 EVT_GRID_RANGE_SELECT
,
4321 m_selectedBottomRight
);
4323 GetEventHandler()->ProcessEvent(gridEvt
);
4327 void wxGrid::SelectBlock( int topRow
, int leftCol
, int bottomRow
, int rightCol
)
4330 bool changed
= false;
4331 wxGridCellCoords updateTopLeft
, updateBottomRight
;
4333 if ( topRow
> bottomRow
)
4340 if ( leftCol
> rightCol
)
4347 updateTopLeft
= m_selectedTopLeft
;
4348 if (m_selectedTopLeft
!= wxGridCellCoords( topRow
, leftCol
) )
4350 m_selectedTopLeft
= wxGridCellCoords( topRow
, leftCol
);
4351 if (updateTopLeft
== wxGridNoCellCoords
)
4353 updateTopLeft
= m_selectedTopLeft
;
4357 if(updateTopLeft
.GetRow() > topRow
)
4358 updateTopLeft
.SetRow(topRow
);
4359 if (updateTopLeft
.GetCol() > leftCol
)
4360 updateTopLeft
.SetCol(leftCol
);
4365 updateBottomRight
= m_selectedBottomRight
;
4366 if (m_selectedBottomRight
!= wxGridCellCoords( bottomRow
, rightCol
) )
4368 m_selectedBottomRight
= wxGridCellCoords( bottomRow
, rightCol
);
4369 if (updateBottomRight
== wxGridNoCellCoords
)
4371 updateBottomRight
= m_selectedBottomRight
;
4375 if (updateBottomRight
.GetRow() < bottomRow
)
4376 updateBottomRight
.SetRow(bottomRow
);
4377 if (updateBottomRight
.GetCol() < rightCol
)
4378 updateBottomRight
.SetCol(rightCol
);
4385 wxRect
r( BlockToDeviceRect( updateTopLeft
, updateBottomRight
) );
4386 m_gridWin
->Refresh( TRUE
, &r
);
4389 // only generate an event if the block is not being selected by
4390 // dragging the mouse (in which case the event will be generated in
4391 // the mouse event handler)
4392 if ( !m_isDragging
)
4394 wxGridRangeSelectEvent
gridEvt( GetId(),
4395 EVT_GRID_RANGE_SELECT
,
4398 m_selectedBottomRight
);
4400 GetEventHandler()->ProcessEvent(gridEvt
);
4404 void wxGrid::SelectAll()
4406 m_selectedTopLeft
.Set( 0, 0 );
4407 m_selectedBottomRight
.Set( m_numRows
-1, m_numCols
-1 );
4409 m_gridWin
->Refresh();
4413 void wxGrid::ClearSelection()
4415 m_selectedTopLeft
= wxGridNoCellCoords
;
4416 m_selectedBottomRight
= wxGridNoCellCoords
;
4420 // This function returns the rectangle that encloses the given block
4421 // in device coords clipped to the client size of the grid window.
4423 wxRect
wxGrid::BlockToDeviceRect( const wxGridCellCoords
&topLeft
,
4424 const wxGridCellCoords
&bottomRight
)
4426 wxRect
rect( wxGridNoCellRect
);
4429 cellRect
= CellToRect( topLeft
);
4430 if ( cellRect
!= wxGridNoCellRect
)
4436 rect
= wxRect( 0, 0, 0, 0 );
4439 cellRect
= CellToRect( bottomRight
);
4440 if ( cellRect
!= wxGridNoCellRect
)
4446 return wxGridNoCellRect
;
4449 // convert to scrolled coords
4451 int left
, top
, right
, bottom
;
4452 CalcScrolledPosition( rect
.GetLeft(), rect
.GetTop(), &left
, &top
);
4453 CalcScrolledPosition( rect
.GetRight(), rect
.GetBottom(), &right
, &bottom
);
4456 m_gridWin
->GetClientSize( &cw
, &ch
);
4458 rect
.SetLeft( wxMax(0, left
) );
4459 rect
.SetTop( wxMax(0, top
) );
4460 rect
.SetRight( wxMin(cw
, right
) );
4461 rect
.SetBottom( wxMin(ch
, bottom
) );
4469 // ------ Grid event classes
4472 IMPLEMENT_DYNAMIC_CLASS( wxGridEvent
, wxEvent
)
4474 wxGridEvent::wxGridEvent( int id
, wxEventType type
, wxObject
* obj
,
4475 int row
, int col
, int x
, int y
,
4476 bool control
, bool shift
, bool alt
, bool meta
)
4477 : wxNotifyEvent( type
, id
)
4483 m_control
= control
;
4488 SetEventObject(obj
);
4492 IMPLEMENT_DYNAMIC_CLASS( wxGridSizeEvent
, wxEvent
)
4494 wxGridSizeEvent::wxGridSizeEvent( int id
, wxEventType type
, wxObject
* obj
,
4495 int rowOrCol
, int x
, int y
,
4496 bool control
, bool shift
, bool alt
, bool meta
)
4497 : wxNotifyEvent( type
, id
)
4499 m_rowOrCol
= rowOrCol
;
4502 m_control
= control
;
4507 SetEventObject(obj
);
4511 IMPLEMENT_DYNAMIC_CLASS( wxGridRangeSelectEvent
, wxEvent
)
4513 wxGridRangeSelectEvent::wxGridRangeSelectEvent(int id
, wxEventType type
, wxObject
* obj
,
4514 const wxGridCellCoords
& topLeft
,
4515 const wxGridCellCoords
& bottomRight
,
4516 bool control
, bool shift
, bool alt
, bool meta
)
4517 : wxNotifyEvent( type
, id
)
4519 m_topLeft
= topLeft
;
4520 m_bottomRight
= bottomRight
;
4521 m_control
= control
;
4526 SetEventObject(obj
);
4530 #endif // ifndef wxUSE_NEW_GRID