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"
40 //////////////////////////////////////////////////////////////////////
42 wxGridCellCoords
wxGridNoCellCoords( -1, -1 );
43 wxRect
wxGridNoCellRect( -1, -1, -1, -1 );
45 // this is a magic incantation which must be done!
46 #include "wx/arrimpl.cpp"
48 WX_DEFINE_OBJARRAY(wxGridCellCoordsArray
)
51 //////////////////////////////////////////////////////////////////////
53 // Abstract base class for grid data (the model)
55 IMPLEMENT_ABSTRACT_CLASS( wxGridTableBase
, wxObject
)
58 wxGridTableBase::wxGridTableBase()
61 m_view
= (wxGrid
*) NULL
;
64 wxGridTableBase::~wxGridTableBase()
69 bool wxGridTableBase::InsertRows( size_t pos
, size_t numRows
)
71 wxLogWarning( wxT("Called grid table class function InsertRows(pos=%d, N=%d)\n"
72 "but your derived table class does not override this function"),
78 bool wxGridTableBase::AppendRows( size_t numRows
)
80 wxLogWarning( wxT("Called grid table class function AppendRows(N=%d)\n"
81 "but your derived table class does not override this function"),
87 bool wxGridTableBase::DeleteRows( size_t pos
, size_t numRows
)
89 wxLogWarning( wxT("Called grid table class function DeleteRows(pos=%d, N=%d)\n"
90 "but your derived table class does not override this function"),
96 bool wxGridTableBase::InsertCols( size_t pos
, size_t numCols
)
98 wxLogWarning( wxT("Called grid table class function InsertCols(pos=%d, N=%d)\n"
99 "but your derived table class does not override this function"),
105 bool wxGridTableBase::AppendCols( size_t numCols
)
107 wxLogWarning( wxT("Called grid table class function AppendCols(N=%d)\n"
108 "but your derived table class does not override this function"),
114 bool wxGridTableBase::DeleteCols( size_t pos
, size_t numCols
)
116 wxLogWarning( wxT("Called grid table class function DeleteCols(pos=%d, N=%d)\n"
117 "but your derived table class does not override this function"),
124 wxString
wxGridTableBase::GetRowLabelValue( int row
)
131 wxString
wxGridTableBase::GetColLabelValue( int col
)
133 // default col labels are:
134 // cols 0 to 25 : A-Z
135 // cols 26 to 675 : AA-ZZ
142 s
+= ('A' + (char)( col%26
));
144 if ( col
< 0 ) break;
147 // reverse the string...
149 for ( i
= 0; i
< n
; i
++ )
159 //////////////////////////////////////////////////////////////////////
161 // Message class for the grid table to send requests and notifications
165 wxGridTableMessage::wxGridTableMessage()
167 m_table
= (wxGridTableBase
*) NULL
;
173 wxGridTableMessage::wxGridTableMessage( wxGridTableBase
*table
, int id
,
174 int commandInt1
, int commandInt2
)
178 m_comInt1
= commandInt1
;
179 m_comInt2
= commandInt2
;
184 //////////////////////////////////////////////////////////////////////
186 // A basic grid table for string data. An object of this class will
187 // created by wxGrid if you don't specify an alternative table class.
190 WX_DEFINE_OBJARRAY(wxGridStringArray
)
192 IMPLEMENT_DYNAMIC_CLASS( wxGridStringTable
, wxGridTableBase
)
194 wxGridStringTable::wxGridStringTable()
199 wxGridStringTable::wxGridStringTable( int numRows
, int numCols
)
204 m_data
.Alloc( numRows
);
208 for ( col
= 0; col
< numCols
; col
++ )
210 sa
.Add( wxEmptyString
);
213 for ( row
= 0; row
< numRows
; row
++ )
219 wxGridStringTable::~wxGridStringTable()
223 long wxGridStringTable::GetNumberRows()
225 return m_data
.GetCount();
228 long wxGridStringTable::GetNumberCols()
230 if ( m_data
.GetCount() > 0 )
231 return m_data
[0].GetCount();
236 wxString
wxGridStringTable::GetValue( int row
, int col
)
238 // TODO: bounds checking
240 return m_data
[row
][col
];
243 void wxGridStringTable::SetValue( int row
, int col
, const wxString
& s
)
245 // TODO: bounds checking
247 m_data
[row
][col
] = s
;
250 bool wxGridStringTable::IsEmptyCell( int row
, int col
)
252 // TODO: bounds checking
254 return (m_data
[row
][col
] == wxEmptyString
);
258 void wxGridStringTable::Clear()
261 int numRows
, numCols
;
263 numRows
= m_data
.GetCount();
266 numCols
= m_data
[0].GetCount();
268 for ( row
= 0; row
< numRows
; row
++ )
270 for ( col
= 0; col
< numCols
; col
++ )
272 m_data
[row
][col
] = wxEmptyString
;
279 bool wxGridStringTable::InsertRows( size_t pos
, size_t numRows
)
283 size_t curNumRows
= m_data
.GetCount();
284 size_t curNumCols
= ( curNumRows
> 0 ? m_data
[0].GetCount() : 0 );
286 if ( pos
>= curNumRows
)
288 return AppendRows( numRows
);
292 sa
.Alloc( curNumCols
);
293 for ( col
= 0; col
< curNumCols
; col
++ )
295 sa
.Add( wxEmptyString
);
298 for ( row
= pos
; row
< pos
+ numRows
; row
++ )
300 m_data
.Insert( sa
, row
);
305 wxGridTableMessage
msg( this,
306 wxGRIDTABLE_NOTIFY_ROWS_INSERTED
,
310 GetView()->ProcessTableMessage( msg
);
316 bool wxGridStringTable::AppendRows( size_t numRows
)
320 size_t curNumRows
= m_data
.GetCount();
321 size_t curNumCols
= ( curNumRows
> 0 ? m_data
[0].GetCount() : 0 );
324 if ( curNumCols
> 0 )
326 sa
.Alloc( curNumCols
);
327 for ( col
= 0; col
< curNumCols
; col
++ )
329 sa
.Add( wxEmptyString
);
333 for ( row
= 0; row
< numRows
; row
++ )
340 wxGridTableMessage
msg( this,
341 wxGRIDTABLE_NOTIFY_ROWS_APPENDED
,
344 GetView()->ProcessTableMessage( msg
);
350 bool wxGridStringTable::DeleteRows( size_t pos
, size_t numRows
)
354 size_t curNumRows
= m_data
.GetCount();
356 if ( pos
>= curNumRows
)
358 wxLogError( wxT("Called wxGridStringTable::DeleteRows(pos=%d, N=%d)...\n"
359 "Pos value is invalid for present table with %d rows"),
360 pos
, numRows
, curNumRows
);
364 if ( numRows
> curNumRows
- pos
)
366 numRows
= curNumRows
- pos
;
369 if ( numRows
>= curNumRows
)
371 m_data
.Empty(); // don't release memory just yet
375 for ( n
= 0; n
< numRows
; n
++ )
377 m_data
.Remove( pos
);
383 wxGridTableMessage
msg( this,
384 wxGRIDTABLE_NOTIFY_ROWS_DELETED
,
388 GetView()->ProcessTableMessage( msg
);
394 bool wxGridStringTable::InsertCols( size_t pos
, size_t numCols
)
398 size_t curNumRows
= m_data
.GetCount();
399 size_t curNumCols
= ( curNumRows
> 0 ? m_data
[0].GetCount() : 0 );
401 if ( pos
>= curNumCols
)
403 return AppendCols( numCols
);
406 for ( row
= 0; row
< curNumRows
; row
++ )
408 for ( col
= pos
; col
< pos
+ numCols
; col
++ )
410 m_data
[row
].Insert( wxEmptyString
, col
);
416 wxGridTableMessage
msg( this,
417 wxGRIDTABLE_NOTIFY_COLS_INSERTED
,
421 GetView()->ProcessTableMessage( msg
);
427 bool wxGridStringTable::AppendCols( size_t numCols
)
431 size_t curNumRows
= m_data
.GetCount();
434 // TODO: something better than this ?
436 wxLogError( wxT("Unable to append cols to a grid table with no rows.\n"
437 "Call AppendRows() first") );
441 for ( row
= 0; row
< curNumRows
; row
++ )
443 for ( n
= 0; n
< numCols
; n
++ )
445 m_data
[row
].Add( wxEmptyString
);
451 wxGridTableMessage
msg( this,
452 wxGRIDTABLE_NOTIFY_COLS_APPENDED
,
455 GetView()->ProcessTableMessage( msg
);
461 bool wxGridStringTable::DeleteCols( size_t pos
, size_t numCols
)
465 size_t curNumRows
= m_data
.GetCount();
466 size_t curNumCols
= ( curNumRows
> 0 ? m_data
[0].GetCount() : 0 );
468 if ( pos
>= curNumCols
)
470 wxLogError( wxT("Called wxGridStringTable::DeleteCols(pos=%d, N=%d)...\n"
471 "Pos value is invalid for present table with %d cols"),
472 pos
, numCols
, curNumCols
);
476 if ( numCols
> curNumCols
- pos
)
478 numCols
= curNumCols
- pos
;
481 for ( row
= 0; row
< curNumRows
; row
++ )
483 if ( numCols
>= curNumCols
)
489 for ( n
= 0; n
< numCols
; n
++ )
491 m_data
[row
].Remove( pos
);
498 wxGridTableMessage
msg( this,
499 wxGRIDTABLE_NOTIFY_COLS_DELETED
,
503 GetView()->ProcessTableMessage( msg
);
509 wxString
wxGridStringTable::GetRowLabelValue( int row
)
511 if ( row
> (int)(m_rowLabels
.GetCount()) - 1 )
513 // using default label
515 return wxGridTableBase::GetRowLabelValue( row
);
519 return m_rowLabels
[ row
];
523 wxString
wxGridStringTable::GetColLabelValue( int col
)
525 if ( col
> (int)(m_colLabels
.GetCount()) - 1 )
527 // using default label
529 return wxGridTableBase::GetColLabelValue( col
);
533 return m_colLabels
[ col
];
537 void wxGridStringTable::SetRowLabelValue( int row
, const wxString
& value
)
539 if ( row
> (int)(m_rowLabels
.GetCount()) - 1 )
541 int n
= m_rowLabels
.GetCount();
543 for ( i
= n
; i
<= row
; i
++ )
545 m_rowLabels
.Add( wxGridTableBase::GetRowLabelValue(i
) );
549 m_rowLabels
[row
] = value
;
552 void wxGridStringTable::SetColLabelValue( int col
, const wxString
& value
)
554 if ( col
> (int)(m_colLabels
.GetCount()) - 1 )
556 int n
= m_colLabels
.GetCount();
558 for ( i
= n
; i
<= col
; i
++ )
560 m_colLabels
.Add( wxGridTableBase::GetColLabelValue(i
) );
564 m_colLabels
[col
] = value
;
570 //////////////////////////////////////////////////////////////////////
572 IMPLEMENT_DYNAMIC_CLASS( wxGridTextCtrl
, wxTextCtrl
)
574 BEGIN_EVENT_TABLE( wxGridTextCtrl
, wxTextCtrl
)
575 EVT_KEY_DOWN( wxGridTextCtrl::OnKeyDown
)
579 wxGridTextCtrl::wxGridTextCtrl( wxWindow
*par
,
583 const wxString
& value
,
587 : wxTextCtrl( par
, id
, value
, pos
, size
, style
)
590 m_isCellControl
= isCellControl
;
594 void wxGridTextCtrl::OnKeyDown( wxKeyEvent
& event
)
596 switch ( event
.KeyCode() )
599 m_grid
->SetEditControlValue( startValue
);
600 SetInsertionPointEnd();
609 if ( m_isCellControl
)
611 // send the event to the parent grid, skipping the
612 // event if nothing happens
614 event
.Skip( m_grid
->ProcessEvent( event
) );
618 // default text control response within the top edit
626 if ( m_isCellControl
)
628 if ( !m_grid
->ProcessEvent( event
) )
631 // wxMotif needs a little extra help...
633 int pos
= GetInsertionPoint();
634 wxString
s( GetValue() );
635 s
= s
.Left(pos
) + "\n" + s
.Mid(pos
);
637 SetInsertionPoint( pos
);
639 // the other ports can handle a Return key press
649 if ( m_isCellControl
)
651 // send the event to the parent grid, skipping the
652 // event if nothing happens
654 event
.Skip( m_grid
->ProcessEvent( event
) );
658 // default text control response within the top edit
670 void wxGridTextCtrl::SetStartValue( const wxString
& s
)
673 wxTextCtrl::SetValue( s
.c_str() );
678 //////////////////////////////////////////////////////////////////////
680 IMPLEMENT_DYNAMIC_CLASS( wxGridRowLabelWindow
, wxWindow
)
682 BEGIN_EVENT_TABLE( wxGridRowLabelWindow
, wxWindow
)
683 EVT_PAINT( wxGridRowLabelWindow::OnPaint
)
684 EVT_MOUSE_EVENTS( wxGridRowLabelWindow::OnMouseEvent
)
685 EVT_KEY_DOWN( wxGridRowLabelWindow::OnKeyDown
)
688 wxGridRowLabelWindow::wxGridRowLabelWindow( wxGrid
*parent
,
690 const wxPoint
&pos
, const wxSize
&size
)
691 : wxWindow( parent
, id
, pos
, size
)
696 void wxGridRowLabelWindow::OnPaint( wxPaintEvent
&event
)
700 // NO - don't do this because it will set both the x and y origin
701 // coords to match the parent scrolled window and we just want to
702 // set the y coord - MB
704 // m_owner->PrepareDC( dc );
707 m_owner
->CalcUnscrolledPosition( 0, 0, &x
, &y
);
708 dc
.SetDeviceOrigin( 0, -y
);
710 m_owner
->CalcRowLabelsExposed( GetUpdateRegion() );
711 m_owner
->DrawRowLabels( dc
);
715 void wxGridRowLabelWindow::OnMouseEvent( wxMouseEvent
& event
)
717 m_owner
->ProcessRowLabelMouseEvent( event
);
721 // This seems to be required for wxMotif otherwise the mouse
722 // cursor must be in the cell edit control to get key events
724 void wxGridRowLabelWindow::OnKeyDown( wxKeyEvent
& event
)
726 if ( !m_owner
->ProcessEvent( event
) ) event
.Skip();
731 //////////////////////////////////////////////////////////////////////
733 IMPLEMENT_DYNAMIC_CLASS( wxGridColLabelWindow
, wxWindow
)
735 BEGIN_EVENT_TABLE( wxGridColLabelWindow
, wxWindow
)
736 EVT_PAINT( wxGridColLabelWindow::OnPaint
)
737 EVT_MOUSE_EVENTS( wxGridColLabelWindow::OnMouseEvent
)
738 EVT_KEY_DOWN( wxGridColLabelWindow::OnKeyDown
)
741 wxGridColLabelWindow::wxGridColLabelWindow( wxGrid
*parent
,
743 const wxPoint
&pos
, const wxSize
&size
)
744 : wxWindow( parent
, id
, pos
, size
)
749 void wxGridColLabelWindow::OnPaint( wxPaintEvent
&event
)
753 // NO - don't do this because it will set both the x and y origin
754 // coords to match the parent scrolled window and we just want to
755 // set the x coord - MB
757 // m_owner->PrepareDC( dc );
760 m_owner
->CalcUnscrolledPosition( 0, 0, &x
, &y
);
761 dc
.SetDeviceOrigin( -x
, 0 );
763 m_owner
->CalcColLabelsExposed( GetUpdateRegion() );
764 m_owner
->DrawColLabels( dc
);
768 void wxGridColLabelWindow::OnMouseEvent( wxMouseEvent
& event
)
770 m_owner
->ProcessColLabelMouseEvent( event
);
774 // This seems to be required for wxMotif otherwise the mouse
775 // cursor must be in the cell edit control to get key events
777 void wxGridColLabelWindow::OnKeyDown( wxKeyEvent
& event
)
779 if ( !m_owner
->ProcessEvent( event
) ) event
.Skip();
784 //////////////////////////////////////////////////////////////////////
786 IMPLEMENT_DYNAMIC_CLASS( wxGridCornerLabelWindow
, wxWindow
)
788 BEGIN_EVENT_TABLE( wxGridCornerLabelWindow
, wxWindow
)
789 EVT_MOUSE_EVENTS( wxGridCornerLabelWindow::OnMouseEvent
)
790 EVT_KEY_DOWN( wxGridCornerLabelWindow::OnKeyDown
)
793 wxGridCornerLabelWindow::wxGridCornerLabelWindow( wxGrid
*parent
,
795 const wxPoint
&pos
, const wxSize
&size
)
796 : wxWindow( parent
, id
, pos
, size
, wxRAISED_BORDER
)
802 void wxGridCornerLabelWindow::OnMouseEvent( wxMouseEvent
& event
)
804 m_owner
->ProcessCornerLabelMouseEvent( event
);
808 // This seems to be required for wxMotif otherwise the mouse
809 // cursor must be in the cell edit control to get key events
811 void wxGridCornerLabelWindow::OnKeyDown( wxKeyEvent
& event
)
813 if ( !m_owner
->ProcessEvent( event
) ) event
.Skip();
818 //////////////////////////////////////////////////////////////////////
820 IMPLEMENT_DYNAMIC_CLASS( wxGridWindow
, wxPanel
)
822 BEGIN_EVENT_TABLE( wxGridWindow
, wxPanel
)
823 EVT_PAINT( wxGridWindow::OnPaint
)
824 EVT_SCROLLWIN( wxGridWindow::ScrollWindow
)
825 EVT_MOUSE_EVENTS( wxGridWindow::OnMouseEvent
)
826 EVT_KEY_DOWN( wxGridWindow::OnKeyDown
)
829 wxGridWindow::wxGridWindow( wxGrid
*parent
,
830 wxGridRowLabelWindow
*rowLblWin
,
831 wxGridColLabelWindow
*colLblWin
,
832 wxWindowID id
, const wxPoint
&pos
, const wxSize
&size
)
833 : wxPanel( parent
, id
, pos
, size
, wxSUNKEN_BORDER
, "grid window" )
836 m_rowLabelWin
= rowLblWin
;
837 m_colLabelWin
= colLblWin
;
839 SetBackgroundColour( "WHITE" );
843 wxGridWindow::~wxGridWindow()
848 void wxGridWindow::OnPaint( wxPaintEvent
&WXUNUSED(event
) )
850 wxPaintDC
dc( this );
851 m_owner
->PrepareDC( dc
);
853 m_owner
->CalcCellsExposed( GetUpdateRegion() );
854 m_owner
->DrawGridCellArea( dc
);
858 void wxGridWindow::ScrollWindow( int dx
, int dy
, const wxRect
*rect
)
860 wxPanel::ScrollWindow( dx
, dy
, rect
);
861 m_rowLabelWin
->ScrollWindow( 0, dy
, rect
);
862 m_colLabelWin
->ScrollWindow( dx
, 0, rect
);
866 void wxGridWindow::OnMouseEvent( wxMouseEvent
& event
)
868 m_owner
->ProcessGridCellMouseEvent( event
);
872 // This seems to be required for wxMotif otherwise the mouse
873 // cursor must be in the cell edit control to get key events
875 void wxGridWindow::OnKeyDown( wxKeyEvent
& event
)
877 if ( !m_owner
->ProcessEvent( event
) ) event
.Skip();
882 //////////////////////////////////////////////////////////////////////
884 IMPLEMENT_DYNAMIC_CLASS( wxGrid
, wxScrolledWindow
)
886 BEGIN_EVENT_TABLE( wxGrid
, wxScrolledWindow
)
887 EVT_PAINT( wxGrid::OnPaint
)
888 EVT_SIZE( wxGrid::OnSize
)
889 EVT_KEY_DOWN( wxGrid::OnKeyDown
)
892 wxGrid::wxGrid( wxWindow
*parent
,
897 const wxString
& name
)
898 : wxScrolledWindow( parent
, id
, pos
, size
, style
, name
)
911 // ----- internal init and update functions
914 void wxGrid::Create()
916 int colLblH
= WXGRID_DEFAULT_COL_LABEL_HEIGHT
;
917 int rowLblW
= WXGRID_DEFAULT_ROW_LABEL_WIDTH
;
919 m_rowLabelWin
= new wxGridRowLabelWindow( this,
922 wxSize(rowLblW
,-1) );
924 m_colLabelWin
= new wxGridColLabelWindow( this,
927 wxSize(-1, colLblH
) );
929 m_cornerLabelWin
= new wxGridCornerLabelWindow( this,
932 wxSize(rowLblW
, colLblH
) );
934 m_gridWin
= new wxGridWindow( this,
941 SetTargetWindow( m_gridWin
);
943 m_mainSizer
= new wxBoxSizer( wxVERTICAL
);
945 m_topSizer
= new wxBoxSizer( wxHORIZONTAL
);
946 m_topSizer
->Add( m_cornerLabelWin
, 0 );
947 m_topSizer
->Add( m_colLabelWin
, 1 );
949 m_mainSizer
->Add( m_topSizer
, 0, wxEXPAND
);
951 m_middleSizer
= new wxBoxSizer( wxHORIZONTAL
);
952 m_middleSizer
->Add( m_rowLabelWin
, 0, wxEXPAND
);
953 m_middleSizer
->Add( m_gridWin
, 1, wxEXPAND
);
955 m_mainSizer
->Add( m_middleSizer
, 1, wxEXPAND
);
957 SetAutoLayout( TRUE
);
958 SetSizer( m_mainSizer
);
962 bool wxGrid::CreateGrid( int numRows
, int numCols
)
966 wxLogError( wxT("wxGrid::CreateGrid(numRows, numCols) called more than once") );
974 m_table
= new wxGridStringTable( m_numRows
, m_numCols
);
975 m_table
->SetView( this );
988 if ( m_numRows
<= 0 )
989 m_numRows
= WXGRID_DEFAULT_NUMBER_ROWS
;
991 if ( m_numCols
<= 0 )
992 m_numCols
= WXGRID_DEFAULT_NUMBER_COLS
;
994 m_rowLabelWidth
= WXGRID_DEFAULT_ROW_LABEL_WIDTH
;
995 m_colLabelHeight
= WXGRID_DEFAULT_COL_LABEL_HEIGHT
;
999 m_labelBackgroundColour
= m_rowLabelWin
->GetBackgroundColour();
1003 m_labelBackgroundColour
= wxColour( _T("WHITE") );
1006 m_labelTextColour
= wxColour( _T("BLACK") );
1008 // TODO: something better than this ?
1010 m_labelFont
= this->GetFont();
1011 m_labelFont
.SetWeight( m_labelFont
.GetWeight() + 2 );
1013 m_rowLabelHorizAlign
= wxLEFT
;
1014 m_rowLabelVertAlign
= wxCENTRE
;
1016 m_colLabelHorizAlign
= wxCENTRE
;
1017 m_colLabelVertAlign
= wxTOP
;
1019 m_defaultColWidth
= WXGRID_DEFAULT_COL_WIDTH
;
1020 m_defaultRowHeight
= m_gridWin
->GetCharHeight();
1022 #if defined (__WXMOTIF__) // see also text ctrl sizing in ShowCellEditControl()
1023 m_defaultRowHeight
+= 8;
1025 m_defaultRowHeight
+= 4;
1028 m_rowHeights
.Alloc( m_numRows
);
1029 m_rowBottoms
.Alloc( m_numRows
);
1031 for ( i
= 0; i
< m_numRows
; i
++ )
1033 m_rowHeights
.Add( m_defaultRowHeight
);
1034 rowBottom
+= m_defaultRowHeight
;
1035 m_rowBottoms
.Add( rowBottom
);
1038 m_colWidths
.Alloc( m_numCols
);
1039 m_colRights
.Alloc( m_numCols
);
1041 for ( i
= 0; i
< m_numCols
; i
++ )
1043 m_colWidths
.Add( m_defaultColWidth
);
1044 colRight
+= m_defaultColWidth
;
1045 m_colRights
.Add( colRight
);
1048 // TODO: improve this ?
1050 m_defaultCellFont
= this->GetFont();
1052 m_gridLineColour
= wxColour( 128, 128, 255 );
1053 m_gridLinesEnabled
= TRUE
;
1055 m_cursorMode
= WXGRID_CURSOR_SELECT_CELL
;
1057 m_dragRowOrCol
= -1;
1058 m_isDragging
= FALSE
;
1060 m_rowResizeCursor
= wxCursor( wxCURSOR_SIZENS
);
1061 m_colResizeCursor
= wxCursor( wxCURSOR_SIZEWE
);
1063 m_currentCellCoords
= wxGridNoCellCoords
;
1065 m_selectedTopLeft
= wxGridNoCellCoords
;
1066 m_selectedBottomRight
= wxGridNoCellCoords
;
1068 m_editable
= TRUE
; // default for whole grid
1070 m_inOnKeyDown
= FALSE
;
1073 // TODO: extend this to other types of controls
1075 m_cellEditCtrl
= new wxGridTextCtrl( m_gridWin
,
1083 , wxTE_MULTILINE
| wxTE_NO_VSCROLL
1087 m_cellEditCtrl
->Show( FALSE
);
1088 m_cellEditCtrlEnabled
= TRUE
;
1089 m_editCtrlType
= wxGRID_TEXTCTRL
;
1093 void wxGrid::CalcDimensions()
1096 GetClientSize( &cw
, &ch
);
1098 if ( m_numRows
> 0 && m_numCols
> 0 )
1100 int right
= m_colRights
[ m_numCols
-1 ] + 20;
1101 int bottom
= m_rowBottoms
[ m_numRows
-1 ] + 20;
1103 // TODO: restore the scroll position that we had before sizing
1106 GetViewStart( &x
, &y
);
1107 SetScrollbars( 10, 10,
1108 right
/10, bottom
/10,
1114 // this is called when the grid table sends a message to say that it
1115 // has been redimensioned
1117 bool wxGrid::Redimension( wxGridTableMessage
& msg
)
1121 switch ( msg
.GetId() )
1123 case wxGRIDTABLE_NOTIFY_ROWS_INSERTED
:
1125 size_t pos
= msg
.GetCommandInt();
1126 int numRows
= msg
.GetCommandInt2();
1127 for ( i
= 0; i
< numRows
; i
++ )
1129 m_rowHeights
.Insert( m_defaultRowHeight
, pos
);
1130 m_rowBottoms
.Insert( 0, pos
);
1132 m_numRows
+= numRows
;
1135 if ( pos
> 0 ) bottom
= m_rowBottoms
[pos
-1];
1137 for ( i
= pos
; i
< m_numRows
; i
++ )
1139 bottom
+= m_rowHeights
[i
];
1140 m_rowBottoms
[i
] = bottom
;
1146 case wxGRIDTABLE_NOTIFY_ROWS_APPENDED
:
1148 int numRows
= msg
.GetCommandInt();
1149 for ( i
= 0; i
< numRows
; i
++ )
1151 m_rowHeights
.Add( m_defaultRowHeight
);
1152 m_rowBottoms
.Add( 0 );
1155 int oldNumRows
= m_numRows
;
1156 m_numRows
+= numRows
;
1159 if ( oldNumRows
> 0 ) bottom
= m_rowBottoms
[oldNumRows
-1];
1161 for ( i
= oldNumRows
; i
< m_numRows
; i
++ )
1163 bottom
+= m_rowHeights
[i
];
1164 m_rowBottoms
[i
] = bottom
;
1170 case wxGRIDTABLE_NOTIFY_ROWS_DELETED
:
1172 size_t pos
= msg
.GetCommandInt();
1173 int numRows
= msg
.GetCommandInt2();
1174 for ( i
= 0; i
< numRows
; i
++ )
1176 m_rowHeights
.Remove( pos
);
1177 m_rowBottoms
.Remove( pos
);
1179 m_numRows
-= numRows
;
1184 m_colWidths
.Clear();
1185 m_colRights
.Clear();
1186 m_currentCellCoords
= wxGridNoCellCoords
;
1190 if ( m_currentCellCoords
.GetRow() >= m_numRows
)
1191 m_currentCellCoords
.Set( 0, 0 );
1194 for ( i
= 0; i
< m_numRows
; i
++ )
1196 h
+= m_rowHeights
[i
];
1197 m_rowBottoms
[i
] = h
;
1205 case wxGRIDTABLE_NOTIFY_COLS_INSERTED
:
1207 size_t pos
= msg
.GetCommandInt();
1208 int numCols
= msg
.GetCommandInt2();
1209 for ( i
= 0; i
< numCols
; i
++ )
1211 m_colWidths
.Insert( m_defaultColWidth
, pos
);
1212 m_colRights
.Insert( 0, pos
);
1214 m_numCols
+= numCols
;
1217 if ( pos
> 0 ) right
= m_colRights
[pos
-1];
1219 for ( i
= pos
; i
< m_numCols
; i
++ )
1221 right
+= m_colWidths
[i
];
1222 m_colRights
[i
] = right
;
1228 case wxGRIDTABLE_NOTIFY_COLS_APPENDED
:
1230 int numCols
= msg
.GetCommandInt();
1231 for ( i
= 0; i
< numCols
; i
++ )
1233 m_colWidths
.Add( m_defaultColWidth
);
1234 m_colRights
.Add( 0 );
1237 int oldNumCols
= m_numCols
;
1238 m_numCols
+= numCols
;
1241 if ( oldNumCols
> 0 ) right
= m_colRights
[oldNumCols
-1];
1243 for ( i
= oldNumCols
; i
< m_numCols
; i
++ )
1245 right
+= m_colWidths
[i
];
1246 m_colRights
[i
] = right
;
1252 case wxGRIDTABLE_NOTIFY_COLS_DELETED
:
1254 size_t pos
= msg
.GetCommandInt();
1255 int numCols
= msg
.GetCommandInt2();
1256 for ( i
= 0; i
< numCols
; i
++ )
1258 m_colWidths
.Remove( pos
);
1259 m_colRights
.Remove( pos
);
1261 m_numCols
-= numCols
;
1265 #if 0 // leave the row alone here so that AppendCols will work subsequently
1267 m_rowHeights
.Clear();
1268 m_rowBottoms
.Clear();
1270 m_currentCellCoords
= wxGridNoCellCoords
;
1274 if ( m_currentCellCoords
.GetCol() >= m_numCols
)
1275 m_currentCellCoords
.Set( 0, 0 );
1278 for ( i
= 0; i
< m_numCols
; i
++ )
1280 w
+= m_colWidths
[i
];
1293 void wxGrid::CalcRowLabelsExposed( wxRegion
& reg
)
1295 wxRegionIterator
iter( reg
);
1298 m_rowLabelsExposed
.Empty();
1305 // TODO: remove this when we can...
1306 // There is a bug in wxMotif that gives garbage update
1307 // rectangles if you jump-scroll a long way by clicking the
1308 // scrollbar with middle button. This is a work-around
1310 #if defined(__WXMOTIF__)
1312 m_gridWin
->GetClientSize( &cw
, &ch
);
1313 if ( r
.GetTop() > ch
) r
.SetTop( 0 );
1314 r
.SetBottom( wxMin( r
.GetBottom(), ch
) );
1317 // logical bounds of update region
1320 CalcUnscrolledPosition( 0, r
.GetTop(), &dummy
, &top
);
1321 CalcUnscrolledPosition( 0, r
.GetBottom(), &dummy
, &bottom
);
1323 // find the row labels within these bounds
1327 for ( row
= 0; row
< m_numRows
; row
++ )
1329 if ( m_rowBottoms
[row
] < top
) continue;
1331 rowTop
= m_rowBottoms
[row
] - m_rowHeights
[row
];
1332 if ( rowTop
> bottom
) break;
1334 m_rowLabelsExposed
.Add( row
);
1342 void wxGrid::CalcColLabelsExposed( wxRegion
& reg
)
1344 wxRegionIterator
iter( reg
);
1347 m_colLabelsExposed
.Empty();
1354 // TODO: remove this when we can...
1355 // There is a bug in wxMotif that gives garbage update
1356 // rectangles if you jump-scroll a long way by clicking the
1357 // scrollbar with middle button. This is a work-around
1359 #if defined(__WXMOTIF__)
1361 m_gridWin
->GetClientSize( &cw
, &ch
);
1362 if ( r
.GetLeft() > cw
) r
.SetLeft( 0 );
1363 r
.SetRight( wxMin( r
.GetRight(), cw
) );
1366 // logical bounds of update region
1369 CalcUnscrolledPosition( r
.GetLeft(), 0, &left
, &dummy
);
1370 CalcUnscrolledPosition( r
.GetRight(), 0, &right
, &dummy
);
1372 // find the cells within these bounds
1376 for ( col
= 0; col
< m_numCols
; col
++ )
1378 if ( m_colRights
[col
] < left
) continue;
1380 colLeft
= m_colRights
[col
] - m_colWidths
[col
];
1381 if ( colLeft
> right
) break;
1383 m_colLabelsExposed
.Add( col
);
1391 void wxGrid::CalcCellsExposed( wxRegion
& reg
)
1393 wxRegionIterator
iter( reg
);
1396 m_cellsExposed
.Empty();
1397 m_rowsExposed
.Empty();
1398 m_colsExposed
.Empty();
1400 int left
, top
, right
, bottom
;
1405 // TODO: remove this when we can...
1406 // There is a bug in wxMotif that gives garbage update
1407 // rectangles if you jump-scroll a long way by clicking the
1408 // scrollbar with middle button. This is a work-around
1410 #if defined(__WXMOTIF__)
1412 m_gridWin
->GetClientSize( &cw
, &ch
);
1413 if ( r
.GetTop() > ch
) r
.SetTop( 0 );
1414 if ( r
.GetLeft() > cw
) r
.SetLeft( 0 );
1415 r
.SetRight( wxMin( r
.GetRight(), cw
) );
1416 r
.SetBottom( wxMin( r
.GetBottom(), ch
) );
1419 // logical bounds of update region
1421 CalcUnscrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top
);
1422 CalcUnscrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom
);
1424 // find the cells within these bounds
1427 int colLeft
, rowTop
;
1428 for ( row
= 0; row
< m_numRows
; row
++ )
1430 if ( m_rowBottoms
[row
] < top
) continue;
1432 rowTop
= m_rowBottoms
[row
] - m_rowHeights
[row
];
1433 if ( rowTop
> bottom
) break;
1435 m_rowsExposed
.Add( row
);
1437 for ( col
= 0; col
< m_numCols
; col
++ )
1439 if ( m_colRights
[col
] < left
) continue;
1441 colLeft
= m_colRights
[col
] - m_colWidths
[col
];
1442 if ( colLeft
> right
) break;
1444 if ( m_colsExposed
.Index( col
) == wxNOT_FOUND
) m_colsExposed
.Add( col
);
1445 m_cellsExposed
.Add( wxGridCellCoords( row
, col
) );
1454 void wxGrid::ProcessRowLabelMouseEvent( wxMouseEvent
& event
)
1457 wxPoint
pos( event
.GetPosition() );
1458 CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y
);
1460 if ( event
.Dragging() )
1462 m_isDragging
= TRUE
;
1464 if ( event
.LeftIsDown() )
1466 switch( m_cursorMode
)
1468 case WXGRID_CURSOR_RESIZE_ROW
:
1470 int cw
, ch
, left
, dummy
;
1471 m_gridWin
->GetClientSize( &cw
, &ch
);
1472 CalcUnscrolledPosition( 0, 0, &left
, &dummy
);
1474 wxClientDC
dc( m_gridWin
);
1476 dc
.SetLogicalFunction(wxXOR
);
1477 if ( m_dragLastPos
>= 0 )
1479 dc
.DrawLine( left
, m_dragLastPos
, left
+cw
, m_dragLastPos
);
1481 dc
.DrawLine( left
, y
, left
+cw
, y
);
1486 case WXGRID_CURSOR_SELECT_ROW
:
1488 if ( (row
= YToRow( y
)) >= 0 &&
1489 !IsInSelection( row
, 0 ) )
1491 SelectRow( row
, TRUE
);
1500 m_isDragging
= FALSE
;
1503 // ------------ Left button pressed
1505 if ( event
.LeftDown() )
1507 // don't send a label click event for a hit on the
1508 // edge of the row label - this is probably the user
1509 // wanting to resize the row
1511 if ( YToEdgeOfRow(y
) < 0 )
1515 !SendEvent( EVT_GRID_LABEL_LEFT_CLICK
, row
, -1, event
) )
1517 SelectRow( row
, event
.ShiftDown() );
1518 m_cursorMode
= WXGRID_CURSOR_SELECT_ROW
;
1523 // starting to drag-resize a row
1525 m_rowLabelWin
->CaptureMouse();
1530 // ------------ Left double click
1532 else if (event
.LeftDClick() )
1534 if ( YToEdgeOfRow(y
) < 0 )
1537 SendEvent( EVT_GRID_LABEL_LEFT_DCLICK
, row
, -1, event
);
1542 // ------------ Left button released
1544 else if ( event
.LeftUp() )
1546 if ( m_cursorMode
== WXGRID_CURSOR_RESIZE_ROW
)
1548 m_rowLabelWin
->ReleaseMouse();
1550 if ( m_dragLastPos
>= 0 )
1552 // erase the last line and resize the row
1554 int cw
, ch
, left
, dummy
;
1555 m_gridWin
->GetClientSize( &cw
, &ch
);
1556 CalcUnscrolledPosition( 0, 0, &left
, &dummy
);
1558 wxClientDC
dc( m_gridWin
);
1560 dc
.SetLogicalFunction( wxINVERT
);
1561 dc
.DrawLine( left
, m_dragLastPos
, left
+cw
, m_dragLastPos
);
1562 HideCellEditControl();
1564 int rowTop
= m_rowBottoms
[m_dragRowOrCol
] - m_rowHeights
[m_dragRowOrCol
];
1565 SetRowSize( m_dragRowOrCol
, wxMax( y
- rowTop
, WXGRID_MIN_ROW_HEIGHT
) );
1566 if ( !GetBatchCount() )
1568 // TODO: optimize this
1569 m_rowLabelWin
->Refresh();
1570 m_gridWin
->Refresh();
1573 ShowCellEditControl();
1575 // Note: we are ending the event *after* doing
1576 // default processing in this case
1578 SendEvent( EVT_GRID_ROW_SIZE
, m_dragRowOrCol
, -1, event
);
1586 // ------------ Right button down
1588 else if ( event
.RightDown() )
1591 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_CLICK
, row
, -1, event
) )
1593 // no default action at the moment
1598 // ------------ Right double click
1600 else if ( event
.RightDClick() )
1603 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_DCLICK
, row
, -1, event
) )
1605 // no default action at the moment
1610 // ------------ No buttons down and mouse moving
1612 else if ( event
.Moving() )
1614 m_dragRowOrCol
= YToEdgeOfRow( y
);
1615 if ( m_dragRowOrCol
>= 0 )
1617 if ( m_cursorMode
== WXGRID_CURSOR_SELECT_CELL
)
1619 m_cursorMode
= WXGRID_CURSOR_RESIZE_ROW
;
1620 m_rowLabelWin
->SetCursor( m_rowResizeCursor
);
1625 if ( m_cursorMode
!= WXGRID_CURSOR_SELECT_CELL
)
1627 m_cursorMode
= WXGRID_CURSOR_SELECT_CELL
;
1628 m_rowLabelWin
->SetCursor( *wxSTANDARD_CURSOR
);
1635 void wxGrid::ProcessColLabelMouseEvent( wxMouseEvent
& event
)
1638 wxPoint
pos( event
.GetPosition() );
1639 CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y
);
1641 if ( event
.Dragging() )
1643 m_isDragging
= TRUE
;
1645 if ( event
.LeftIsDown() )
1647 switch( m_cursorMode
)
1649 case WXGRID_CURSOR_RESIZE_COL
:
1651 int cw
, ch
, dummy
, top
;
1652 m_gridWin
->GetClientSize( &cw
, &ch
);
1653 CalcUnscrolledPosition( 0, 0, &dummy
, &top
);
1655 wxClientDC
dc( m_gridWin
);
1657 dc
.SetLogicalFunction(wxXOR
);
1658 if ( m_dragLastPos
>= 0 )
1660 dc
.DrawLine( m_dragLastPos
, top
, m_dragLastPos
, top
+ch
);
1662 dc
.DrawLine( x
, top
, x
, top
+ch
);
1667 case WXGRID_CURSOR_SELECT_COL
:
1669 if ( (col
= XToCol( x
)) >= 0 &&
1670 !IsInSelection( 0, col
) )
1672 SelectCol( col
, TRUE
);
1681 m_isDragging
= FALSE
;
1684 // ------------ Left button pressed
1686 if ( event
.LeftDown() )
1688 // don't send a label click event for a hit on the
1689 // edge of the col label - this is probably the user
1690 // wanting to resize the col
1692 if ( XToEdgeOfCol(x
) < 0 )
1696 !SendEvent( EVT_GRID_LABEL_LEFT_CLICK
, -1, col
, event
) )
1698 SelectCol( col
, event
.ShiftDown() );
1699 m_cursorMode
= WXGRID_CURSOR_SELECT_COL
;
1704 // starting to drag-resize a col
1706 m_colLabelWin
->CaptureMouse();
1711 // ------------ Left double click
1713 if ( event
.LeftDClick() )
1715 if ( XToEdgeOfCol(x
) < 0 )
1718 SendEvent( EVT_GRID_LABEL_LEFT_DCLICK
, -1, col
, event
);
1723 // ------------ Left button released
1725 else if ( event
.LeftUp() )
1727 if ( m_cursorMode
== WXGRID_CURSOR_RESIZE_COL
)
1729 m_colLabelWin
->ReleaseMouse();
1731 if ( m_dragLastPos
>= 0 )
1733 // erase the last line and resize the col
1735 int cw
, ch
, dummy
, top
;
1736 m_gridWin
->GetClientSize( &cw
, &ch
);
1737 CalcUnscrolledPosition( 0, 0, &dummy
, &top
);
1739 wxClientDC
dc( m_gridWin
);
1741 dc
.SetLogicalFunction( wxINVERT
);
1742 dc
.DrawLine( m_dragLastPos
, top
, m_dragLastPos
, top
+ch
);
1743 HideCellEditControl();
1745 int colLeft
= m_colRights
[m_dragRowOrCol
] - m_colWidths
[m_dragRowOrCol
];
1746 SetColSize( m_dragRowOrCol
, wxMax( x
- colLeft
, WXGRID_MIN_COL_WIDTH
) );
1748 if ( !GetBatchCount() )
1750 // TODO: optimize this
1751 m_colLabelWin
->Refresh();
1752 m_gridWin
->Refresh();
1755 ShowCellEditControl();
1757 // Note: we are ending the event *after* doing
1758 // default processing in this case
1760 SendEvent( EVT_GRID_COL_SIZE
, -1, m_dragRowOrCol
, event
);
1768 // ------------ Right button down
1770 else if ( event
.RightDown() )
1773 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_CLICK
, -1, col
, event
) )
1775 // no default action at the moment
1780 // ------------ Right double click
1782 else if ( event
.RightDClick() )
1785 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_DCLICK
, -1, col
, event
) )
1787 // no default action at the moment
1792 // ------------ No buttons down and mouse moving
1794 else if ( event
.Moving() )
1796 m_dragRowOrCol
= XToEdgeOfCol( x
);
1797 if ( m_dragRowOrCol
>= 0 )
1799 if ( m_cursorMode
== WXGRID_CURSOR_SELECT_CELL
)
1801 m_cursorMode
= WXGRID_CURSOR_RESIZE_COL
;
1802 m_colLabelWin
->SetCursor( m_colResizeCursor
);
1807 if ( m_cursorMode
!= WXGRID_CURSOR_SELECT_CELL
)
1809 m_cursorMode
= WXGRID_CURSOR_SELECT_CELL
;
1810 m_colLabelWin
->SetCursor( *wxSTANDARD_CURSOR
);
1817 void wxGrid::ProcessCornerLabelMouseEvent( wxMouseEvent
& event
)
1819 if ( event
.LeftDown() )
1821 // indicate corner label by having both row and
1824 if ( !SendEvent( EVT_GRID_LABEL_LEFT_CLICK
, -1, -1, event
) )
1830 else if ( event
.LeftDClick() )
1832 SendEvent( EVT_GRID_LABEL_LEFT_DCLICK
, -1, -1, event
);
1835 else if ( event
.RightDown() )
1837 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_CLICK
, -1, -1, event
) )
1839 // no default action at the moment
1843 else if ( event
.RightDClick() )
1845 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_DCLICK
, -1, -1, event
) )
1847 // no default action at the moment
1853 void wxGrid::ProcessGridCellMouseEvent( wxMouseEvent
& event
)
1856 wxPoint
pos( event
.GetPosition() );
1857 CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y
);
1859 wxGridCellCoords coords
;
1860 XYToCell( x
, y
, coords
);
1862 if ( event
.Dragging() )
1864 m_isDragging
= TRUE
;
1865 if ( m_cursorMode
== WXGRID_CURSOR_SELECT_CELL
)
1867 // Hide the edit control, so it
1868 // won't interfer with drag-shrinking.
1869 if ( IsCellEditControlEnabled() )
1870 HideCellEditControl();
1871 if ( coords
!= wxGridNoCellCoords
)
1873 if ( !IsSelection() )
1875 SelectBlock( coords
, coords
);
1879 SelectBlock( m_currentCellCoords
, coords
);
1887 m_isDragging
= FALSE
;
1889 if ( coords
!= wxGridNoCellCoords
)
1891 if ( event
.LeftDown() )
1893 if ( event
.ShiftDown() )
1895 SelectBlock( m_currentCellCoords
, coords
);
1899 if ( !SendEvent( EVT_GRID_CELL_LEFT_CLICK
,
1904 MakeCellVisible( coords
);
1905 SetCurrentCell( coords
);
1911 // ------------ Left double click
1913 else if ( event
.LeftDClick() )
1915 SendEvent( EVT_GRID_CELL_LEFT_DCLICK
,
1922 // ------------ Left button released
1924 else if ( event
.LeftUp() )
1926 if ( m_cursorMode
== WXGRID_CURSOR_SELECT_CELL
)
1928 if ( IsSelection() )
1930 SendEvent( EVT_GRID_RANGE_SELECT
, -1, -1, event
);
1934 // Show the edit control, if it has
1935 // been hidden for drag-shrinking.
1936 if ( IsCellEditControlEnabled() )
1937 ShowCellEditControl();
1943 // ------------ Right button down
1945 else if ( event
.RightDown() )
1947 if ( !SendEvent( EVT_GRID_CELL_RIGHT_CLICK
,
1952 // no default action at the moment
1957 // ------------ Right double click
1959 else if ( event
.RightDClick() )
1961 if ( !SendEvent( EVT_GRID_CELL_RIGHT_DCLICK
,
1966 // no default action at the moment
1974 // ------ interaction with data model
1976 bool wxGrid::ProcessTableMessage( wxGridTableMessage
& msg
)
1978 switch ( msg
.GetId() )
1980 case wxGRIDTABLE_REQUEST_VIEW_GET_VALUES
:
1981 return GetModelValues();
1983 case wxGRIDTABLE_REQUEST_VIEW_SEND_VALUES
:
1984 return SetModelValues();
1986 case wxGRIDTABLE_NOTIFY_ROWS_INSERTED
:
1987 case wxGRIDTABLE_NOTIFY_ROWS_APPENDED
:
1988 case wxGRIDTABLE_NOTIFY_ROWS_DELETED
:
1989 case wxGRIDTABLE_NOTIFY_COLS_INSERTED
:
1990 case wxGRIDTABLE_NOTIFY_COLS_APPENDED
:
1991 case wxGRIDTABLE_NOTIFY_COLS_DELETED
:
1992 return Redimension( msg
);
2001 // The behaviour of this function depends on the grid table class
2002 // Clear() function. For the default wxGridStringTable class the
2003 // behavious is to replace all cell contents with wxEmptyString but
2004 // not to change the number of rows or cols.
2006 void wxGrid::ClearGrid()
2011 SetEditControlValue();
2012 if ( !GetBatchCount() ) m_gridWin
->Refresh();
2017 bool wxGrid::InsertRows( int pos
, int numRows
, bool WXUNUSED(updateLabels
) )
2019 // TODO: something with updateLabels flag
2023 wxLogError( wxT("Called wxGrid::InsertRows() before calling CreateGrid()") );
2029 bool ok
= m_table
->InsertRows( pos
, numRows
);
2031 // the table will have sent the results of the insert row
2032 // operation to this view object as a grid table message
2036 if ( m_numCols
== 0 )
2038 m_table
->AppendCols( WXGRID_DEFAULT_NUMBER_COLS
);
2040 // TODO: perhaps instead of appending the default number of cols
2041 // we should remember what the last non-zero number of cols was ?
2045 if ( m_currentCellCoords
== wxGridNoCellCoords
)
2047 // if we have just inserted cols into an empty grid the current
2048 // cell will be undefined...
2050 SetCurrentCell( 0, 0 );
2054 if ( !GetBatchCount() ) Refresh();
2057 SetEditControlValue();
2067 bool wxGrid::AppendRows( int numRows
, bool WXUNUSED(updateLabels
) )
2069 // TODO: something with updateLabels flag
2073 wxLogError( wxT("Called wxGrid::AppendRows() before calling CreateGrid()") );
2077 if ( m_table
&& m_table
->AppendRows( numRows
) )
2079 if ( m_currentCellCoords
== wxGridNoCellCoords
)
2081 // if we have just inserted cols into an empty grid the current
2082 // cell will be undefined...
2084 SetCurrentCell( 0, 0 );
2087 // the table will have sent the results of the append row
2088 // operation to this view object as a grid table message
2091 if ( !GetBatchCount() ) Refresh();
2101 bool wxGrid::DeleteRows( int pos
, int numRows
, bool WXUNUSED(updateLabels
) )
2103 // TODO: something with updateLabels flag
2107 wxLogError( wxT("Called wxGrid::DeleteRows() before calling CreateGrid()") );
2111 if ( m_table
&& m_table
->DeleteRows( pos
, numRows
) )
2113 // the table will have sent the results of the delete row
2114 // operation to this view object as a grid table message
2116 if ( m_numRows
> 0 )
2117 SetEditControlValue();
2119 HideCellEditControl();
2122 if ( !GetBatchCount() ) Refresh();
2132 bool wxGrid::InsertCols( int pos
, int numCols
, bool WXUNUSED(updateLabels
) )
2134 // TODO: something with updateLabels flag
2138 wxLogError( wxT("Called wxGrid::InsertCols() before calling CreateGrid()") );
2144 HideCellEditControl();
2145 bool ok
= m_table
->InsertCols( pos
, numCols
);
2147 // the table will have sent the results of the insert col
2148 // operation to this view object as a grid table message
2152 if ( m_currentCellCoords
== wxGridNoCellCoords
)
2154 // if we have just inserted cols into an empty grid the current
2155 // cell will be undefined...
2157 SetCurrentCell( 0, 0 );
2161 if ( !GetBatchCount() ) Refresh();
2164 SetEditControlValue();
2174 bool wxGrid::AppendCols( int numCols
, bool WXUNUSED(updateLabels
) )
2176 // TODO: something with updateLabels flag
2180 wxLogError( wxT("Called wxGrid::AppendCols() before calling CreateGrid()") );
2184 if ( m_table
&& m_table
->AppendCols( numCols
) )
2186 // the table will have sent the results of the append col
2187 // operation to this view object as a grid table message
2189 if ( m_currentCellCoords
== wxGridNoCellCoords
)
2191 // if we have just inserted cols into an empty grid the current
2192 // cell will be undefined...
2194 SetCurrentCell( 0, 0 );
2198 if ( !GetBatchCount() ) Refresh();
2208 bool wxGrid::DeleteCols( int pos
, int numCols
, bool WXUNUSED(updateLabels
) )
2210 // TODO: something with updateLabels flag
2214 wxLogError( wxT("Called wxGrid::DeleteCols() before calling CreateGrid()") );
2218 if ( m_table
&& m_table
->DeleteCols( pos
, numCols
) )
2220 // the table will have sent the results of the delete col
2221 // operation to this view object as a grid table message
2223 if ( m_numCols
> 0 )
2224 SetEditControlValue();
2226 HideCellEditControl();
2229 if ( !GetBatchCount() ) Refresh();
2241 // ----- event handlers
2244 // Generate a grid event based on a mouse event and
2245 // return the result of ProcessEvent()
2247 bool wxGrid::SendEvent( const wxEventType type
,
2249 wxMouseEvent
& mouseEv
)
2251 if ( type
== EVT_GRID_ROW_SIZE
||
2252 type
== EVT_GRID_COL_SIZE
)
2254 int rowOrCol
= (row
== -1 ? col
: row
);
2256 wxGridSizeEvent
gridEvt( GetId(),
2260 mouseEv
.GetX(), mouseEv
.GetY(),
2261 mouseEv
.ControlDown(),
2262 mouseEv
.ShiftDown(),
2264 mouseEv
.MetaDown() );
2266 return GetEventHandler()->ProcessEvent(gridEvt
);
2268 else if ( type
== EVT_GRID_RANGE_SELECT
)
2270 wxGridRangeSelectEvent
gridEvt( GetId(),
2274 m_selectedBottomRight
,
2275 mouseEv
.ControlDown(),
2276 mouseEv
.ShiftDown(),
2278 mouseEv
.MetaDown() );
2280 return GetEventHandler()->ProcessEvent(gridEvt
);
2284 wxGridEvent
gridEvt( GetId(),
2288 mouseEv
.GetX(), mouseEv
.GetY(),
2289 mouseEv
.ControlDown(),
2290 mouseEv
.ShiftDown(),
2292 mouseEv
.MetaDown() );
2294 return GetEventHandler()->ProcessEvent(gridEvt
);
2299 // Generate a grid event of specified type and return the result
2300 // of ProcessEvent().
2302 bool wxGrid::SendEvent( const wxEventType type
,
2305 if ( type
== EVT_GRID_ROW_SIZE
||
2306 type
== EVT_GRID_COL_SIZE
)
2308 int rowOrCol
= (row
== -1 ? col
: row
);
2310 wxGridSizeEvent
gridEvt( GetId(),
2315 return GetEventHandler()->ProcessEvent(gridEvt
);
2319 wxGridEvent
gridEvt( GetId(),
2324 return GetEventHandler()->ProcessEvent(gridEvt
);
2329 void wxGrid::OnPaint( wxPaintEvent
& WXUNUSED(event
) )
2331 wxPaintDC
dc( this );
2333 if ( m_currentCellCoords
== wxGridNoCellCoords
&&
2334 m_numRows
&& m_numCols
)
2336 m_currentCellCoords
.Set(0, 0);
2337 SetEditControlValue();
2338 ShowCellEditControl();
2343 // This is just here to make sure that CalcDimensions gets called when
2344 // the grid view is resized... then the size event is skipped to allow
2345 // the box sizers to handle everything
2347 void wxGrid::OnSize( wxSizeEvent
& event
)
2354 void wxGrid::OnKeyDown( wxKeyEvent
& event
)
2356 if ( m_inOnKeyDown
)
2358 // shouldn't be here - we are going round in circles...
2360 wxLogFatalError( wxT("wxGrid::OnKeyDown called while alread active") );
2363 m_inOnKeyDown
= TRUE
;
2365 // propagate the event up and see if it gets processed
2367 wxWindow
*parent
= GetParent();
2368 wxKeyEvent
keyEvt( event
);
2369 keyEvt
.SetEventObject( parent
);
2371 if ( !parent
->GetEventHandler()->ProcessEvent( keyEvt
) )
2373 // try local handlers
2375 switch ( event
.KeyCode() )
2378 if ( event
.ControlDown() )
2380 MoveCursorUpBlock();
2389 if ( event
.ControlDown() )
2391 MoveCursorDownBlock();
2400 if ( event
.ControlDown() )
2402 MoveCursorLeftBlock();
2411 if ( event
.ControlDown() )
2413 MoveCursorRightBlock();
2422 if ( event
.ControlDown() )
2424 event
.Skip(); // to let the edit control have the return
2433 if ( event
.ControlDown() )
2435 MakeCellVisible( 0, 0 );
2436 SetCurrentCell( 0, 0 );
2445 if ( event
.ControlDown() )
2447 MakeCellVisible( m_numRows
-1, m_numCols
-1 );
2448 SetCurrentCell( m_numRows
-1, m_numCols
-1 );
2465 // now try the cell edit control
2467 if ( IsCellEditControlEnabled() )
2469 event
.SetEventObject( m_cellEditCtrl
);
2470 m_cellEditCtrl
->GetEventHandler()->ProcessEvent( event
);
2476 m_inOnKeyDown
= FALSE
;
2480 void wxGrid::SetCurrentCell( const wxGridCellCoords
& coords
)
2482 if ( SendEvent( EVT_GRID_SELECT_CELL
, coords
.GetRow(), coords
.GetCol() ) )
2484 // the event has been intercepted - do nothing
2488 wxClientDC
dc( m_gridWin
);
2491 if ( m_currentCellCoords
!= wxGridNoCellCoords
)
2493 HideCellEditControl();
2494 SaveEditControlValue();
2497 m_currentCellCoords
= coords
;
2499 SetEditControlValue();
2500 ShowCellEditControl();
2502 if ( IsSelection() )
2504 wxRect
r( SelectionToDeviceRect() );
2506 if ( !GetBatchCount() ) m_gridWin
->Refresh( TRUE
, &r
);
2512 // ------ functions to get/send data (see also public functions)
2515 bool wxGrid::GetModelValues()
2519 // all we need to do is repaint the grid
2521 m_gridWin
->Refresh();
2529 bool wxGrid::SetModelValues()
2535 for ( row
= 0; row
< m_numRows
; row
++ )
2537 for ( col
= 0; col
< m_numCols
; col
++ )
2539 m_table
->SetValue( row
, col
, GetCellValue(row
, col
) );
2551 // Note - this function only draws cells that are in the list of
2552 // exposed cells (usually set from the update region by
2553 // CalcExposedCells)
2555 void wxGrid::DrawGridCellArea( wxDC
& dc
)
2557 if ( !m_numRows
|| !m_numCols
) return;
2560 size_t numCells
= m_cellsExposed
.GetCount();
2562 for ( i
= 0; i
< numCells
; i
++ )
2564 DrawCell( dc
, m_cellsExposed
[i
] );
2569 void wxGrid::DrawCell( wxDC
& dc
, const wxGridCellCoords
& coords
)
2571 if ( m_colWidths
[coords
.GetCol()] <=0 ||
2572 m_rowHeights
[coords
.GetRow()] <= 0 ) return;
2574 if ( m_gridLinesEnabled
)
2575 DrawCellBorder( dc
, coords
);
2577 DrawCellBackground( dc
, coords
);
2579 // TODO: separate functions here for different kinds of cells ?
2582 DrawCellValue( dc
, coords
);
2586 void wxGrid::DrawCellBorder( wxDC
& dc
, const wxGridCellCoords
& coords
)
2588 if ( m_colWidths
[coords
.GetCol()] <=0 ||
2589 m_rowHeights
[coords
.GetRow()] <= 0 ) return;
2591 dc
.SetPen( wxPen(GetGridLineColour(), 1, wxSOLID
) );
2592 int row
= coords
.GetRow();
2593 int col
= coords
.GetCol();
2595 // right hand border
2597 dc
.DrawLine( m_colRights
[col
], m_rowBottoms
[row
] - m_rowHeights
[row
],
2598 m_colRights
[col
], m_rowBottoms
[row
] );
2602 dc
.DrawLine( m_colRights
[col
] - m_colWidths
[col
], m_rowBottoms
[row
],
2603 m_colRights
[col
], m_rowBottoms
[row
] );
2607 void wxGrid::DrawCellBackground( wxDC
& dc
, const wxGridCellCoords
& coords
)
2609 if ( m_colWidths
[coords
.GetCol()] <=0 ||
2610 m_rowHeights
[coords
.GetRow()] <= 0 ) return;
2612 int row
= coords
.GetRow();
2613 int col
= coords
.GetCol();
2615 dc
.SetBackgroundMode( wxSOLID
);
2617 if ( IsInSelection( coords
) )
2619 // TODO: improve this
2621 dc
.SetBrush( *wxBLACK_BRUSH
);
2625 dc
.SetBrush( wxBrush(GetCellBackgroundColour(row
, col
), wxSOLID
) );
2628 dc
.SetPen( *wxTRANSPARENT_PEN
);
2630 dc
.DrawRectangle( m_colRights
[col
] - m_colWidths
[col
] + 1,
2631 m_rowBottoms
[row
] - m_rowHeights
[row
] + 1,
2633 m_rowHeights
[row
]-1 );
2637 void wxGrid::DrawCellValue( wxDC
& dc
, const wxGridCellCoords
& coords
)
2639 if ( m_colWidths
[coords
.GetCol()] <=0 ||
2640 m_rowHeights
[coords
.GetRow()] <= 0 ) return;
2642 int row
= coords
.GetRow();
2643 int col
= coords
.GetCol();
2645 dc
.SetBackgroundMode( wxTRANSPARENT
);
2647 if ( IsInSelection( row
, col
) )
2649 // TODO: improve this
2651 dc
.SetTextBackground( wxColour(0, 0, 0) );
2652 dc
.SetTextForeground( wxColour(255, 255, 255) );
2656 dc
.SetTextBackground( GetCellBackgroundColour(row
, col
) );
2657 dc
.SetTextForeground( GetCellTextColour(row
, col
) );
2659 dc
.SetFont( GetCellFont(row
, col
) );
2662 GetCellAlignment( row
, col
, &hAlign
, &vAlign
);
2665 rect
.SetX( m_colRights
[col
] - m_colWidths
[col
] + 2 );
2666 rect
.SetY( m_rowBottoms
[row
] - m_rowHeights
[row
] + 2 );
2667 rect
.SetWidth( m_colWidths
[col
] - 4 );
2668 rect
.SetHeight( m_rowHeights
[row
] - 4 );
2670 DrawTextRectangle( dc
, GetCellValue( row
, col
), rect
, hAlign
, vAlign
);
2675 // TODO: remove this ???
2676 // This is used to redraw all grid lines e.g. when the grid line colour
2679 void wxGrid::DrawAllGridLines( wxDC
& dc
)
2681 if ( !m_gridLinesEnabled
||
2683 !m_numCols
) return;
2686 m_gridWin
->GetClientSize(&cw
, &ch
);
2688 // virtual coords of visible area
2690 int top
, bottom
, left
, right
;
2691 CalcUnscrolledPosition( 0, 0, &left
, &top
);
2692 CalcUnscrolledPosition( cw
, ch
, &right
, &bottom
);
2694 dc
.SetPen( wxPen(GetGridLineColour(), 1, wxSOLID
) );
2696 // horizontal grid lines
2699 for ( i
= 0; i
<= m_numRows
; i
++ )
2701 if ( m_rowBottoms
[i
] > bottom
)
2705 else if ( m_rowBottoms
[i
] >= top
)
2707 dc
.DrawLine( left
, m_rowBottoms
[i
], right
, m_rowBottoms
[i
] );
2712 // vertical grid lines
2714 for ( i
= 0; i
<= m_numCols
; i
++ )
2716 if ( m_colRights
[i
] > right
)
2720 else if ( m_colRights
[i
] >= left
)
2722 dc
.DrawLine( m_colRights
[i
], top
, m_colRights
[i
], bottom
);
2728 void wxGrid::DrawRowLabels( wxDC
& dc
)
2730 if ( !m_numRows
|| !m_numCols
) return;
2733 size_t numLabels
= m_rowLabelsExposed
.GetCount();
2735 for ( i
= 0; i
< numLabels
; i
++ )
2737 DrawRowLabel( dc
, m_rowLabelsExposed
[i
] );
2742 void wxGrid::DrawRowLabel( wxDC
& dc
, int row
)
2744 if ( m_rowHeights
[row
] <= 0 ) return;
2746 // draw the label's horizontal border (the vertical border is
2747 // provided by the cell area window margin)
2749 dc
.SetPen( *wxBLACK_PEN
);
2751 dc
.DrawLine( 0, m_rowBottoms
[row
]+1,
2752 m_rowLabelWidth
, m_rowBottoms
[row
]+1 );
2754 dc
.SetPen( *wxWHITE_PEN
);
2756 dc
.DrawLine( 0, m_rowBottoms
[row
]+2,
2757 m_rowLabelWidth
, m_rowBottoms
[row
]+2 );
2759 dc
.SetBackgroundMode( wxTRANSPARENT
);
2760 dc
.SetTextForeground( GetLabelTextColour() );
2761 dc
.SetFont( GetLabelFont() );
2764 GetRowLabelAlignment( &hAlign
, &vAlign
);
2768 rect
.SetY( m_rowBottoms
[row
] - m_rowHeights
[row
] + 2 );
2769 rect
.SetWidth( m_rowLabelWidth
- 4 );
2770 rect
.SetHeight( m_rowHeights
[row
] - 4 );
2771 DrawTextRectangle( dc
, GetRowLabelValue( row
), rect
, hAlign
, vAlign
);
2775 void wxGrid::DrawColLabels( wxDC
& dc
)
2777 if ( !m_numRows
|| !m_numCols
) return;
2780 size_t numLabels
= m_colLabelsExposed
.GetCount();
2782 for ( i
= 0; i
< numLabels
; i
++ )
2784 DrawColLabel( dc
, m_colLabelsExposed
[i
] );
2789 void wxGrid::DrawColLabel( wxDC
& dc
, int col
)
2791 if ( m_colWidths
[col
] <= 0 ) return;
2793 // draw the label's vertical border (the horizontal border is
2794 // provided by the cell area window margin)
2796 dc
.SetPen( *wxBLACK_PEN
);
2798 dc
.DrawLine( m_colRights
[col
]+1, 0,
2799 m_colRights
[col
]+1, m_colLabelHeight
);
2801 dc
.SetPen( *wxWHITE_PEN
);
2803 dc
.DrawLine( m_colRights
[col
]+2, 0,
2804 m_colRights
[col
]+2, m_colLabelHeight
);
2806 dc
.SetBackgroundMode( wxTRANSPARENT
);
2807 dc
.SetTextForeground( GetLabelTextColour() );
2808 dc
.SetFont( GetLabelFont() );
2811 GetColLabelAlignment( &hAlign
, &vAlign
);
2814 rect
.SetX( m_colRights
[col
] - m_colWidths
[col
] + 2 );
2816 rect
.SetWidth( m_colWidths
[col
] - 4 );
2817 rect
.SetHeight( m_colLabelHeight
- 4 );
2818 DrawTextRectangle( dc
, GetColLabelValue( col
), rect
, hAlign
, vAlign
);
2822 void wxGrid::DrawTextRectangle( wxDC
& dc
,
2823 const wxString
& value
,
2828 long textWidth
, textHeight
;
2829 long lineWidth
, lineHeight
;
2830 wxArrayString lines
;
2832 dc
.SetClippingRegion( rect
);
2833 StringToLines( value
, lines
);
2834 if ( lines
.GetCount() )
2836 GetTextBoxSize( dc
, lines
, &textWidth
, &textHeight
);
2837 dc
.GetTextExtent( lines
[0], &lineWidth
, &lineHeight
);
2840 switch ( horizAlign
)
2843 x
= rect
.x
+ (rect
.width
- textWidth
- 1);
2847 x
= rect
.x
+ ((rect
.width
- textWidth
)/2);
2856 switch ( vertAlign
)
2859 y
= rect
.y
+ (rect
.height
- textHeight
- 1);
2863 y
= rect
.y
+ ((rect
.height
- textHeight
)/2);
2872 for ( size_t i
= 0; i
< lines
.GetCount(); i
++ )
2874 dc
.DrawText( lines
[i
], (long)x
, (long)y
);
2879 dc
.DestroyClippingRegion();
2883 // Split multi line text up into an array of strings. Any existing
2884 // contents of the string array are preserved.
2886 void wxGrid::StringToLines( const wxString
& value
, wxArrayString
& lines
)
2888 // TODO: this won't work for WXMAC ? (lines end with '\r')
2889 // => use wxTextFile functions then (VZ)
2892 while ( startPos
< (int)value
.Length() )
2894 pos
= value
.Mid(startPos
).Find( '\n' );
2899 else if ( pos
== 0 )
2901 lines
.Add( wxEmptyString
);
2905 if ( value
[startPos
+pos
-1] == '\r' )
2907 lines
.Add( value
.Mid(startPos
, pos
-1) );
2911 lines
.Add( value
.Mid(startPos
, pos
) );
2916 if ( startPos
< (int)value
.Length() )
2918 lines
.Add( value
.Mid( startPos
) );
2923 void wxGrid::GetTextBoxSize( wxDC
& dc
,
2924 wxArrayString
& lines
,
2925 long *width
, long *height
)
2932 for ( i
= 0; i
< lines
.GetCount(); i
++ )
2934 dc
.GetTextExtent( lines
[i
], &lineW
, &lineH
);
2935 w
= wxMax( w
, lineW
);
2945 // ------ Edit control functions
2949 void wxGrid::EnableEditing( bool edit
)
2951 // TODO: improve this ?
2953 if ( edit
!= m_editable
)
2957 // TODO: extend this for other edit control types
2959 if ( m_editCtrlType
== wxGRID_TEXTCTRL
)
2961 ((wxTextCtrl
*)m_cellEditCtrl
)->SetEditable( m_editable
);
2967 #if 0 // disabled for the moment - the cell control is always active
2968 void wxGrid::EnableCellEditControl( bool enable
)
2970 if ( m_cellEditCtrl
&&
2971 enable
!= m_cellEditCtrlEnabled
)
2973 m_cellEditCtrlEnabled
= enable
;
2975 if ( m_cellEditCtrlEnabled
)
2977 SetEditControlValue();
2978 ShowCellEditControl();
2982 HideCellEditControl();
2983 SaveEditControlValue();
2990 void wxGrid::ShowCellEditControl()
2994 if ( IsCellEditControlEnabled() )
2996 if ( !IsVisible( m_currentCellCoords
) )
3002 rect
= CellToRect( m_currentCellCoords
);
3004 // convert to scrolled coords
3006 int left
, top
, right
, bottom
;
3007 CalcScrolledPosition( rect
.GetLeft(), rect
.GetTop(), &left
, &top
);
3008 CalcScrolledPosition( rect
.GetRight(), rect
.GetBottom(), &right
, &bottom
);
3011 m_gridWin
->GetClientSize( &cw
, &ch
);
3013 // Make the edit control large enough to allow for internal margins
3014 // TODO: remove this if the text ctrl sizing is improved esp. for unix
3017 #if defined (__WXMOTIF__)
3018 if ( m_currentCellCoords
.GetRow() == 0 ||
3019 m_currentCellCoords
.GetCol() == 0 )
3028 if ( m_currentCellCoords
.GetRow() == 0 ||
3029 m_currentCellCoords
.GetCol() == 0 )
3038 rect
.SetLeft( wxMax(0, left
- extra
) );
3039 rect
.SetTop( wxMax(0, top
- extra
) );
3040 rect
.SetRight( rect
.GetRight() + 2*extra
);
3041 rect
.SetBottom( rect
.GetBottom() + 2*extra
);
3044 m_cellEditCtrl
->SetSize( rect
);
3045 m_cellEditCtrl
->Show( TRUE
);
3047 switch ( m_editCtrlType
)
3049 case wxGRID_TEXTCTRL
:
3050 ((wxTextCtrl
*) m_cellEditCtrl
)->SetInsertionPointEnd();
3053 case wxGRID_CHECKBOX
:
3054 // TODO: anything ???
3059 // TODO: anything ???
3063 case wxGRID_COMBOBOX
:
3064 // TODO: anything ???
3069 m_cellEditCtrl
->SetFocus();
3075 void wxGrid::HideCellEditControl()
3077 if ( IsCellEditControlEnabled() )
3079 m_cellEditCtrl
->Show( FALSE
);
3084 void wxGrid::SetEditControlValue( const wxString
& value
)
3090 s
= GetCellValue(m_currentCellCoords
);
3094 if ( IsCellEditControlEnabled() )
3096 switch ( m_editCtrlType
)
3098 case wxGRID_TEXTCTRL
:
3099 ((wxGridTextCtrl
*)m_cellEditCtrl
)->SetStartValue(s
);
3102 case wxGRID_CHECKBOX
:
3103 // TODO: implement this
3108 // TODO: implement this
3112 case wxGRID_COMBOBOX
:
3113 // TODO: implement this
3122 void wxGrid::SaveEditControlValue()
3126 wxWindow
*ctrl
= (wxWindow
*)NULL
;
3128 if ( IsCellEditControlEnabled() )
3130 ctrl
= m_cellEditCtrl
;
3137 bool valueChanged
= FALSE
;
3139 switch ( m_editCtrlType
)
3141 case wxGRID_TEXTCTRL
:
3142 valueChanged
= (((wxGridTextCtrl
*)ctrl
)->GetValue() !=
3143 ((wxGridTextCtrl
*)ctrl
)->GetStartValue());
3144 SetCellValue( m_currentCellCoords
,
3145 ((wxTextCtrl
*) ctrl
)->GetValue() );
3148 case wxGRID_CHECKBOX
:
3149 // TODO: implement this
3154 // TODO: implement this
3158 case wxGRID_COMBOBOX
:
3159 // TODO: implement this
3166 SendEvent( EVT_GRID_CELL_CHANGE
,
3167 m_currentCellCoords
.GetRow(),
3168 m_currentCellCoords
.GetCol() );
3175 // ------ Grid location functions
3176 // Note that all of these functions work with the logical coordinates of
3177 // grid cells and labels so you will need to convert from device
3178 // coordinates for mouse events etc.
3181 void wxGrid::XYToCell( int x
, int y
, wxGridCellCoords
& coords
)
3183 int row
= YToRow(y
);
3184 int col
= XToCol(x
);
3186 if ( row
== -1 || col
== -1 )
3188 coords
= wxGridNoCellCoords
;
3192 coords
.Set( row
, col
);
3197 int wxGrid::YToRow( int y
)
3201 for ( i
= 0; i
< m_numRows
; i
++ )
3203 if ( y
< m_rowBottoms
[i
] ) return i
;
3210 int wxGrid::XToCol( int x
)
3214 for ( i
= 0; i
< m_numCols
; i
++ )
3216 if ( x
< m_colRights
[i
] ) return i
;
3223 // return the row number that that the y coord is near the edge of, or
3224 // -1 if not near an edge
3226 int wxGrid::YToEdgeOfRow( int y
)
3230 for ( i
= 0; i
< m_numRows
; i
++ )
3232 if ( m_rowHeights
[i
] > WXGRID_LABEL_EDGE_ZONE
)
3234 d
= abs( y
- m_rowBottoms
[i
] );
3236 if ( d
< WXGRID_LABEL_EDGE_ZONE
) return i
;
3245 // return the col number that that the x coord is near the edge of, or
3246 // -1 if not near an edge
3248 int wxGrid::XToEdgeOfCol( int x
)
3252 for ( i
= 0; i
< m_numCols
; i
++ )
3254 if ( m_colWidths
[i
] > WXGRID_LABEL_EDGE_ZONE
)
3256 d
= abs( x
- m_colRights
[i
] );
3258 if ( d
< WXGRID_LABEL_EDGE_ZONE
) return i
;
3267 wxRect
wxGrid::CellToRect( int row
, int col
)
3269 wxRect
rect( -1, -1, -1, -1 );
3271 if ( row
>= 0 && row
< m_numRows
&&
3272 col
>= 0 && col
< m_numCols
)
3274 rect
.x
= m_colRights
[col
] - m_colWidths
[col
];
3275 rect
.y
= m_rowBottoms
[row
] - m_rowHeights
[row
];
3276 rect
.width
= m_colWidths
[col
];
3277 rect
.height
= m_rowHeights
[ row
];
3284 bool wxGrid::IsVisible( int row
, int col
, bool wholeCellVisible
)
3286 // get the cell rectangle in logical coords
3288 wxRect
r( CellToRect( row
, col
) );
3290 // convert to device coords
3292 int left
, top
, right
, bottom
;
3293 CalcScrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top
);
3294 CalcScrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom
);
3296 // check against the client area of the grid window
3299 m_gridWin
->GetClientSize( &cw
, &ch
);
3301 if ( wholeCellVisible
)
3303 // is the cell wholly visible ?
3305 return ( left
>= 0 && right
<= cw
&&
3306 top
>= 0 && bottom
<= ch
);
3310 // is the cell partly visible ?
3312 return ( ((left
>=0 && left
< cw
) || (right
> 0 && right
<= cw
)) &&
3313 ((top
>=0 && top
< ch
) || (bottom
> 0 && bottom
<= ch
)) );
3318 // make the specified cell location visible by doing a minimal amount
3321 void wxGrid::MakeCellVisible( int row
, int col
)
3324 int xpos
= -1, ypos
= -1;
3326 if ( row
>= 0 && row
< m_numRows
&&
3327 col
>= 0 && col
< m_numCols
)
3329 // get the cell rectangle in logical coords
3331 wxRect
r( CellToRect( row
, col
) );
3333 // convert to device coords
3335 int left
, top
, right
, bottom
;
3336 CalcScrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top
);
3337 CalcScrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom
);
3340 m_gridWin
->GetClientSize( &cw
, &ch
);
3346 else if ( bottom
> ch
)
3348 int h
= r
.GetHeight();
3350 for ( i
= row
-1; i
>= 0; i
-- )
3352 if ( h
+ m_rowHeights
[i
] > ch
) break;
3354 h
+= m_rowHeights
[i
];
3355 ypos
-= m_rowHeights
[i
];
3363 else if ( right
> cw
)
3365 int w
= r
.GetWidth();
3367 for ( i
= col
-1; i
>= 0; i
-- )
3369 if ( w
+ m_colWidths
[i
] > cw
) break;
3371 w
+= m_colWidths
[i
];
3372 xpos
-= m_colWidths
[i
];
3376 if ( xpos
!= -1 || ypos
!= -1 )
3378 if ( xpos
!= -1 ) xpos
= xpos
/10;
3379 if ( ypos
!= -1 ) ypos
= ypos
/10;
3380 Scroll( xpos
, ypos
);
3388 // ------ Grid cursor movement functions
3391 bool wxGrid::MoveCursorUp()
3393 if ( m_currentCellCoords
!= wxGridNoCellCoords
&&
3394 m_currentCellCoords
.GetRow() > 0 )
3396 MakeCellVisible( m_currentCellCoords
.GetRow() - 1,
3397 m_currentCellCoords
.GetCol() );
3399 SetCurrentCell( m_currentCellCoords
.GetRow() - 1,
3400 m_currentCellCoords
.GetCol() );
3409 bool wxGrid::MoveCursorDown()
3411 // TODO: allow for scrolling
3413 if ( m_currentCellCoords
!= wxGridNoCellCoords
&&
3414 m_currentCellCoords
.GetRow() < m_numRows
-1 )
3416 MakeCellVisible( m_currentCellCoords
.GetRow() + 1,
3417 m_currentCellCoords
.GetCol() );
3419 SetCurrentCell( m_currentCellCoords
.GetRow() + 1,
3420 m_currentCellCoords
.GetCol() );
3429 bool wxGrid::MoveCursorLeft()
3431 if ( m_currentCellCoords
!= wxGridNoCellCoords
&&
3432 m_currentCellCoords
.GetCol() > 0 )
3434 MakeCellVisible( m_currentCellCoords
.GetRow(),
3435 m_currentCellCoords
.GetCol() - 1 );
3437 SetCurrentCell( m_currentCellCoords
.GetRow(),
3438 m_currentCellCoords
.GetCol() - 1 );
3447 bool wxGrid::MoveCursorRight()
3449 if ( m_currentCellCoords
!= wxGridNoCellCoords
&&
3450 m_currentCellCoords
.GetCol() < m_numCols
- 1 )
3452 MakeCellVisible( m_currentCellCoords
.GetRow(),
3453 m_currentCellCoords
.GetCol() + 1 );
3455 SetCurrentCell( m_currentCellCoords
.GetRow(),
3456 m_currentCellCoords
.GetCol() + 1 );
3465 bool wxGrid::MovePageUp()
3467 if ( m_currentCellCoords
== wxGridNoCellCoords
) return FALSE
;
3469 int row
= m_currentCellCoords
.GetRow();
3473 m_gridWin
->GetClientSize( &cw
, &ch
);
3475 int y
= m_rowBottoms
[ row
] - m_rowHeights
[ row
];
3476 int newRow
= YToRow( y
- ch
+ 1 );
3481 else if ( newRow
== row
)
3486 MakeCellVisible( newRow
, m_currentCellCoords
.GetCol() );
3487 SetCurrentCell( newRow
, m_currentCellCoords
.GetCol() );
3495 bool wxGrid::MovePageDown()
3497 if ( m_currentCellCoords
== wxGridNoCellCoords
) return FALSE
;
3499 int row
= m_currentCellCoords
.GetRow();
3500 if ( row
< m_numRows
)
3503 m_gridWin
->GetClientSize( &cw
, &ch
);
3505 int y
= m_rowBottoms
[ row
] - m_rowHeights
[ row
];
3506 int newRow
= YToRow( y
+ ch
);
3509 newRow
= m_numRows
- 1;
3511 else if ( newRow
== row
)
3516 MakeCellVisible( newRow
, m_currentCellCoords
.GetCol() );
3517 SetCurrentCell( newRow
, m_currentCellCoords
.GetCol() );
3525 bool wxGrid::MoveCursorUpBlock()
3528 m_currentCellCoords
!= wxGridNoCellCoords
&&
3529 m_currentCellCoords
.GetRow() > 0 )
3531 int row
= m_currentCellCoords
.GetRow();
3532 int col
= m_currentCellCoords
.GetCol();
3534 if ( m_table
->IsEmptyCell(row
, col
) )
3536 // starting in an empty cell: find the next block of
3542 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
3545 else if ( m_table
->IsEmptyCell(row
-1, col
) )
3547 // starting at the top of a block: find the next block
3553 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
3558 // starting within a block: find the top of the block
3563 if ( m_table
->IsEmptyCell(row
, col
) )
3571 MakeCellVisible( row
, col
);
3572 SetCurrentCell( row
, col
);
3580 bool wxGrid::MoveCursorDownBlock()
3583 m_currentCellCoords
!= wxGridNoCellCoords
&&
3584 m_currentCellCoords
.GetRow() < m_numRows
-1 )
3586 int row
= m_currentCellCoords
.GetRow();
3587 int col
= m_currentCellCoords
.GetCol();
3589 if ( m_table
->IsEmptyCell(row
, col
) )
3591 // starting in an empty cell: find the next block of
3594 while ( row
< m_numRows
-1 )
3597 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
3600 else if ( m_table
->IsEmptyCell(row
+1, col
) )
3602 // starting at the bottom of a block: find the next block
3605 while ( row
< m_numRows
-1 )
3608 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
3613 // starting within a block: find the bottom of the block
3615 while ( row
< m_numRows
-1 )
3618 if ( m_table
->IsEmptyCell(row
, col
) )
3626 MakeCellVisible( row
, col
);
3627 SetCurrentCell( row
, col
);
3635 bool wxGrid::MoveCursorLeftBlock()
3638 m_currentCellCoords
!= wxGridNoCellCoords
&&
3639 m_currentCellCoords
.GetCol() > 0 )
3641 int row
= m_currentCellCoords
.GetRow();
3642 int col
= m_currentCellCoords
.GetCol();
3644 if ( m_table
->IsEmptyCell(row
, col
) )
3646 // starting in an empty cell: find the next block of
3652 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
3655 else if ( m_table
->IsEmptyCell(row
, col
-1) )
3657 // starting at the left of a block: find the next block
3663 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
3668 // starting within a block: find the left of the block
3673 if ( m_table
->IsEmptyCell(row
, col
) )
3681 MakeCellVisible( row
, col
);
3682 SetCurrentCell( row
, col
);
3690 bool wxGrid::MoveCursorRightBlock()
3693 m_currentCellCoords
!= wxGridNoCellCoords
&&
3694 m_currentCellCoords
.GetCol() < m_numCols
-1 )
3696 int row
= m_currentCellCoords
.GetRow();
3697 int col
= m_currentCellCoords
.GetCol();
3699 if ( m_table
->IsEmptyCell(row
, col
) )
3701 // starting in an empty cell: find the next block of
3704 while ( col
< m_numCols
-1 )
3707 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
3710 else if ( m_table
->IsEmptyCell(row
, col
+1) )
3712 // starting at the right of a block: find the next block
3715 while ( col
< m_numCols
-1 )
3718 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
3723 // starting within a block: find the right of the block
3725 while ( col
< m_numCols
-1 )
3728 if ( m_table
->IsEmptyCell(row
, col
) )
3736 MakeCellVisible( row
, col
);
3737 SetCurrentCell( row
, col
);
3748 // ------ Label values and formatting
3751 void wxGrid::GetRowLabelAlignment( int *horiz
, int *vert
)
3753 *horiz
= m_rowLabelHorizAlign
;
3754 *vert
= m_rowLabelVertAlign
;
3757 void wxGrid::GetColLabelAlignment( int *horiz
, int *vert
)
3759 *horiz
= m_colLabelHorizAlign
;
3760 *vert
= m_colLabelVertAlign
;
3763 wxString
wxGrid::GetRowLabelValue( int row
)
3767 return m_table
->GetRowLabelValue( row
);
3777 wxString
wxGrid::GetColLabelValue( int col
)
3781 return m_table
->GetColLabelValue( col
);
3791 void wxGrid::SetRowLabelSize( int width
)
3793 // TODO: how to do this with the box sizers ?
3796 void wxGrid::SetColLabelSize( int height
)
3798 // TODO: how to do this with the box sizers ?
3801 void wxGrid::SetLabelBackgroundColour( const wxColour
& colour
)
3803 if ( m_labelBackgroundColour
!= colour
)
3805 m_labelBackgroundColour
= colour
;
3806 m_rowLabelWin
->SetBackgroundColour( colour
);
3807 m_colLabelWin
->SetBackgroundColour( colour
);
3808 m_cornerLabelWin
->SetBackgroundColour( colour
);
3810 if ( !GetBatchCount() )
3812 m_rowLabelWin
->Refresh();
3813 m_colLabelWin
->Refresh();
3814 m_cornerLabelWin
->Refresh();
3819 void wxGrid::SetLabelTextColour( const wxColour
& colour
)
3821 if ( m_labelTextColour
!= colour
)
3823 m_labelTextColour
= colour
;
3824 if ( !GetBatchCount() )
3826 m_rowLabelWin
->Refresh();
3827 m_colLabelWin
->Refresh();
3832 void wxGrid::SetLabelFont( const wxFont
& font
)
3835 if ( !GetBatchCount() )
3837 m_rowLabelWin
->Refresh();
3838 m_colLabelWin
->Refresh();
3842 void wxGrid::SetRowLabelAlignment( int horiz
, int vert
)
3844 if ( horiz
== wxLEFT
|| horiz
== wxCENTRE
|| horiz
== wxRIGHT
)
3846 m_rowLabelHorizAlign
= horiz
;
3849 if ( vert
== wxTOP
|| vert
== wxCENTRE
|| vert
== wxBOTTOM
)
3851 m_rowLabelVertAlign
= vert
;
3854 if ( !GetBatchCount() )
3856 m_rowLabelWin
->Refresh();
3857 m_colLabelWin
->Refresh();
3861 void wxGrid::SetColLabelAlignment( int horiz
, int vert
)
3863 if ( horiz
== wxLEFT
|| horiz
== wxCENTRE
|| horiz
== wxRIGHT
)
3865 m_colLabelHorizAlign
= horiz
;
3868 if ( vert
== wxTOP
|| vert
== wxCENTRE
|| vert
== wxBOTTOM
)
3870 m_colLabelVertAlign
= vert
;
3873 if ( !GetBatchCount() )
3875 m_rowLabelWin
->Refresh();
3876 m_colLabelWin
->Refresh();
3880 void wxGrid::SetRowLabelValue( int row
, const wxString
& s
)
3884 m_table
->SetRowLabelValue( row
, s
);
3885 if ( !GetBatchCount() )
3887 // TODO: Optimize this
3889 m_rowLabelWin
->Refresh();
3894 void wxGrid::SetColLabelValue( int col
, const wxString
& s
)
3898 m_table
->SetColLabelValue( col
, s
);
3899 if ( !GetBatchCount() )
3901 // TODO: optimize this
3903 m_colLabelWin
->Refresh();
3908 void wxGrid::SetGridLineColour( const wxColour
& colour
)
3910 if ( m_gridLineColour
!= colour
)
3912 m_gridLineColour
= colour
;
3914 wxClientDC
dc( m_gridWin
);
3916 DrawAllGridLines( dc
);
3920 void wxGrid::EnableGridLines( bool enable
)
3922 if ( enable
!= m_gridLinesEnabled
)
3924 m_gridLinesEnabled
= enable
;
3926 if ( !GetBatchCount() )
3930 wxClientDC
dc( m_gridWin
);
3932 DrawAllGridLines( dc
);
3936 m_gridWin
->Refresh();
3943 int wxGrid::GetDefaultRowSize()
3945 return m_defaultRowHeight
;
3948 int wxGrid::GetRowSize( int row
)
3950 if ( row
>= 0 && row
< m_numRows
)
3951 return m_rowHeights
[row
];
3953 return 0; // TODO: log an error here
3956 int wxGrid::GetDefaultColSize()
3958 return m_defaultColWidth
;
3961 int wxGrid::GetColSize( int col
)
3963 if ( col
>= 0 && col
< m_numCols
)
3964 return m_colWidths
[col
];
3966 return 0; // TODO: log an error here
3969 wxColour
wxGrid::GetDefaultCellBackgroundColour()
3971 // TODO: replace this temp test code
3973 return wxColour( 255, 255, 255 );
3976 wxColour
wxGrid::GetCellBackgroundColour( int WXUNUSED(row
), int WXUNUSED(col
) )
3978 // TODO: replace this temp test code
3980 return wxColour( 255, 255, 255 );
3983 wxColour
wxGrid::GetDefaultCellTextColour()
3985 // TODO: replace this temp test code
3987 return wxColour( 0, 0, 0 );
3990 wxColour
wxGrid::GetCellTextColour( int WXUNUSED(row
), int WXUNUSED(col
) )
3992 // TODO: replace this temp test code
3994 return wxColour( 0, 0, 0 );
3998 wxFont
wxGrid::GetDefaultCellFont()
4000 return m_defaultCellFont
;
4003 wxFont
wxGrid::GetCellFont( int WXUNUSED(row
), int WXUNUSED(col
) )
4005 // TODO: replace this temp test code
4007 return m_defaultCellFont
;
4010 void wxGrid::GetDefaultCellAlignment( int *horiz
, int *vert
)
4012 // TODO: replace this temp test code
4018 void wxGrid::GetCellAlignment( int WXUNUSED(row
), int WXUNUSED(col
), int *horiz
, int *vert
)
4020 // TODO: replace this temp test code
4026 void wxGrid::SetDefaultRowSize( int height
, bool resizeExistingRows
)
4028 m_defaultRowHeight
= wxMax( height
, WXGRID_MIN_ROW_HEIGHT
);
4030 if ( resizeExistingRows
)
4034 for ( row
= 0; row
< m_numRows
; row
++ )
4036 m_rowHeights
[row
] = m_defaultRowHeight
;
4037 bottom
+= m_defaultRowHeight
;
4038 m_rowBottoms
[row
] = bottom
;
4044 void wxGrid::SetRowSize( int row
, int height
)
4048 if ( row
>= 0 && row
< m_numRows
)
4050 int h
= wxMax( 0, height
);
4051 int diff
= h
- m_rowHeights
[row
];
4053 m_rowHeights
[row
] = h
;
4054 for ( i
= row
; i
< m_numRows
; i
++ )
4056 m_rowBottoms
[i
] += diff
;
4060 // Note: we are ending the event *after* doing
4061 // default processing in this case
4063 SendEvent( EVT_GRID_ROW_SIZE
,
4068 // TODO: log an error here
4072 void wxGrid::SetDefaultColSize( int width
, bool resizeExistingCols
)
4074 m_defaultColWidth
= wxMax( width
, WXGRID_MIN_COL_WIDTH
);
4076 if ( resizeExistingCols
)
4080 for ( col
= 0; col
< m_numCols
; col
++ )
4082 m_colWidths
[col
] = m_defaultColWidth
;
4083 right
+= m_defaultColWidth
;
4084 m_colRights
[col
] = right
;
4090 void wxGrid::SetColSize( int col
, int width
)
4094 if ( col
>= 0 && col
< m_numCols
)
4096 int w
= wxMax( 0, width
);
4097 int diff
= w
- m_colWidths
[col
];
4098 m_colWidths
[col
] = w
;
4100 for ( i
= col
; i
< m_numCols
; i
++ )
4102 m_colRights
[i
] += diff
;
4106 // Note: we are ending the event *after* doing
4107 // default processing in this case
4109 SendEvent( EVT_GRID_COL_SIZE
,
4114 // TODO: log an error here
4118 void wxGrid::SetDefaultCellBackgroundColour( const wxColour
& )
4120 // TODO: everything !!!
4124 void wxGrid::SetCellBackgroundColour( int WXUNUSED(row
), int WXUNUSED(col
), const wxColour
& )
4126 // TODO: everything !!!
4130 void wxGrid::SetDefaultCellTextColour( const wxColour
& )
4132 // TODO: everything !!!
4136 void wxGrid::SetCellTextColour( int WXUNUSED(row
), int WXUNUSED(col
), const wxColour
& )
4138 // TODO: everything !!!
4142 void wxGrid::SetDefaultCellFont( const wxFont
& )
4144 // TODO: everything !!!
4148 void wxGrid::SetCellFont( int WXUNUSED(row
), int WXUNUSED(col
), const wxFont
& )
4150 // TODO: everything !!!
4154 void wxGrid::SetDefaultCellAlignment( int WXUNUSED(horiz
), int WXUNUSED(vert
) )
4156 // TODO: everything !!!
4160 void wxGrid::SetCellAlignment( int WXUNUSED(row
), int WXUNUSED(col
), int WXUNUSED(horiz
), int WXUNUSED(vert
) )
4162 // TODO: everything !!!
4169 // ------ cell value accessor functions
4172 void wxGrid::SetCellValue( int row
, int col
, const wxString
& s
)
4176 m_table
->SetValue( row
, col
, s
.c_str() );
4177 if ( !GetBatchCount() )
4179 wxClientDC
dc( m_gridWin
);
4181 DrawCell( dc
, wxGridCellCoords(row
, col
) );
4184 #if 0 // TODO: edit in place
4186 if ( m_currentCellCoords
.GetRow() == row
&&
4187 m_currentCellCoords
.GetCol() == col
)
4189 SetEditControlValue( s
);
4198 // ------ Block, row and col selection
4201 void wxGrid::SelectRow( int row
, bool addToSelected
)
4205 if ( IsSelection() && addToSelected
)
4207 if ( m_selectedTopLeft
.GetRow() > row
)
4208 m_selectedTopLeft
.SetRow( row
);
4210 m_selectedTopLeft
.SetCol( 0 );
4212 if ( m_selectedBottomRight
.GetRow() < row
)
4213 m_selectedBottomRight
.SetRow( row
);
4215 m_selectedBottomRight
.SetCol( m_numCols
- 1 );
4217 // TODO: optimize this so that we only refresh the newly
4220 r
= SelectionToDeviceRect();
4221 if ( r
!= wxGridNoCellRect
) m_gridWin
->Refresh( TRUE
, &r
);
4225 r
= SelectionToDeviceRect();
4227 if ( r
!= wxGridNoCellRect
) m_gridWin
->Refresh( TRUE
, &r
);
4229 m_selectedTopLeft
.Set( row
, 0 );
4230 m_selectedBottomRight
.Set( row
, m_numCols
-1 );
4231 r
= SelectionToDeviceRect();
4232 m_gridWin
->Refresh( TRUE
, &r
);
4235 wxGridRangeSelectEvent
gridEvt( GetId(),
4236 EVT_GRID_RANGE_SELECT
,
4239 m_selectedBottomRight
);
4241 GetEventHandler()->ProcessEvent(gridEvt
);
4245 void wxGrid::SelectCol( int col
, bool addToSelected
)
4249 if ( IsSelection() && addToSelected
)
4251 if ( m_selectedTopLeft
.GetCol() > col
)
4252 m_selectedTopLeft
.SetCol( col
);
4254 m_selectedTopLeft
.SetRow( 0 );
4256 if ( m_selectedBottomRight
.GetCol() < col
)
4257 m_selectedBottomRight
.SetCol( col
);
4259 m_selectedBottomRight
.SetRow( m_numRows
- 1 );
4261 // TODO: optimize this so that we only refresh the newly
4264 r
= SelectionToDeviceRect();
4265 if ( r
!= wxGridNoCellRect
) m_gridWin
->Refresh( TRUE
, &r
);
4269 r
= SelectionToDeviceRect();
4271 if ( r
!= wxGridNoCellRect
) m_gridWin
->Refresh( TRUE
, &r
);
4273 m_selectedTopLeft
.Set( 0, col
);
4274 m_selectedBottomRight
.Set( m_numRows
-1, col
);
4275 r
= SelectionToDeviceRect();
4276 m_gridWin
->Refresh( TRUE
, &r
);
4279 wxGridRangeSelectEvent
gridEvt( GetId(),
4280 EVT_GRID_RANGE_SELECT
,
4283 m_selectedBottomRight
);
4285 GetEventHandler()->ProcessEvent(gridEvt
);
4289 void wxGrid::SelectBlock( int topRow
, int leftCol
, int bottomRow
, int rightCol
)
4292 bool changed
= false;
4293 wxGridCellCoords updateTopLeft
, updateBottomRight
;
4295 if ( topRow
> bottomRow
)
4302 if ( leftCol
> rightCol
)
4309 updateTopLeft
= m_selectedTopLeft
;
4310 if (m_selectedTopLeft
!= wxGridCellCoords( topRow
, leftCol
) )
4312 m_selectedTopLeft
= wxGridCellCoords( topRow
, leftCol
);
4313 if (updateTopLeft
== wxGridNoCellCoords
)
4315 updateTopLeft
= m_selectedTopLeft
;
4319 if(updateTopLeft
.GetRow() > topRow
)
4320 updateTopLeft
.SetRow(topRow
);
4321 if (updateTopLeft
.GetCol() > leftCol
)
4322 updateTopLeft
.SetCol(leftCol
);
4327 updateBottomRight
= m_selectedBottomRight
;
4328 if (m_selectedBottomRight
!= wxGridCellCoords( bottomRow
, rightCol
) )
4330 m_selectedBottomRight
= wxGridCellCoords( bottomRow
, rightCol
);
4331 if (updateBottomRight
== wxGridNoCellCoords
)
4333 updateBottomRight
= m_selectedBottomRight
;
4337 if (updateBottomRight
.GetRow() < bottomRow
)
4338 updateBottomRight
.SetRow(bottomRow
);
4339 if (updateBottomRight
.GetCol() < rightCol
)
4340 updateBottomRight
.SetCol(rightCol
);
4347 wxRect
r( BlockToDeviceRect( updateTopLeft
, updateBottomRight
) );
4348 m_gridWin
->Refresh( TRUE
, &r
);
4351 // only generate an event if the block is not being selected by
4352 // dragging the mouse (in which case the event will be generated in
4353 // the mouse event handler)
4354 if ( !m_isDragging
)
4356 wxGridRangeSelectEvent
gridEvt( GetId(),
4357 EVT_GRID_RANGE_SELECT
,
4360 m_selectedBottomRight
);
4362 GetEventHandler()->ProcessEvent(gridEvt
);
4366 void wxGrid::SelectAll()
4368 m_selectedTopLeft
.Set( 0, 0 );
4369 m_selectedBottomRight
.Set( m_numRows
-1, m_numCols
-1 );
4371 m_gridWin
->Refresh();
4375 void wxGrid::ClearSelection()
4377 m_selectedTopLeft
= wxGridNoCellCoords
;
4378 m_selectedBottomRight
= wxGridNoCellCoords
;
4382 // This function returns the rectangle that encloses the given block
4383 // in device coords clipped to the client size of the grid window.
4385 wxRect
wxGrid::BlockToDeviceRect( const wxGridCellCoords
&topLeft
,
4386 const wxGridCellCoords
&bottomRight
)
4388 wxRect
rect( wxGridNoCellRect
);
4391 cellRect
= CellToRect( topLeft
);
4392 if ( cellRect
!= wxGridNoCellRect
)
4398 rect
= wxRect( 0, 0, 0, 0 );
4401 cellRect
= CellToRect( bottomRight
);
4402 if ( cellRect
!= wxGridNoCellRect
)
4408 return wxGridNoCellRect
;
4411 // convert to scrolled coords
4413 int left
, top
, right
, bottom
;
4414 CalcScrolledPosition( rect
.GetLeft(), rect
.GetTop(), &left
, &top
);
4415 CalcScrolledPosition( rect
.GetRight(), rect
.GetBottom(), &right
, &bottom
);
4418 m_gridWin
->GetClientSize( &cw
, &ch
);
4420 rect
.SetLeft( wxMax(0, left
) );
4421 rect
.SetTop( wxMax(0, top
) );
4422 rect
.SetRight( wxMin(cw
, right
) );
4423 rect
.SetBottom( wxMin(ch
, bottom
) );
4431 // ------ Grid event classes
4434 IMPLEMENT_DYNAMIC_CLASS( wxGridEvent
, wxEvent
)
4436 wxGridEvent::wxGridEvent( int id
, wxEventType type
, wxObject
* obj
,
4437 int row
, int col
, int x
, int y
,
4438 bool control
, bool shift
, bool alt
, bool meta
)
4439 : wxNotifyEvent( type
, id
)
4445 m_control
= control
;
4450 SetEventObject(obj
);
4454 IMPLEMENT_DYNAMIC_CLASS( wxGridSizeEvent
, wxEvent
)
4456 wxGridSizeEvent::wxGridSizeEvent( int id
, wxEventType type
, wxObject
* obj
,
4457 int rowOrCol
, int x
, int y
,
4458 bool control
, bool shift
, bool alt
, bool meta
)
4459 : wxNotifyEvent( type
, id
)
4461 m_rowOrCol
= rowOrCol
;
4464 m_control
= control
;
4469 SetEventObject(obj
);
4473 IMPLEMENT_DYNAMIC_CLASS( wxGridRangeSelectEvent
, wxEvent
)
4475 wxGridRangeSelectEvent::wxGridRangeSelectEvent(int id
, wxEventType type
, wxObject
* obj
,
4476 const wxGridCellCoords
& topLeft
,
4477 const wxGridCellCoords
& bottomRight
,
4478 bool control
, bool shift
, bool alt
, bool meta
)
4479 : wxNotifyEvent( type
, id
)
4481 m_topLeft
= topLeft
;
4482 m_bottomRight
= bottomRight
;
4483 m_control
= control
;
4488 SetEventObject(obj
);
4492 #endif // ifndef wxUSE_NEW_GRID