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 /////////////////////////////////////////////////////////////////////////////
12 // For compilers that support precompilation, includes "wx/wx.h".
13 #include "wx/wxprec.h"
17 #if !defined(wxUSE_NEW_GRID) || !(wxUSE_NEW_GRID)
22 #pragma implementation "grid.h"
32 #include "wx/dcclient.h"
33 #include "wx/settings.h"
37 #include "wx/generic/grid.h"
40 //////////////////////////////////////////////////////////////////////
42 wxGridCellCoords
wxGridNoCellCoords( -1, -1 );
43 wxRect
wxGridNoCellRect( -1, -1, -1, -1 );
47 //////////////////////////////////////////////////////////////////////
49 // Abstract base class for grid data (the model)
51 IMPLEMENT_ABSTRACT_CLASS( wxGridTableBase
, wxObject
)
54 wxGridTableBase::wxGridTableBase()
57 m_view
= (wxGrid
*) NULL
;
60 wxGridTableBase::~wxGridTableBase()
65 bool wxGridTableBase::InsertRows( size_t pos
, size_t numRows
)
67 wxLogWarning( "Called grid table class function InsertRows(pos=%d, N=%d)\n"
68 "but your derived table class does not override this function",
74 bool wxGridTableBase::AppendRows( size_t numRows
)
76 wxLogWarning( "Called grid table class function AppendRows(N=%d)\n"
77 "but your derived table class does not override this function",
83 bool wxGridTableBase::DeleteRows( size_t pos
, size_t numRows
)
85 wxLogWarning( "Called grid table class function DeleteRows(pos=%d, N=%d)\n"
86 "but your derived table class does not override this function",
92 bool wxGridTableBase::InsertCols( size_t pos
, size_t numCols
)
94 wxLogWarning( "Called grid table class function InsertCols(pos=%d, N=%d)\n"
95 "but your derived table class does not override this function",
101 bool wxGridTableBase::AppendCols( size_t numCols
)
103 wxLogWarning( "Called grid table class function AppendCols(N=%d)\n"
104 "but your derived table class does not override this function",
110 bool wxGridTableBase::DeleteCols( size_t pos
, size_t numCols
)
112 wxLogWarning( "Called grid table class function DeleteCols(pos=%d, N=%d)\n"
113 "but your derived table class does not override this function",
120 wxString
wxGridTableBase::GetRowLabelValue( int row
)
127 wxString
wxGridTableBase::GetColLabelValue( int col
)
129 // default col labels are:
130 // cols 0 to 25 : A-Z
131 // cols 26 to 675 : AA-ZZ
138 s
+= ('A' + (char)( col%26
));
140 if ( col
< 0 ) break;
143 // reverse the string...
145 for ( i
= 0; i
< n
; i
++ )
155 //////////////////////////////////////////////////////////////////////
157 // Message class for the grid table to send requests and notifications
161 wxGridTableMessage::wxGridTableMessage()
163 m_table
= (wxGridTableBase
*) NULL
;
169 wxGridTableMessage::wxGridTableMessage( wxGridTableBase
*table
, int id
,
170 int commandInt1
, int commandInt2
)
174 m_comInt1
= commandInt1
;
175 m_comInt2
= commandInt2
;
180 //////////////////////////////////////////////////////////////////////
182 // A basic grid table for string data. An object of this class will
183 // created by wxGrid if you don't specify an alternative table class.
187 // this is a magic incantation which must be done!
188 #include <wx/arrimpl.cpp>
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( "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( "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( "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
,
582 const wxString
& value
,
586 : wxTextCtrl( par
, id
, value
, pos
, size
, style
)
588 m_isCellControl
= isCellControl
;
592 void wxGridTextCtrl::OnKeyDown( wxKeyEvent
& ev
)
594 switch ( ev
.KeyCode() )
597 ((wxGrid
*)GetParent())->SetEditControlValue( startValue
);
598 SetInsertionPointEnd();
606 if ( m_isCellControl
)
608 // send the event to the parent grid, skipping the
609 // event if nothing happens
611 ev
.Skip( !GetParent()->ProcessEvent( ev
) );
615 // default text control response within the top edit
624 if ( m_isCellControl
)
626 // send the event to the parent grid, skipping the
627 // event if nothing happens
629 ev
.Skip( !GetParent()->ProcessEvent( ev
) );
633 // default text control response within the top edit
645 void wxGridTextCtrl::SetStartValue( const wxString
& s
)
648 wxTextCtrl::SetValue( s
.c_str() );
652 //////////////////////////////////////////////////////////////////////
654 IMPLEMENT_DYNAMIC_CLASS( wxGrid
, wxPanel
)
657 BEGIN_EVENT_TABLE( wxGrid
, wxPanel
)
658 EVT_PAINT( wxGrid::OnPaint
)
659 EVT_SIZE( wxGrid::OnSize
)
660 EVT_MOUSE_EVENTS( wxGrid::OnMouse
)
661 EVT_KEY_DOWN( wxGrid::OnKeyDown
)
662 EVT_TEXT( wxGRID_CELLCTRL
, wxGrid::OnText
)
663 EVT_TEXT( wxGRID_TOPCTRL
, wxGrid::OnText
)
664 EVT_COMMAND_SCROLL( wxGRID_HORIZSCROLL
, wxGrid::OnGridScroll
)
665 EVT_COMMAND_SCROLL( wxGRID_VERTSCROLL
, wxGrid::OnGridScroll
)
676 // ----- internal init and update functions
679 void wxGrid::Create()
681 m_table
= (wxGridTableBase
*) NULL
;
682 m_topEditCtrl
= (wxWindow
*) NULL
;
683 m_cellEditCtrl
= (wxWindow
*) NULL
;
684 m_horizScrollBar
= (wxScrollBar
*) NULL
;
685 m_vertScrollBar
= (wxScrollBar
*) NULL
;
699 // TODO: perhaps have a style flag for control panel
701 m_topEditCtrlEnabled
= FALSE
;
702 m_topEditCtrl
= new wxGridTextCtrl( this,
707 wxSize(WXGRID_DEFAULT_TOPEDIT_WIDTH
,
708 WXGRID_DEFAULT_TOPEDIT_HEIGHT
),
710 m_topEditCtrl
->Show( FALSE
);
712 if ( m_numRows
<= 0 )
713 m_numRows
= WXGRID_DEFAULT_NUMBER_ROWS
;
715 if ( m_numCols
<= 0 )
716 m_numCols
= WXGRID_DEFAULT_NUMBER_COLS
;
718 m_rowLabelWidth
= WXGRID_DEFAULT_ROW_LABEL_WIDTH
;
719 m_colLabelHeight
= WXGRID_DEFAULT_COL_LABEL_HEIGHT
;
721 // default labels are pale grey with black text
723 m_labelBackgroundColour
= wxColour( 192, 192, 192 );
724 m_labelTextColour
= wxColour( 0, 0, 0 );
726 // TODO: something better than this ?
728 m_labelFont
= this->GetFont();
729 m_labelFont
.SetWeight( m_labelFont
.GetWeight() + 2 );
731 m_rowLabelHorizAlign
= wxLEFT
;
732 m_rowLabelVertAlign
= wxCENTRE
;
734 m_colLabelHorizAlign
= wxCENTRE
;
735 m_colLabelVertAlign
= wxTOP
;
737 m_defaultRowHeight
= WXGRID_DEFAULT_ROW_HEIGHT
;
738 m_defaultColWidth
= WXGRID_DEFAULT_COL_WIDTH
;
740 m_rowHeights
.Alloc( m_numRows
);
741 m_rowBottoms
.Alloc( m_numRows
);
742 for ( i
= 0; i
< m_numRows
; i
++ )
744 m_rowHeights
.Add( m_defaultRowHeight
);
745 m_rowBottoms
.Add( 0 ); // set by CalcDimensions()
748 m_colWidths
.Alloc( m_numCols
);
749 m_colRights
.Alloc( m_numRows
);
750 for ( i
= 0; i
< m_numCols
; i
++ )
752 m_colWidths
.Add( m_defaultColWidth
);
753 m_colRights
.Add( 0 ); // set by CalcDimensions()
756 // TODO: improve this ?
758 m_defaultCellFont
= this->GetFont();
760 m_gridLineColour
= wxColour( 0, 0, 255 );
761 m_gridLinesEnabled
= TRUE
;
763 m_scrollBarWidth
= WXGRID_DEFAULT_SCROLLBAR_WIDTH
;
765 m_horizScrollBar
= new wxScrollBar( this,
771 m_vertScrollBar
= new wxScrollBar( this,
778 m_wholeColsVisible
= 0;
779 m_wholeRowsVisible
= 0;
782 m_inOnKeyDown
= FALSE
;
785 m_cursorMode
= WXGRID_CURSOR_DEFAULT
;
788 m_isDragging
= FALSE
;
790 m_rowResizeCursor
= wxCursor( wxCURSOR_SIZENS
);
791 m_colResizeCursor
= wxCursor( wxCURSOR_SIZEWE
);
793 m_currentCellCoords
= wxGridNoCellCoords
;
794 m_currentCellHighlighted
= FALSE
;
796 m_selectedTopLeft
= wxGridNoCellCoords
;
797 m_selectedBottomRight
= wxGridNoCellCoords
;
799 m_editable
= TRUE
; // default for whole grid
801 // TODO: extend this to other types of controls
803 m_cellEditCtrl
= new wxGridTextCtrl( this,
811 | wxTE_MULTILINE
| wxTE_NO_VSCROLL
815 m_cellEditCtrl
->Show( FALSE
);
816 m_cellEditCtrlEnabled
= TRUE
;
817 m_editCtrlType
= wxGRID_TEXTCTRL
;
819 // Not really needed here, it gets called by OnSize()
825 void wxGrid::CalcDimensions()
829 if ( IsTopEditControlEnabled() )
832 m_topEditCtrl
->GetSize( &ctrlW
, &ctrlH
);
841 int bottom
= m_top
+ m_colLabelHeight
;
842 for ( i
= m_scrollPosY
; i
< m_numRows
; i
++ )
844 bottom
+= m_rowHeights
[i
];
845 m_rowBottoms
[i
] = bottom
;
848 int right
= m_left
+ m_rowLabelWidth
;
849 for ( i
= m_scrollPosX
; i
< m_numCols
; i
++ )
851 right
+= m_colWidths
[i
];
852 m_colRights
[i
] = right
;
855 // adjust the scroll bars
859 GetClientSize(&cw
, &ch
);
861 // begin by assuming that we don't need either scroll bar
863 int vertScrollBarWidth
= 0;
864 int horizScrollBarHeight
= 0;
866 // Each scroll bar needs to eventually know if the other one is
867 // required in deciding whether or not it is also required - hence
868 // this loop. A bit inelegant but simple and it works.
871 for ( check
= 0; check
< 2; check
++ )
873 if ( m_numRows
> 0 &&
874 m_rowBottoms
[m_numRows
-1] + horizScrollBarHeight
> ch
)
876 vertScrollBarWidth
= m_scrollBarWidth
;
878 m_wholeRowsVisible
= 0;
879 for ( i
= m_scrollPosY
; i
< m_numRows
; i
++ )
881 // A partial row doesn't count, we still have to scroll to
882 // see the rest of it
883 if ( m_rowBottoms
[i
] + horizScrollBarHeight
> ch
) break;
885 m_wholeRowsVisible
++ ;
890 m_wholeRowsVisible
= m_numRows
- m_scrollPosY
;
893 vertScrollBarWidth
= m_scrollBarWidth
;
899 m_colRights
[m_numCols
-1] + vertScrollBarWidth
> cw
)
901 horizScrollBarHeight
= m_scrollBarWidth
;
903 m_wholeColsVisible
= 0;
904 for ( i
= m_scrollPosX
; i
< m_numCols
; i
++ )
906 // A partial col doesn't count, we still have to scroll to
907 // see the rest of it
908 if ( m_colRights
[i
] + vertScrollBarWidth
> cw
) break;
910 m_wholeColsVisible
++ ;
915 // we can see the right-most column
917 m_wholeColsVisible
= m_numCols
- m_scrollPosX
;
920 horizScrollBarHeight
= m_scrollBarWidth
;
926 if ( m_vertScrollBar
)
928 if ( !vertScrollBarWidth
)
930 m_vertScrollBar
->Show(FALSE
);
934 m_vertScrollBar
->Show(TRUE
);
935 m_vertScrollBar
->SetScrollbar(
937 wxMax(m_wholeRowsVisible
, 1),
938 (m_wholeRowsVisible
== 0 ? 1 : m_numRows
),
939 wxMax(m_wholeRowsVisible
, 1) );
941 m_vertScrollBar
->SetSize( cw
- m_scrollBarWidth
,
944 ch
- m_top
- horizScrollBarHeight
);
948 if ( m_horizScrollBar
)
950 if ( !horizScrollBarHeight
)
952 m_horizScrollBar
->Show(FALSE
);
956 m_horizScrollBar
->Show(TRUE
);
958 m_horizScrollBar
->SetScrollbar(
960 wxMax(m_wholeColsVisible
, 1),
961 (m_wholeColsVisible
== 0) ? 1 : m_numCols
,
962 wxMax(m_wholeColsVisible
, 1) );
964 m_horizScrollBar
->SetSize( m_left
,
965 ch
- m_scrollBarWidth
,
966 cw
- m_left
- vertScrollBarWidth
,
971 m_bottom
= m_right
= 0;
974 m_bottom
= wxMin( m_rowBottoms
[m_numRows
-1],
975 ch
- horizScrollBarHeight
);
979 m_right
= wxMin( m_colRights
[m_numCols
-1],
980 cw
- vertScrollBarWidth
);
985 bool wxGrid::IsOnScreen()
988 GetClientSize( &cw
, &ch
);
993 // this is called when the grid table sends a message to say that it
994 // has been redimensioned
996 bool wxGrid::Redimension( wxGridTableMessage
& msg
)
1000 switch ( msg
.GetId() )
1002 case wxGRIDTABLE_NOTIFY_ROWS_INSERTED
:
1004 size_t pos
= msg
.GetCommandInt();
1005 int numRows
= msg
.GetCommandInt2();
1006 for ( i
= 0; i
< numRows
; i
++ )
1008 m_rowHeights
.Insert( m_defaultRowHeight
, pos
);
1009 m_rowBottoms
.Insert( 0, pos
);
1011 m_numRows
+= numRows
;
1016 case wxGRIDTABLE_NOTIFY_ROWS_APPENDED
:
1018 int numRows
= msg
.GetCommandInt();
1019 for ( i
= 0; i
< numRows
; i
++ )
1021 m_rowHeights
.Add( m_defaultRowHeight
);
1022 m_rowBottoms
.Add( 0 );
1024 m_numRows
+= numRows
;
1029 case wxGRIDTABLE_NOTIFY_ROWS_DELETED
:
1031 size_t pos
= msg
.GetCommandInt();
1032 int numRows
= msg
.GetCommandInt2();
1033 for ( i
= 0; i
< numRows
; i
++ )
1035 m_rowHeights
.Remove( pos
);
1036 m_rowBottoms
.Remove( pos
);
1038 m_numRows
-= numRows
;
1040 // TODO: improve these adjustments...
1042 if ( m_scrollPosY
>= m_numRows
)
1048 m_colWidths
.Clear();
1049 m_colRights
.Clear();
1050 m_currentCellCoords
= wxGridNoCellCoords
;
1052 else if ( m_currentCellCoords
.GetRow() >= m_numRows
)
1054 m_currentCellCoords
.Set( 0, 0 );
1060 case wxGRIDTABLE_NOTIFY_COLS_INSERTED
:
1062 size_t pos
= msg
.GetCommandInt();
1063 int numCols
= msg
.GetCommandInt2();
1064 for ( i
= 0; i
< numCols
; i
++ )
1066 m_colWidths
.Insert( m_defaultColWidth
, pos
);
1067 m_colRights
.Insert( 0, pos
);
1069 m_numCols
+= numCols
;
1074 case wxGRIDTABLE_NOTIFY_COLS_APPENDED
:
1076 int numCols
= msg
.GetCommandInt();
1077 for ( i
= 0; i
< numCols
; i
++ )
1079 m_colWidths
.Add( m_defaultColWidth
);
1080 m_colRights
.Add( 0 );
1082 m_numCols
+= numCols
;
1087 case wxGRIDTABLE_NOTIFY_COLS_DELETED
:
1089 size_t pos
= msg
.GetCommandInt();
1090 int numCols
= msg
.GetCommandInt2();
1091 for ( i
= 0; i
< numCols
; i
++ )
1093 m_colWidths
.Remove( pos
);
1094 m_colRights
.Remove( pos
);
1096 m_numCols
-= numCols
;
1098 // TODO: improve these adjustments...
1100 if ( m_scrollPosX
>= m_numCols
)
1105 #if 0 // leave the row alone here so that AppendCols will work subsequently
1107 m_rowHeights
.Clear();
1108 m_rowBottoms
.Clear();
1110 m_currentCellCoords
= wxGridNoCellCoords
;
1112 else if ( m_currentCellCoords
.GetCol() >= m_numCols
)
1114 m_currentCellCoords
.Set( 0, 0 );
1126 // ----- event handlers
1129 // Generate a grid event based on a mouse event and
1130 // return the result of ProcessEvent()
1132 bool wxGrid::SendEvent( const wxEventType type
,
1134 wxMouseEvent
& mouseEv
)
1136 if ( type
== EVT_GRID_ROW_SIZE
||
1137 type
== EVT_GRID_COL_SIZE
)
1139 int rowOrCol
= (row
== -1 ? col
: row
);
1141 wxGridSizeEvent
gridEvt( GetId(),
1145 mouseEv
.GetX(), mouseEv
.GetY(),
1146 mouseEv
.ControlDown(),
1147 mouseEv
.ShiftDown(),
1149 mouseEv
.MetaDown() );
1151 return GetEventHandler()->ProcessEvent(gridEvt
);
1153 else if ( type
== EVT_GRID_RANGE_SELECT
)
1155 wxGridRangeSelectEvent
gridEvt( GetId(),
1159 m_selectedBottomRight
,
1160 mouseEv
.ControlDown(),
1161 mouseEv
.ShiftDown(),
1163 mouseEv
.MetaDown() );
1165 return GetEventHandler()->ProcessEvent(gridEvt
);
1169 wxGridEvent
gridEvt( GetId(),
1173 mouseEv
.GetX(), mouseEv
.GetY(),
1174 mouseEv
.ControlDown(),
1175 mouseEv
.ShiftDown(),
1177 mouseEv
.MetaDown() );
1179 return GetEventHandler()->ProcessEvent(gridEvt
);
1184 // Generate a grid event of specified type and return the result
1185 // of ProcessEvent().
1187 bool wxGrid::SendEvent( const wxEventType type
,
1190 if ( type
== EVT_GRID_ROW_SIZE
||
1191 type
== EVT_GRID_COL_SIZE
)
1193 int rowOrCol
= (row
== -1 ? col
: row
);
1195 wxGridSizeEvent
gridEvt( GetId(),
1200 return GetEventHandler()->ProcessEvent(gridEvt
);
1204 wxGridEvent
gridEvt( GetId(),
1209 return GetEventHandler()->ProcessEvent(gridEvt
);
1214 void wxGrid::OnPaint( wxPaintEvent
& ev
)
1216 wxPaintDC
dc( this );
1218 if ( !m_batchCount
)
1220 // define a clipping region to avoid painting over the scroll bars
1223 if ( m_vertScrollBar
&& m_vertScrollBar
->IsShown() )
1224 vs
= m_scrollBarWidth
;
1227 if ( m_horizScrollBar
&& m_horizScrollBar
->IsShown() )
1228 hs
= m_scrollBarWidth
;
1231 GetClientSize( &cw
, &ch
);
1232 dc
.SetClippingRegion( 0, 0, cw
- vs
, ch
- hs
);
1234 HideCurrentCellHighlight( dc
);
1236 DrawLabelAreas( dc
);
1237 DrawColLabels( dc
);
1238 DrawRowLabels( dc
);
1240 DrawGridLines( dc
);
1243 // TODO: something more elegant than this...
1247 if ( m_currentCellCoords
== wxGridNoCellCoords
)
1248 m_currentCellCoords
.Set(0, 0);
1250 SetEditControlValue();
1251 ShowCellEditControl();
1252 m_firstPaint
= FALSE
;
1255 ShowCurrentCellHighlight( dc
);
1257 dc
.DestroyClippingRegion();
1262 void wxGrid::OnSize( wxSizeEvent
& ev
)
1268 void wxGrid::OnMouse( wxMouseEvent
& ev
)
1274 // ------------------------------------------------------------
1278 if ( ev
.Dragging() )
1280 m_isDragging
= TRUE
;
1282 if ( ev
.LeftIsDown() )
1284 switch( m_cursorMode
)
1286 case WXGRID_CURSOR_SELECT_CELL
:
1288 wxGridCellCoords cellCoords
;
1289 XYToCell( x
, y
, cellCoords
);
1290 if ( cellCoords
!= wxGridNoCellCoords
)
1292 if ( !IsSelection() )
1294 SelectBlock( cellCoords
, cellCoords
);
1296 else if ( !IsInSelection( cellCoords
) )
1298 SelectBlock( m_currentCellCoords
, cellCoords
);
1304 case WXGRID_CURSOR_RESIZE_ROW
:
1306 wxClientDC
dc(this);
1307 dc
.SetLogicalFunction(wxXOR
);
1308 if ( m_dragLastPos
>= 0 )
1310 dc
.DrawLine( m_left
, m_dragLastPos
,
1311 m_right
, m_dragLastPos
);
1313 dc
.DrawLine( m_left
, ev
.GetY(),
1314 m_right
, ev
.GetY());
1316 m_dragLastPos
= ev
.GetY();
1320 case WXGRID_CURSOR_RESIZE_COL
:
1322 wxClientDC
dc(this);
1323 dc
.SetLogicalFunction(wxINVERT
);
1324 if ( m_dragLastPos
>= 0 )
1326 dc
.DrawLine( m_dragLastPos
, m_top
,
1327 m_dragLastPos
, m_bottom
);
1329 dc
.DrawLine( ev
.GetX(), m_top
,
1330 ev
.GetX(), m_bottom
);
1332 m_dragLastPos
= ev
.GetX();
1336 case WXGRID_CURSOR_SELECT_ROW
:
1338 if ( (row
= YToRow( y
)) >= 0 &&
1339 !IsInSelection( row
, 0 ) )
1341 SelectRow( row
, TRUE
);
1346 case WXGRID_CURSOR_SELECT_COL
:
1348 if ( (col
= XToCol( x
)) >= 0 &&
1349 !IsInSelection( 0, col
) )
1351 SelectCol( col
, TRUE
);
1360 m_isDragging
= FALSE
;
1362 // ------------------------------------------------------------
1364 // Left mouse button down
1366 if ( ev
.LeftDown() )
1370 wxGridCellCoords cellCoords
;
1372 switch( XYToArea( x
, y
) )
1374 case WXGRID_ROWLABEL
:
1376 // don't send a label click event for a hit on the
1377 // edge of the row label - this is probably the user
1378 // wanting to resize the row
1380 if ( YToEdgeOfRow(y
) < 0 )
1383 if ( !SendEvent( EVT_GRID_LABEL_LEFT_CLICK
, row
, col
, ev
) )
1385 SelectRow( row
, ev
.ShiftDown() );
1386 m_cursorMode
= WXGRID_CURSOR_SELECT_ROW
;
1392 case WXGRID_COLLABEL
:
1394 // don't send a label click event for a hit on the
1395 // edge of the col label - this is probably the user
1396 // wanting to resize the col
1398 if ( XToEdgeOfCol(x
) < 0 )
1401 if ( !SendEvent( EVT_GRID_LABEL_LEFT_CLICK
, row
, col
, ev
) )
1403 SelectCol( col
, ev
.ShiftDown() );
1404 m_cursorMode
= WXGRID_CURSOR_SELECT_COL
;
1410 case WXGRID_CORNERLABEL
:
1412 // leave both row and col as -1
1414 if ( !SendEvent( EVT_GRID_LABEL_LEFT_CLICK
, row
, col
, ev
) )
1423 XYToCell( x
, y
, cellCoords
);
1424 if ( !SendEvent( EVT_GRID_CELL_LEFT_CLICK
,
1425 cellCoords
.GetRow(),
1426 cellCoords
.GetCol(),
1429 MakeCellVisible( cellCoords
);
1430 SelectCell( cellCoords
);
1437 wxLogMessage( "outside grid area" );
1442 // ------------------------------------------------------------
1444 // Left mouse button double click
1446 else if ( ev
.LeftDClick() )
1450 wxGridCellCoords cellCoords
;
1452 switch( XYToArea( x
, y
) )
1454 case WXGRID_ROWLABEL
:
1456 // don't send a label click event for a hit on the
1457 // edge of the row label - this is probably the user
1458 // wanting to resize the row
1460 if ( YToEdgeOfRow(y
) < 0 )
1463 SendEvent( EVT_GRID_LABEL_LEFT_DCLICK
, row
, col
, ev
);
1468 case WXGRID_COLLABEL
:
1470 // don't send a label click event for a hit on the
1471 // edge of the col label - this is probably the user
1472 // wanting to resize the col
1474 if ( XToEdgeOfCol(x
) < 0 )
1477 SendEvent( EVT_GRID_LABEL_LEFT_DCLICK
, row
, col
, ev
);
1482 case WXGRID_CORNERLABEL
:
1484 // leave both row and col as -1
1486 SendEvent( EVT_GRID_LABEL_LEFT_DCLICK
, row
, col
, ev
);
1492 XYToCell( x
, y
, cellCoords
);
1493 SendEvent( EVT_GRID_CELL_LEFT_DCLICK
,
1494 cellCoords
.GetRow(),
1495 cellCoords
.GetCol(),
1502 wxLogMessage( "outside grid area" );
1507 // ------------------------------------------------------------
1509 // Left mouse button released
1511 else if ( ev
.LeftUp() )
1513 switch ( m_cursorMode
)
1515 case WXGRID_CURSOR_RESIZE_ROW
:
1517 if ( m_dragLastPos
>= 0 )
1519 // erase the last line and resize the row
1521 wxClientDC
dc( this );
1522 dc
.SetLogicalFunction( wxINVERT
);
1523 dc
.DrawLine( m_left
, m_dragLastPos
,
1524 m_right
, m_dragLastPos
);
1525 HideCellEditControl();
1526 int top
= m_top
+ m_colLabelHeight
;
1527 if ( m_dragRowOrCol
> 0 )
1528 top
= m_rowBottoms
[m_dragRowOrCol
-1];
1529 m_rowHeights
[m_dragRowOrCol
] = wxMax( ev
.GetY() - top
,
1530 WXGRID_MIN_ROW_HEIGHT
);
1532 ShowCellEditControl();
1535 // Note: we are ending the event *after* doing
1536 // default processing in this case
1538 SendEvent( EVT_GRID_ROW_SIZE
, m_dragRowOrCol
, -1, ev
);
1543 case WXGRID_CURSOR_RESIZE_COL
:
1545 if ( m_dragLastPos
>= 0 )
1547 // erase the last line and resize the col
1549 wxClientDC
dc( this );
1550 dc
.SetLogicalFunction( wxINVERT
);
1551 dc
.DrawLine( m_left
, m_dragLastPos
,
1552 m_right
, m_dragLastPos
);
1553 HideCellEditControl();
1554 int left
= m_left
+ m_rowLabelWidth
;
1555 if ( m_dragRowOrCol
> 0 )
1556 left
= m_colRights
[m_dragRowOrCol
-1];
1557 m_colWidths
[m_dragRowOrCol
] = wxMax( ev
.GetX() - left
,
1558 WXGRID_MIN_COL_WIDTH
);
1560 ShowCellEditControl();
1563 // Note: we are ending the event *after* doing
1564 // default processing in this case
1566 SendEvent( EVT_GRID_COL_SIZE
, -1, m_dragRowOrCol
, ev
);
1571 case WXGRID_CURSOR_SELECT_CELL
:
1573 if ( IsSelection() )
1575 // Note: we are ending the event *after* doing
1576 // default processing in this case
1578 SendEvent( EVT_GRID_RANGE_SELECT
, -1, -1, ev
);
1586 // ------------------------------------------------------------
1588 // Right mouse button down
1590 else if ( ev
.RightDown() )
1594 wxGridCellCoords cellCoords
;
1596 switch( XYToArea( x
, y
) )
1599 case WXGRID_ROWLABEL
:
1602 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_CLICK
, row
, col
, ev
) )
1604 // TODO: default processing ?
1609 case WXGRID_COLLABEL
:
1612 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_CLICK
, row
, col
, ev
) )
1614 // TODO: default processing ?
1619 case WXGRID_CORNERLABEL
:
1621 // leave both row and col as -1
1623 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_CLICK
, row
, col
, ev
) )
1625 // TODO: default processing ?
1632 XYToCell( x
, y
, cellCoords
);
1633 if ( !SendEvent( EVT_GRID_CELL_RIGHT_CLICK
,
1634 cellCoords
.GetRow(),
1635 cellCoords
.GetCol(),
1638 // TODO: default processing ?
1645 wxLogMessage( "outside grid area" );
1650 // ------------------------------------------------------------
1652 // Right mouse button double click
1654 else if ( ev
.RightDClick() )
1658 wxGridCellCoords cellCoords
;
1660 switch( XYToArea( x
, y
) )
1663 case WXGRID_ROWLABEL
:
1666 SendEvent( EVT_GRID_LABEL_RIGHT_DCLICK
, row
, col
, ev
);
1670 case WXGRID_COLLABEL
:
1673 SendEvent( EVT_GRID_LABEL_RIGHT_DCLICK
, row
, col
, ev
);
1677 case WXGRID_CORNERLABEL
:
1679 // leave both row and col as -1
1681 SendEvent( EVT_GRID_LABEL_RIGHT_DCLICK
, row
, col
, ev
);
1687 XYToCell( x
, y
, cellCoords
);
1688 SendEvent( EVT_GRID_CELL_RIGHT_DCLICK
,
1689 cellCoords
.GetRow(),
1690 cellCoords
.GetCol(),
1697 wxLogMessage( "outside grid area" );
1702 // ------------------------------------------------------------
1704 // No buttons down and mouse moving
1706 else if ( ev
.Moving() )
1708 switch( XYToArea( x
, y
) )
1710 case WXGRID_ROWLABEL
:
1712 m_dragRowOrCol
= YToEdgeOfRow( y
);
1713 if ( m_dragRowOrCol
>= 0 )
1715 if ( m_cursorMode
== WXGRID_CURSOR_SELECT_CELL
)
1717 m_cursorMode
= WXGRID_CURSOR_RESIZE_ROW
;
1718 SetCursor( m_rowResizeCursor
);
1723 if ( m_cursorMode
!= WXGRID_CURSOR_SELECT_CELL
)
1725 m_cursorMode
= WXGRID_CURSOR_SELECT_CELL
;
1726 SetCursor( *wxSTANDARD_CURSOR
);
1732 case WXGRID_COLLABEL
:
1734 m_dragRowOrCol
= XToEdgeOfCol( x
);
1735 if ( m_dragRowOrCol
>= 0 )
1737 if ( m_cursorMode
== WXGRID_CURSOR_SELECT_CELL
)
1739 m_cursorMode
= WXGRID_CURSOR_RESIZE_COL
;
1740 SetCursor( m_colResizeCursor
);
1745 if ( m_cursorMode
!= WXGRID_CURSOR_SELECT_CELL
)
1747 m_cursorMode
= WXGRID_CURSOR_SELECT_CELL
;
1748 SetCursor( *wxSTANDARD_CURSOR
);
1756 if ( m_cursorMode
!= WXGRID_CURSOR_SELECT_CELL
)
1758 m_cursorMode
= WXGRID_CURSOR_SELECT_CELL
;
1759 SetCursor( *wxSTANDARD_CURSOR
);
1768 void wxGrid::OnKeyDown( wxKeyEvent
& ev
)
1770 if ( m_inOnKeyDown
)
1772 // shouldn't be here - we are going round in circles...
1774 wxLogFatalError( "wxGrid::OnKeyDown called while alread active" );
1777 m_inOnKeyDown
= TRUE
;
1779 // propagate the event up and see if it gets processed
1781 wxWindow
*parent
= GetParent();
1782 wxKeyEvent
keyEvt( ev
);
1783 keyEvt
.SetEventObject( parent
);
1785 if ( !parent
->GetEventHandler()->ProcessEvent( keyEvt
) )
1787 // try local handlers
1789 switch ( ev
.KeyCode() )
1792 if ( ev
.ControlDown() )
1794 MoveCursorUpBlock();
1803 if ( ev
.ControlDown() )
1805 MoveCursorDownBlock();
1814 if ( ev
.ControlDown() )
1816 MoveCursorLeftBlock();
1825 if ( ev
.ControlDown() )
1827 MoveCursorRightBlock();
1840 if ( ev
.ControlDown() )
1842 MakeCellVisible( 0, 0 );
1852 if ( ev
.ControlDown() )
1854 MakeCellVisible( m_numRows
-1, m_numCols
-1 );
1855 SelectCell( m_numRows
-1, m_numCols
-1 );
1872 // now try the cell edit control
1874 if ( IsCellEditControlEnabled() )
1876 ev
.SetEventObject( m_cellEditCtrl
);
1877 m_cellEditCtrl
->GetEventHandler()->ProcessEvent( ev
);
1883 m_inOnKeyDown
= FALSE
;
1887 // Text updated in an edit control - either a text control or a
1890 void wxGrid::OnText( wxKeyEvent
& ev
)
1895 wxWindow
*ctrl
= (wxWindow
*)ev
.GetEventObject();
1897 if ( ctrl
== m_cellEditCtrl
&&
1898 IsTopEditControlEnabled() )
1900 // set the value of the top edit control
1902 switch ( m_editCtrlType
)
1904 case wxGRID_TEXTCTRL
:
1905 ((wxTextCtrl
*)m_topEditCtrl
)->
1906 SetValue(((wxTextCtrl
*)ctrl
)->GetValue());
1909 case wxGRID_COMBOBOX
:
1910 ((wxComboBox
*)m_topEditCtrl
)->
1911 SetValue(((wxComboBox
*)ctrl
)->GetValue());
1915 else if ( ctrl
== m_topEditCtrl
&&
1916 IsCellEditControlEnabled() )
1918 switch ( m_editCtrlType
)
1920 case wxGRID_TEXTCTRL
:
1921 ((wxTextCtrl
*)m_cellEditCtrl
)->
1922 SetValue(((wxTextCtrl
*)ctrl
)->GetValue());
1925 case wxGRID_COMBOBOX
:
1926 ((wxComboBox
*)m_cellEditCtrl
)->
1927 SetValue(((wxComboBox
*)ctrl
)->GetValue());
1936 void wxGrid::OnGridScroll( wxScrollEvent
& ev
)
1938 // propagate the event up and see if it gets processed
1940 wxWindow
*parent
= GetParent();
1941 wxScrollEvent
scrollEvt( ev
);
1942 if (parent
->GetEventHandler()->ProcessEvent( scrollEvt
)) return;
1944 HideCellEditControl();
1946 if ( ev
.GetEventObject() == m_horizScrollBar
)
1948 if ( ev
.GetPosition() != m_scrollPosX
)
1950 SetHorizontalScrollPos( ev
.GetPosition() );
1955 if ( ev
.GetPosition() != m_scrollPosY
)
1957 SetVerticalScrollPos( ev
.GetPosition() );
1961 ShowCellEditControl();
1965 void wxGrid::SelectCell( const wxGridCellCoords
& coords
)
1967 wxClientDC
dc( this );
1969 if ( m_currentCellCoords
!= wxGridNoCellCoords
)
1971 HideCurrentCellHighlight( dc
);
1972 HideCellEditControl();
1973 SaveEditControlValue();
1976 m_currentCellCoords
= coords
;
1978 SetEditControlValue();
1981 ShowCellEditControl();
1982 ShowCurrentCellHighlight( dc
);
1985 if ( IsSelection() )
1988 if ( !GetBatchCount() ) Refresh();
1993 void wxGrid::ShowCellEditControl()
1997 if ( IsCellEditControlEnabled() )
1999 if ( !IsVisible( m_currentCellCoords
) )
2005 rect
= CellToRect( m_currentCellCoords
);
2007 m_cellEditCtrl
->SetSize( rect
);
2008 m_cellEditCtrl
->Show( TRUE
);
2010 switch ( m_editCtrlType
)
2012 case wxGRID_TEXTCTRL
:
2013 ((wxTextCtrl
*) m_cellEditCtrl
)->SetInsertionPointEnd();
2016 case wxGRID_CHECKBOX
:
2017 // TODO: anything ???
2022 // TODO: anything ???
2026 case wxGRID_COMBOBOX
:
2027 // TODO: anything ???
2032 m_cellEditCtrl
->SetFocus();
2038 void wxGrid::HideCellEditControl()
2040 if ( IsCellEditControlEnabled() )
2042 m_cellEditCtrl
->Show( FALSE
);
2046 void wxGrid::SetEditControlValue( const wxString
& value
)
2052 s
= GetCellValue(m_currentCellCoords
);
2056 if ( IsTopEditControlEnabled() )
2058 switch ( m_editCtrlType
)
2060 case wxGRID_TEXTCTRL
:
2061 ((wxGridTextCtrl
*)m_topEditCtrl
)->SetStartValue(s
);
2064 case wxGRID_CHECKBOX
:
2065 // TODO: implement this
2070 // TODO: implement this
2074 case wxGRID_COMBOBOX
:
2075 // TODO: implement this
2081 if ( IsCellEditControlEnabled() )
2083 switch ( m_editCtrlType
)
2085 case wxGRID_TEXTCTRL
:
2086 ((wxGridTextCtrl
*)m_cellEditCtrl
)->SetStartValue(s
);
2089 case wxGRID_CHECKBOX
:
2090 // TODO: implement this
2095 // TODO: implement this
2099 case wxGRID_COMBOBOX
:
2100 // TODO: implement this
2108 void wxGrid::SaveEditControlValue()
2112 wxWindow
*ctrl
= (wxWindow
*)NULL
;
2114 if ( IsCellEditControlEnabled() )
2116 ctrl
= m_cellEditCtrl
;
2118 else if ( IsTopEditControlEnabled() )
2120 ctrl
= m_topEditCtrl
;
2127 bool valueChanged
= FALSE
;
2129 switch ( m_editCtrlType
)
2131 case wxGRID_TEXTCTRL
:
2132 valueChanged
= (((wxGridTextCtrl
*)ctrl
)->GetValue() !=
2133 ((wxGridTextCtrl
*)ctrl
)->GetStartValue());
2134 SetCellValue( m_currentCellCoords
,
2135 ((wxTextCtrl
*) ctrl
)->GetValue() );
2138 case wxGRID_CHECKBOX
:
2139 // TODO: implement this
2144 // TODO: implement this
2148 case wxGRID_COMBOBOX
:
2149 // TODO: implement this
2156 SendEvent( EVT_GRID_CELL_CHANGE
,
2157 m_currentCellCoords
.GetRow(),
2158 m_currentCellCoords
.GetCol() );
2164 int wxGrid::XYToArea( int x
, int y
)
2166 if ( x
> m_left
&& x
< m_right
&&
2167 y
> m_top
&& y
< m_bottom
)
2169 if ( y
< m_top
+ m_colLabelHeight
)
2171 if ( x
> m_left
+ m_rowLabelWidth
)
2173 return WXGRID_COLLABEL
;
2177 return WXGRID_CORNERLABEL
;
2180 else if ( x
<= m_left
+ m_rowLabelWidth
)
2182 return WXGRID_ROWLABEL
;
2190 return WXGRID_NOAREA
;
2194 void wxGrid::XYToCell( int x
, int y
, wxGridCellCoords
& coords
)
2196 coords
.SetRow( YToRow(y
) );
2197 coords
.SetCol( XToCol(x
) );
2201 int wxGrid::YToRow( int y
)
2205 if ( y
> m_top
+ m_colLabelHeight
)
2207 for ( i
= m_scrollPosY
; i
< m_numRows
; i
++ )
2209 if ( y
< m_rowBottoms
[i
] )
2220 int wxGrid::XToCol( int x
)
2224 if ( x
> m_left
+ m_rowLabelWidth
)
2226 for ( i
= m_scrollPosX
; i
< m_numCols
; i
++ )
2228 if ( x
< m_colRights
[i
] )
2239 // return the row number that that the y coord is near the edge of, or
2240 // -1 if not near an edge
2242 int wxGrid::YToEdgeOfRow( int y
)
2246 if ( y
> m_top
+ m_colLabelHeight
)
2248 for ( i
= m_scrollPosY
; i
< m_numRows
; i
++ )
2250 if ( m_rowHeights
[i
] > WXGRID_LABEL_EDGE_ZONE
)
2252 d
= abs( y
- m_rowBottoms
[i
] );
2254 if ( d
< WXGRID_LABEL_EDGE_ZONE
) return i
;
2265 // return the col number that that the x coord is near the edge of, or
2266 // -1 if not near an edge
2268 int wxGrid::XToEdgeOfCol( int x
)
2272 if ( x
> m_left
+ m_rowLabelWidth
)
2274 for ( i
= m_scrollPosX
; i
< m_numCols
; i
++ )
2276 if ( m_colWidths
[i
] > WXGRID_LABEL_EDGE_ZONE
)
2278 d
= abs( x
- m_colRights
[i
] );
2280 if ( d
< WXGRID_LABEL_EDGE_ZONE
) return i
;
2291 wxRect
wxGrid::CellToRect( int row
, int col
)
2293 wxRect
rect( -1, -1, -1, -1 );
2295 if ( row
>= m_scrollPosY
&& col
>= m_scrollPosX
)
2297 rect
.x
= m_colRights
[col
] - m_colWidths
[col
];
2298 rect
.y
= m_rowBottoms
[row
] - m_rowHeights
[row
];
2299 rect
.width
= m_colWidths
[col
];
2300 rect
.height
= m_rowHeights
[ row
];
2307 bool wxGrid::MoveCursorUp()
2309 if ( m_currentCellCoords
!= wxGridNoCellCoords
&&
2310 m_currentCellCoords
.GetRow() > 0 )
2312 SelectCell( m_currentCellCoords
.GetRow() - 1,
2313 m_currentCellCoords
.GetCol() );
2315 if ( !IsVisible( m_currentCellCoords
) )
2316 MakeCellVisible( m_currentCellCoords
);
2324 bool wxGrid::MoveCursorDown()
2326 // TODO: allow for scrolling
2328 if ( m_currentCellCoords
!= wxGridNoCellCoords
&&
2329 m_currentCellCoords
.GetRow() < m_numRows
-1 )
2331 SelectCell( m_currentCellCoords
.GetRow() + 1,
2332 m_currentCellCoords
.GetCol() );
2334 if ( !IsVisible( m_currentCellCoords
) )
2335 MakeCellVisible( m_currentCellCoords
);
2343 bool wxGrid::MoveCursorLeft()
2345 if ( m_currentCellCoords
!= wxGridNoCellCoords
&&
2346 m_currentCellCoords
.GetCol() > 0 )
2348 SelectCell( m_currentCellCoords
.GetRow(),
2349 m_currentCellCoords
.GetCol() - 1 );
2351 if ( !IsVisible( m_currentCellCoords
) )
2352 MakeCellVisible( m_currentCellCoords
);
2360 bool wxGrid::MoveCursorRight()
2362 if ( m_currentCellCoords
!= wxGridNoCellCoords
&&
2363 m_currentCellCoords
.GetCol() < m_numCols
- 1 )
2365 SelectCell( m_currentCellCoords
.GetRow(),
2366 m_currentCellCoords
.GetCol() + 1 );
2368 if ( !IsVisible( m_currentCellCoords
) )
2369 MakeCellVisible( m_currentCellCoords
);
2377 bool wxGrid::MovePageUp()
2379 if ( m_currentCellCoords
!= wxGridNoCellCoords
&&
2382 int row
= m_currentCellCoords
.GetRow();
2383 int y
= m_rowBottoms
[ row
] - m_rowHeights
[ row
];
2386 if ( y
+ m_rowHeights
[row
-1] > m_bottom
) break;
2387 y
+= m_rowHeights
[ --row
];
2389 SetVerticalScrollPos( row
);
2391 SelectCell( row
, m_currentCellCoords
.GetCol() );
2398 bool wxGrid::MovePageDown()
2400 if ( m_currentCellCoords
!= wxGridNoCellCoords
&&
2401 m_scrollPosY
+ m_wholeRowsVisible
< m_numRows
)
2403 if ( m_wholeRowsVisible
> 0 )
2405 SetVerticalScrollPos( m_scrollPosY
+ m_wholeRowsVisible
);
2407 else if ( m_scrollPosY
< m_numRows
- 1 )
2409 SetVerticalScrollPos( m_scrollPosY
+ 1 );
2416 // m_scrollPosY will have been updated
2418 SelectCell( m_scrollPosY
, m_currentCellCoords
.GetCol() );
2425 bool wxGrid::MoveCursorUpBlock()
2428 m_currentCellCoords
!= wxGridNoCellCoords
&&
2429 m_currentCellCoords
.GetRow() > 0 )
2431 int row
= m_currentCellCoords
.GetRow();
2432 int col
= m_currentCellCoords
.GetCol();
2434 if ( m_table
->IsEmptyCell(row
, col
) )
2436 // starting in an empty cell: find the next block of
2442 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
2445 else if ( m_table
->IsEmptyCell(row
-1, col
) )
2447 // starting at the top of a block: find the next block
2453 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
2458 // starting within a block: find the top of the block
2463 if ( m_table
->IsEmptyCell(row
, col
) )
2471 SelectCell( row
, col
);
2473 if ( !IsVisible( m_currentCellCoords
) )
2474 MakeCellVisible( m_currentCellCoords
);
2482 bool wxGrid::MoveCursorDownBlock()
2485 m_currentCellCoords
!= wxGridNoCellCoords
&&
2486 m_currentCellCoords
.GetRow() < m_numRows
-1 )
2488 int row
= m_currentCellCoords
.GetRow();
2489 int col
= m_currentCellCoords
.GetCol();
2491 if ( m_table
->IsEmptyCell(row
, col
) )
2493 // starting in an empty cell: find the next block of
2496 while ( row
< m_numRows
-1 )
2499 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
2502 else if ( m_table
->IsEmptyCell(row
+1, col
) )
2504 // starting at the bottom of a block: find the next block
2507 while ( row
< m_numRows
-1 )
2510 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
2515 // starting within a block: find the bottom of the block
2517 while ( row
< m_numRows
-1 )
2520 if ( m_table
->IsEmptyCell(row
, col
) )
2528 SelectCell( row
, col
);
2530 if ( !IsVisible( m_currentCellCoords
) )
2531 MakeCellVisible( m_currentCellCoords
);
2539 bool wxGrid::MoveCursorLeftBlock()
2542 m_currentCellCoords
!= wxGridNoCellCoords
&&
2543 m_currentCellCoords
.GetCol() > 0 )
2545 int row
= m_currentCellCoords
.GetRow();
2546 int col
= m_currentCellCoords
.GetCol();
2548 if ( m_table
->IsEmptyCell(row
, col
) )
2550 // starting in an empty cell: find the next block of
2556 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
2559 else if ( m_table
->IsEmptyCell(row
, col
-1) )
2561 // starting at the left of a block: find the next block
2567 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
2572 // starting within a block: find the left of the block
2577 if ( m_table
->IsEmptyCell(row
, col
) )
2585 SelectCell( row
, col
);
2587 if ( !IsVisible( m_currentCellCoords
) )
2588 MakeCellVisible( m_currentCellCoords
);
2596 bool wxGrid::MoveCursorRightBlock()
2599 m_currentCellCoords
!= wxGridNoCellCoords
&&
2600 m_currentCellCoords
.GetCol() < m_numCols
-1 )
2602 int row
= m_currentCellCoords
.GetRow();
2603 int col
= m_currentCellCoords
.GetCol();
2605 if ( m_table
->IsEmptyCell(row
, col
) )
2607 // starting in an empty cell: find the next block of
2610 while ( col
< m_numCols
-1 )
2613 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
2616 else if ( m_table
->IsEmptyCell(row
, col
+1) )
2618 // starting at the right of a block: find the next block
2621 while ( col
< m_numCols
-1 )
2624 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break;
2629 // starting within a block: find the right of the block
2631 while ( col
< m_numCols
-1 )
2634 if ( m_table
->IsEmptyCell(row
, col
) )
2642 SelectCell( row
, col
);
2644 if ( !IsVisible( m_currentCellCoords
) )
2645 MakeCellVisible( m_currentCellCoords
);
2656 // ----- grid drawing functions
2659 void wxGrid::DrawLabelAreas( wxDC
& dc
)
2662 GetClientSize(&cw
, &ch
);
2664 dc
.SetPen(*wxTRANSPARENT_PEN
);
2665 dc
.SetBrush( wxBrush(GetLabelBackgroundColour(), wxSOLID
) );
2667 dc
.DrawRectangle( m_left
, m_top
,
2668 cw
- m_left
, m_colLabelHeight
);
2670 dc
.DrawRectangle( m_left
, m_top
,
2671 m_rowLabelWidth
, ch
- m_top
);
2675 void wxGrid::DrawColLabels( wxDC
& dc
)
2678 GetClientSize(&cw
, &ch
);
2680 if (m_colLabelHeight
== 0) return;
2682 DrawColLabelBorders( dc
);
2686 rect
.height
= m_colLabelHeight
- 1;
2688 int labelLeft
= m_left
+ m_rowLabelWidth
;
2691 for ( i
= m_scrollPosX
; i
< m_numCols
; i
++ )
2693 if ( labelLeft
> cw
) break;
2695 rect
.x
= 1 + labelLeft
;
2696 rect
.width
= m_colWidths
[i
];
2697 DrawColLabel( dc
, rect
, i
);
2699 labelLeft
+= m_colWidths
[i
];
2704 void wxGrid::DrawColLabelBorders( wxDC
& dc
)
2706 if ( m_colLabelHeight
<= 0 ) return;
2710 GetClientSize( &cw
, &ch
);
2712 dc
.SetPen( *wxBLACK_PEN
);
2716 dc
.DrawLine( m_left
, m_top
, cw
, m_top
);
2718 dc
.DrawLine( m_left
, m_top
+ m_colLabelHeight
,
2719 cw
, m_top
+ m_colLabelHeight
);
2723 int colLeft
= m_left
+ m_rowLabelWidth
;
2724 for ( i
= m_scrollPosX
; i
<= m_numCols
; i
++ )
2726 if (colLeft
> cw
) break;
2728 dc
.DrawLine( colLeft
, m_top
,
2729 colLeft
, m_top
+ m_colLabelHeight
);
2731 if ( i
< m_numCols
) colLeft
+= m_colWidths
[i
];
2734 // Draw white highlights for a 3d effect
2736 dc
.SetPen( *wxWHITE_PEN
);
2738 colLeft
= m_left
+ m_rowLabelWidth
;
2739 for ( i
= m_scrollPosX
; i
< m_numCols
; i
++ )
2741 if (colLeft
> cw
) break;
2743 dc
.DrawLine(colLeft
+ 1, m_top
+ 1,
2744 colLeft
+ m_colWidths
[i
], m_top
+ 1);
2746 dc
.DrawLine(colLeft
+ 1, m_top
+ 1,
2747 colLeft
+ 1, m_top
+ m_colLabelHeight
);
2749 colLeft
+= m_colWidths
[i
];
2754 void wxGrid::DrawColLabel( wxDC
& dc
, const wxRect
& rect
, int col
)
2763 dc
.SetBackgroundMode( wxTRANSPARENT
);
2764 dc
.SetTextBackground( GetLabelBackgroundColour() );
2765 dc
.SetTextForeground( GetLabelTextColour() );
2766 dc
.SetFont( GetLabelFont() );
2769 GetColLabelAlignment( &hAlign
, &vAlign
);
2770 DrawTextRectangle( dc
, GetColLabelValue( col
), rect2
, hAlign
, vAlign
);
2774 void wxGrid::DrawRowLabels( wxDC
& dc
)
2777 GetClientSize(&cw
, &ch
);
2779 if (m_rowLabelWidth
== 0) return;
2781 DrawRowLabelBorders( dc
);
2784 rect
.x
= m_left
+ 1;
2785 rect
.width
= m_rowLabelWidth
- 1;
2787 int labelTop
= m_top
+ m_colLabelHeight
;
2790 for ( i
= m_scrollPosY
; i
< m_numRows
; i
++ )
2792 if ( labelTop
> ch
) break;
2794 rect
.y
= 1 + labelTop
;
2795 rect
.height
= m_rowHeights
[i
];
2796 DrawRowLabel( dc
, rect
, i
);
2798 labelTop
+= m_rowHeights
[i
];
2803 void wxGrid::DrawRowLabelBorders( wxDC
& dc
)
2805 if ( m_rowLabelWidth
<= 0 ) return;
2809 GetClientSize( &cw
, &ch
);
2811 dc
.SetPen( *wxBLACK_PEN
);
2815 dc
.DrawLine( m_left
, m_top
, m_left
, ch
);
2817 dc
.DrawLine( m_left
+ m_rowLabelWidth
, m_top
,
2818 m_left
+ m_rowLabelWidth
, ch
);
2822 int rowTop
= m_top
+ m_colLabelHeight
;
2823 for ( i
= m_scrollPosY
; i
<= m_numRows
; i
++ )
2825 if ( rowTop
> ch
) break;
2827 dc
.DrawLine( m_left
, rowTop
,
2828 m_left
+ m_rowLabelWidth
, rowTop
);
2830 if ( i
< m_numRows
) rowTop
+= m_rowHeights
[i
];
2833 // Draw white highlights for a 3d effect
2835 dc
.SetPen( *wxWHITE_PEN
);
2837 rowTop
= m_top
+ m_colLabelHeight
;
2838 for ( i
= m_scrollPosY
; i
< m_numRows
; i
++ )
2840 if ( rowTop
> ch
) break;
2842 dc
.DrawLine( m_left
+ 1, rowTop
+ 1,
2843 m_left
+ m_rowLabelWidth
, rowTop
+ 1 );
2845 dc
.DrawLine( m_left
+ 1, rowTop
+ 1,
2846 m_left
+ 1, rowTop
+ m_rowHeights
[i
] );
2848 rowTop
+= m_rowHeights
[i
];
2853 void wxGrid::DrawRowLabel( wxDC
& dc
, const wxRect
& rect
, int row
)
2862 dc
.SetBackgroundMode( wxTRANSPARENT
);
2863 dc
.SetTextBackground( GetLabelBackgroundColour() );
2864 dc
.SetTextForeground( GetLabelTextColour() );
2865 dc
.SetFont( GetLabelFont() );
2868 GetRowLabelAlignment( &hAlign
, &vAlign
);
2869 DrawTextRectangle( dc
, GetRowLabelValue( row
), rect2
, hAlign
, vAlign
);
2873 void wxGrid::DrawCellArea( wxDC
& dc
)
2876 GetClientSize(&cw
, &ch
);
2878 dc
.SetPen( *wxTRANSPARENT_PEN
);
2879 dc
.SetBrush( wxBrush(GetDefaultCellBackgroundColour(), wxSOLID
) );
2881 int left
= m_left
+ m_rowLabelWidth
+ 1;
2882 int top
= m_top
+ m_colLabelHeight
+ 1;
2884 dc
.DrawRectangle( left
, top
, cw
- left
, ch
- top
);
2888 void wxGrid::DrawGridLines( wxDC
& dc
)
2890 if ( !m_gridLinesEnabled
|| !m_numRows
|| !m_numCols
) return;
2894 GetClientSize(&cw
, &ch
);
2896 dc
.SetPen( wxPen(GetGridLineColour(), 1, wxSOLID
) );
2898 // horizontal grid lines
2900 int rowTop
= m_top
+ m_colLabelHeight
+ m_rowHeights
[m_scrollPosY
];
2901 for ( i
= m_scrollPosY
+ 1; i
<= m_numRows
; i
++ )
2903 if ( rowTop
> ch
) break;
2905 dc
.DrawLine( m_left
+ m_rowLabelWidth
+ 1, rowTop
,
2908 if ( i
< m_numRows
) rowTop
+= m_rowHeights
[i
];
2912 // vertical grid lines
2914 int colLeft
= m_left
+ m_rowLabelWidth
+ m_colWidths
[m_scrollPosX
];
2915 for ( i
= m_scrollPosX
+ 1; i
<= m_numCols
; i
++ )
2917 if ( colLeft
> cw
) break;
2919 dc
.DrawLine( colLeft
, m_top
+ m_colLabelHeight
+ 1,
2920 colLeft
, m_bottom
);
2922 if ( i
< m_numCols
) colLeft
+= m_colWidths
[i
];
2927 void wxGrid::DrawCells( wxDC
& dc
)
2929 if ( !m_numRows
|| !m_numCols
) return;
2934 GetClientSize( &cw
, &ch
);
2940 rect
.y
= m_top
+ m_colLabelHeight
;
2941 for ( row
= m_scrollPosY
; row
< m_numRows
; row
++ )
2943 if ( rect
.y
> ch
) break;
2945 rect
.height
= m_rowHeights
[ row
];
2946 rect
.x
= m_left
+ m_rowLabelWidth
;
2948 for ( col
= m_scrollPosX
; col
< m_numCols
; col
++ )
2950 if ( rect
.x
> cw
) break;
2952 rect
.width
= m_colWidths
[col
];
2953 DrawCellBackground( dc
, rect
, row
, col
);
2954 DrawCellValue( dc
, rect
, row
, col
);
2955 rect
.x
+= rect
.width
;
2957 rect
.y
+= rect
.height
;
2963 void wxGrid::DrawCellBackground( wxDC
& dc
, const wxRect
& rect
, int row
, int col
)
2972 dc
.SetBackgroundMode( wxSOLID
);
2974 if ( IsInSelection( row
, col
) )
2976 // TODO: improve this
2978 dc
.SetBrush( *wxBLACK_BRUSH
);
2982 dc
.SetBrush( wxBrush(GetCellBackgroundColour(row
, col
), wxSOLID
) );
2984 dc
.SetPen( *wxTRANSPARENT_PEN
);
2985 dc
.DrawRectangle( rect2
);
2989 void wxGrid::DrawCellValue( wxDC
& dc
, const wxRect
& rect
, int row
, int col
)
2998 dc
.SetBackgroundMode( wxTRANSPARENT
);
3000 if ( IsInSelection( row
, col
) )
3002 // TODO: improve this
3004 dc
.SetTextBackground( wxColour(0, 0, 0) );
3005 dc
.SetTextForeground( wxColour(255, 255, 255) );
3009 dc
.SetTextBackground( GetCellBackgroundColour(row
, col
) );
3010 dc
.SetTextForeground( GetCellTextColour(row
, col
) );
3012 dc
.SetFont( GetCellFont(row
, col
) );
3015 GetCellAlignment( row
, col
, &hAlign
, &vAlign
);
3016 DrawTextRectangle( dc
, GetCellValue( row
, col
), rect2
, hAlign
, vAlign
);
3020 void wxGrid::DrawCellHighlight( wxDC
& dc
, int row
, int col
)
3022 // TODO: bounds checking on row, col ?
3025 if ( row
>= m_scrollPosY
&& col
>= m_scrollPosX
)
3030 GetClientSize( &cw
, &ch
);
3032 x
= m_colRights
[col
] - m_colWidths
[col
];
3033 if ( x
>= cw
) return;
3035 y
= m_rowBottoms
[row
] - m_rowHeights
[row
];
3036 if ( y
>= ch
) return;
3038 dc
.SetLogicalFunction( wxXOR
);
3039 dc
.SetPen( wxPen(GetCellHighlightColour(), 2, wxSOLID
) );
3040 dc
.SetBrush( *wxTRANSPARENT_BRUSH
);
3042 dc
.DrawRectangle( x
, y
,
3043 m_colWidths
[col
] + 2,
3044 m_rowHeights
[row
] + 2 );
3046 dc
.SetLogicalFunction( wxCOPY
);
3051 // This function is handy when you just want to update one or a few
3052 // cells. For example, it is used by SetCellValue()
3054 void wxGrid::DrawCell( int row
, int col
)
3056 if ( !GetBatchCount() )
3058 if ( !IsVisible( wxGridCellCoords(row
, col
) ) ) return;
3061 GetClientSize( &cw
, &ch
);
3063 wxRect
rect( CellToRect( row
, col
) );
3067 wxClientDC
dc( this );
3068 DrawCellBackground( dc
, rect
, row
, col
);
3069 DrawCellValue( dc
, rect
, row
, col
);
3075 // this is just to make other code more obvious
3077 void wxGrid::HideCurrentCellHighlight( wxDC
& dc
)
3079 if ( m_currentCellHighlighted
&&
3080 m_currentCellCoords
!= wxGridNoCellCoords
)
3082 DrawCellHighlight( dc
, m_currentCellCoords
);
3083 m_currentCellHighlighted
= FALSE
;
3088 // this is just to make other code more obvious
3090 void wxGrid::ShowCurrentCellHighlight( wxDC
& dc
)
3092 if ( !m_currentCellHighlighted
&&
3093 m_currentCellCoords
!= wxGridNoCellCoords
)
3095 DrawCellHighlight( dc
, m_currentCellCoords
);
3096 m_currentCellHighlighted
= TRUE
;
3101 void wxGrid::DrawTextRectangle( wxDC
& dc
,
3102 const wxString
& value
,
3107 long textWidth
, textHeight
;
3108 long lineWidth
, lineHeight
;
3109 wxArrayString lines
;
3111 // see if we are already clipping
3114 dc
.GetClippingBox( clipRect
);
3116 bool alreadyClipping
= TRUE
;
3117 wxRect intersectRect
;
3119 if ( clipRect
.x
== 0 && clipRect
.y
== 0 &&
3120 clipRect
.width
== 0 && clipRect
.height
== 0)
3122 alreadyClipping
= FALSE
;
3123 intersectRect
= rect
;
3127 // Find the intersection of the clipping rectangle and our
3130 wxRegion
region( rect
);
3131 region
.Intersect( clipRect
);
3132 if ( region
.IsEmpty() )
3138 intersectRect
= region
.GetBox();
3141 if ( alreadyClipping
) dc
.DestroyClippingRegion();
3143 dc
.SetClippingRegion( intersectRect
);
3145 StringToLines( value
, lines
);
3146 if ( lines
.GetCount() )
3148 GetTextBoxSize( dc
, lines
, &textWidth
, &textHeight
);
3149 dc
.GetTextExtent( lines
[0], &lineWidth
, &lineHeight
);
3152 switch ( horizAlign
)
3155 x
= rect
.x
+ (rect
.width
- textWidth
- 1.0);
3159 x
= rect
.x
+ ((rect
.width
- textWidth
)/2.0);
3168 switch ( vertAlign
)
3171 y
= rect
.y
+ (rect
.height
- textHeight
- 1);
3175 y
= rect
.y
+ ((rect
.height
- textHeight
)/2.0);
3184 for ( size_t i
= 0; i
< lines
.GetCount(); i
++ )
3186 dc
.DrawText( lines
[i
], (long)x
, (long)y
);
3191 dc
.DestroyClippingRegion();
3192 if (alreadyClipping
) dc
.SetClippingRegion( clipRect
);
3196 // Split multi line text up into an array of strings. Any existing
3197 // contents of the string array are preserved.
3199 void wxGrid::StringToLines( const wxString
& value
, wxArrayString
& lines
)
3201 // TODO: this won't work for WXMAC ? (lines end with '\r')
3202 // => use wxTextFile functions then (VZ)
3205 while ( startPos
< (int)value
.Length() )
3207 pos
= value
.Mid(startPos
).Find( '\n' );
3212 else if ( pos
== 0 )
3214 lines
.Add( wxEmptyString
);
3218 if ( value
[startPos
+pos
-1] == '\r' )
3220 lines
.Add( value
.Mid(startPos
, pos
-1) );
3224 lines
.Add( value
.Mid(startPos
, pos
) );
3229 if ( startPos
< (int)value
.Length() )
3231 lines
.Add( value
.Mid( startPos
) );
3236 void wxGrid::GetTextBoxSize( wxDC
& dc
,
3237 wxArrayString
& lines
,
3238 long *width
, long *height
)
3245 for ( i
= 0; i
< lines
.GetCount(); i
++ )
3247 dc
.GetTextExtent( lines
[i
], &lineW
, &lineH
);
3248 w
= wxMax( w
, lineW
);
3258 // ------ functions to get/send data (see also public functions)
3261 bool wxGrid::GetModelValues()
3265 // all we need to do is repaint the grid
3275 bool wxGrid::SetModelValues()
3281 for ( row
= m_scrollPosY
; row
< m_numRows
; row
++ )
3283 for ( col
= m_scrollPosX
; col
< m_numCols
; col
++ )
3285 m_table
->SetValue( row
, col
, GetCellValue(row
, col
) );
3297 // ------ public functions
3300 bool wxGrid::CreateGrid( int numRows
, int numCols
)
3304 wxLogError( "wxGrid::CreateGrid(numRows, numCols) called more than once" );
3309 m_numRows
= numRows
;
3310 m_numCols
= numCols
;
3312 m_table
= new wxGridStringTable( m_numRows
, m_numCols
);
3313 m_table
->SetView( this );
3322 // The behaviour of this function depends on the grid table class
3323 // Clear() function. For the default wxGridStringTable class the
3324 // behavious is to replace all cell contents with wxEmptyString but
3325 // not to change the number of rows or cols.
3327 void wxGrid::ClearGrid()
3332 SetEditControlValue();
3333 if ( !GetBatchCount() ) Refresh();
3338 bool wxGrid::InsertRows( int pos
, int numRows
, bool WXUNUSED(updateLabels
) )
3340 // TODO: something with updateLabels flag
3344 wxLogError( "Called wxGrid::InsertRows() before calling CreateGrid()" );
3350 bool ok
= m_table
->InsertRows( pos
, numRows
);
3352 // the table will have sent the results of the insert row
3353 // operation to this view object as a grid table message
3357 if ( m_numCols
== 0 )
3359 m_table
->AppendCols( WXGRID_DEFAULT_NUMBER_COLS
);
3361 // TODO: perhaps instead of appending the default number of cols
3362 // we should remember what the last non-zero number of cols was ?
3366 if ( m_currentCellCoords
== wxGridNoCellCoords
)
3368 // if we have just inserted cols into an empty grid the current
3369 // cell will be undefined...
3374 if ( !GetBatchCount() ) Refresh();
3377 SetEditControlValue();
3386 bool wxGrid::AppendRows( int numRows
, bool WXUNUSED(updateLabels
) )
3388 // TODO: something with updateLabels flag
3392 wxLogError( "Called wxGrid::AppendRows() before calling CreateGrid()" );
3396 if ( m_table
&& m_table
->AppendRows( numRows
) )
3398 if ( m_currentCellCoords
== wxGridNoCellCoords
)
3400 // if we have just inserted cols into an empty grid the current
3401 // cell will be undefined...
3406 // the table will have sent the results of the append row
3407 // operation to this view object as a grid table message
3409 if ( !GetBatchCount() ) Refresh();
3418 bool wxGrid::DeleteRows( int pos
, int numRows
, bool WXUNUSED(updateLabels
) )
3420 // TODO: something with updateLabels flag
3424 wxLogError( "Called wxGrid::DeleteRows() before calling CreateGrid()" );
3428 if ( m_table
&& m_table
->DeleteRows( pos
, numRows
) )
3430 // the table will have sent the results of the delete row
3431 // operation to this view object as a grid table message
3433 if ( m_numRows
> 0 )
3434 SetEditControlValue();
3436 HideCellEditControl();
3438 if ( !GetBatchCount() ) Refresh();
3447 bool wxGrid::InsertCols( int pos
, int numCols
, bool WXUNUSED(updateLabels
) )
3449 // TODO: something with updateLabels flag
3453 wxLogError( "Called wxGrid::InsertCols() before calling CreateGrid()" );
3459 HideCellEditControl();
3460 bool ok
= m_table
->InsertCols( pos
, numCols
);
3462 // the table will have sent the results of the insert col
3463 // operation to this view object as a grid table message
3467 if ( m_currentCellCoords
== wxGridNoCellCoords
)
3469 // if we have just inserted cols into an empty grid the current
3470 // cell will be undefined...
3475 if ( !GetBatchCount() ) Refresh();
3478 SetEditControlValue();
3487 bool wxGrid::AppendCols( int numCols
, bool WXUNUSED(updateLabels
) )
3489 // TODO: something with updateLabels flag
3493 wxLogError( "Called wxGrid::AppendCols() before calling CreateGrid()" );
3497 if ( m_table
&& m_table
->AppendCols( numCols
) )
3499 // the table will have sent the results of the append col
3500 // operation to this view object as a grid table message
3502 if ( m_currentCellCoords
== wxGridNoCellCoords
)
3504 // if we have just inserted cols into an empty grid the current
3505 // cell will be undefined...
3509 if ( !GetBatchCount() ) Refresh();
3518 bool wxGrid::DeleteCols( int pos
, int numCols
, bool WXUNUSED(updateLabels
) )
3520 // TODO: something with updateLabels flag
3524 wxLogError( "Called wxGrid::DeleteCols() before calling CreateGrid()" );
3528 if ( m_table
&& m_table
->DeleteCols( pos
, numCols
) )
3530 // the table will have sent the results of the delete col
3531 // operation to this view object as a grid table message
3533 if ( m_numCols
> 0 )
3534 SetEditControlValue();
3536 HideCellEditControl();
3538 if ( !GetBatchCount() ) Refresh();
3550 // ------ control panel and cell edit control
3553 void wxGrid::EnableEditing( bool edit
)
3555 // TODO: improve this ?
3557 if ( edit
!= m_editable
)
3560 if ( !m_editable
) HideCellEditControl();
3561 m_topEditCtrlEnabled
= m_editable
;
3562 m_cellEditCtrlEnabled
= m_editable
;
3563 if ( !m_editable
) ShowCellEditControl();
3568 void wxGrid::EnableTopEditControl( bool enable
)
3570 if ( enable
!= m_topEditCtrlEnabled
)
3572 HideCellEditControl();
3573 m_topEditCtrlEnabled
= enable
;
3575 m_topEditCtrl
->Show( enable
);
3577 if ( m_currentCellCoords
!= wxGridNoCellCoords
)
3578 SetEditControlValue();
3580 ShowCellEditControl();
3581 if ( !GetBatchCount() ) Refresh();
3585 void wxGrid::EnableCellEditControl( bool enable
)
3587 if ( m_cellEditCtrl
&&
3588 enable
!= m_cellEditCtrlEnabled
)
3590 HideCellEditControl();
3591 SaveEditControlValue();
3593 m_cellEditCtrlEnabled
= enable
;
3595 SetEditControlValue();
3596 ShowCellEditControl();
3602 // ------ grid formatting functions
3605 void wxGrid::GetRowLabelAlignment( int *horiz
, int *vert
)
3607 *horiz
= m_rowLabelHorizAlign
;
3608 *vert
= m_rowLabelVertAlign
;
3611 void wxGrid::GetColLabelAlignment( int *horiz
, int *vert
)
3613 *horiz
= m_colLabelHorizAlign
;
3614 *vert
= m_colLabelVertAlign
;
3617 wxString
wxGrid::GetRowLabelValue( int row
)
3621 return m_table
->GetRowLabelValue( row
);
3631 wxString
wxGrid::GetColLabelValue( int col
)
3635 return m_table
->GetColLabelValue( col
);
3645 void wxGrid::SetRowLabelSize( int width
)
3647 m_rowLabelWidth
= wxMax( 0, width
);
3649 ShowCellEditControl();
3650 if ( !GetBatchCount() ) Refresh();
3653 void wxGrid::SetColLabelSize( int height
)
3655 m_colLabelHeight
= wxMax( 0, height
);
3657 ShowCellEditControl();
3658 if ( !GetBatchCount() ) Refresh();
3661 void wxGrid::SetLabelBackgroundColour( const wxColour
& colour
)
3663 m_labelBackgroundColour
= colour
;
3664 if ( !GetBatchCount() ) Refresh();
3667 void wxGrid::SetLabelTextColour( const wxColour
& colour
)
3669 m_labelTextColour
= colour
;
3670 if ( !GetBatchCount() ) Refresh();
3673 void wxGrid::SetLabelFont( const wxFont
& font
)
3676 if ( !GetBatchCount() ) Refresh();
3679 void wxGrid::SetRowLabelAlignment( int horiz
, int vert
)
3681 if ( horiz
== wxLEFT
|| horiz
== wxCENTRE
|| horiz
== wxRIGHT
)
3683 m_rowLabelHorizAlign
= horiz
;
3686 if ( vert
== wxTOP
|| vert
== wxCENTRE
|| vert
== wxBOTTOM
)
3688 m_rowLabelVertAlign
= vert
;
3691 if ( !GetBatchCount() ) Refresh();
3694 void wxGrid::SetColLabelAlignment( int horiz
, int vert
)
3696 if ( horiz
== wxLEFT
|| horiz
== wxCENTRE
|| horiz
== wxRIGHT
)
3698 m_colLabelHorizAlign
= horiz
;
3701 if ( vert
== wxTOP
|| vert
== wxCENTRE
|| vert
== wxBOTTOM
)
3703 m_colLabelVertAlign
= vert
;
3706 if ( !GetBatchCount() ) Refresh();
3709 void wxGrid::SetRowLabelValue( int row
, const wxString
& s
)
3713 m_table
->SetRowLabelValue( row
, s
);
3714 if ( !GetBatchCount() ) Refresh();
3718 void wxGrid::SetColLabelValue( int col
, const wxString
& s
)
3722 m_table
->SetColLabelValue( col
, s
);
3723 if ( !GetBatchCount() ) Refresh();
3727 void wxGrid::SetGridLineColour( const wxColour
& colour
)
3729 m_gridLineColour
= colour
;
3731 wxClientDC
dc( this );
3732 DrawGridLines( dc
);
3735 void wxGrid::EnableGridLines( bool enable
)
3737 if ( enable
!= m_gridLinesEnabled
)
3739 m_gridLinesEnabled
= enable
;
3740 if ( !GetBatchCount() ) Refresh();
3745 int wxGrid::GetDefaultRowSize()
3747 return m_defaultRowHeight
;
3750 int wxGrid::GetRowSize( int row
)
3752 if ( row
>= 0 && row
< m_numRows
)
3753 return m_rowHeights
[row
];
3755 return 0; // TODO: log an error here
3758 int wxGrid::GetDefaultColSize()
3760 return m_defaultColWidth
;
3763 int wxGrid::GetColSize( int col
)
3765 if ( col
>= 0 && col
< m_numCols
)
3766 return m_colWidths
[col
];
3768 return 0; // TODO: log an error here
3771 wxColour
wxGrid::GetDefaultCellBackgroundColour()
3773 // TODO: replace this temp test code
3775 return wxColour( 255, 255, 255 );
3778 wxColour
wxGrid::GetCellBackgroundColour( int row
, int col
)
3780 // TODO: replace this temp test code
3782 return wxColour( 255, 255, 255 );
3785 wxColour
wxGrid::GetDefaultCellTextColour()
3787 // TODO: replace this temp test code
3789 return wxColour( 0, 0, 0 );
3792 wxColour
wxGrid::GetCellTextColour( int row
, int col
)
3794 // TODO: replace this temp test code
3796 return wxColour( 0, 0, 0 );
3800 wxColour
wxGrid::GetCellHighlightColour()
3802 // TODO: replace this temp test code
3804 return wxColour( 0, 0, 0 );
3808 wxFont
wxGrid::GetDefaultCellFont()
3810 return m_defaultCellFont
;
3813 wxFont
wxGrid::GetCellFont( int row
, int col
)
3815 // TODO: replace this temp test code
3817 return m_defaultCellFont
;
3820 void wxGrid::GetDefaultCellAlignment( int *horiz
, int *vert
)
3822 // TODO: replace this temp test code
3828 void wxGrid::GetCellAlignment( int row
, int col
, int *horiz
, int *vert
)
3830 // TODO: replace this temp test code
3836 void wxGrid::SetDefaultRowSize( int height
, bool resizeExistingRows
)
3838 m_defaultRowHeight
= wxMax( height
, WXGRID_MIN_ROW_HEIGHT
);
3840 if ( resizeExistingRows
)
3842 // TODO: what do we do about events here ?
3843 // Generate an event for each resize ?
3846 for ( row
= 0; row
< m_numRows
; row
++ )
3848 m_rowHeights
[row
] = m_defaultRowHeight
;
3851 if ( !GetBatchCount() ) Refresh();
3855 void wxGrid::SetRowSize( int row
, int height
)
3857 if ( row
>= 0 && row
< m_numRows
)
3859 m_rowHeights
[row
] = wxMax( 0, height
);
3861 if ( !GetBatchCount() ) Refresh();
3863 // Note: we are ending the event *after* doing
3864 // default processing in this case
3866 SendEvent( EVT_GRID_ROW_SIZE
,
3871 // TODO: log an error here
3875 void wxGrid::SetDefaultColSize( int width
, bool resizeExistingCols
)
3877 m_defaultColWidth
= wxMax( width
, WXGRID_MIN_COL_WIDTH
);
3879 if ( resizeExistingCols
)
3881 // TODO: what do we do about events here ?
3882 // Generate an event for each resize ?
3885 for ( col
= 0; col
< m_numCols
; col
++ )
3887 m_colWidths
[col
] = m_defaultColWidth
;
3890 if ( !GetBatchCount() ) Refresh();
3894 void wxGrid::SetColSize( int col
, int width
)
3896 if ( col
>= 0 && col
< m_numCols
)
3898 m_colWidths
[col
] = wxMax( 0, width
);
3900 if ( !GetBatchCount() ) Refresh();
3902 // Note: we are ending the event *after* doing
3903 // default processing in this case
3905 SendEvent( EVT_GRID_COL_SIZE
,
3910 // TODO: log an error here
3914 void wxGrid::SetDefaultCellBackgroundColour( const wxColour
& )
3916 // TODO: everything !!!
3920 void wxGrid::SetCellBackgroundColour( int row
, int col
, const wxColour
& )
3922 // TODO: everything !!!
3926 void wxGrid::SetDefaultCellTextColour( const wxColour
& )
3928 // TODO: everything !!!
3932 void wxGrid::SetCellTextColour( int row
, int col
, const wxColour
& )
3934 // TODO: everything !!!
3938 void wxGrid::SetCellHighlightColour( const wxColour
& )
3940 // TODO: everything !!!
3944 void wxGrid::SetDefaultCellFont( const wxFont
& )
3946 // TODO: everything !!!
3950 void wxGrid::SetCellFont( int row
, int col
, const wxFont
& )
3952 // TODO: everything !!!
3956 void wxGrid::SetDefaultCellAlignment( int horiz
, int vert
)
3958 // TODO: everything !!!
3962 void wxGrid::SetCellAlignment( int row
, int col
, int horiz
, int vert
)
3964 // TODO: everything !!!
3970 // ------ cell value accessor functions
3973 void wxGrid::SetCellValue( int row
, int col
, const wxString
& s
)
3977 m_table
->SetValue( row
, col
, s
.c_str() );
3978 DrawCell( row
, col
);
3979 if ( m_currentCellCoords
.GetRow() == row
&&
3980 m_currentCellCoords
.GetCol() == col
)
3982 SetEditControlValue( s
);
3990 // ------ interaction with data model
3992 bool wxGrid::ProcessTableMessage( wxGridTableMessage
& msg
)
3994 switch ( msg
.GetId() )
3996 case wxGRIDTABLE_REQUEST_VIEW_GET_VALUES
:
3997 return GetModelValues();
3999 case wxGRIDTABLE_REQUEST_VIEW_SEND_VALUES
:
4000 return SetModelValues();
4002 case wxGRIDTABLE_NOTIFY_ROWS_INSERTED
:
4003 case wxGRIDTABLE_NOTIFY_ROWS_APPENDED
:
4004 case wxGRIDTABLE_NOTIFY_ROWS_DELETED
:
4005 case wxGRIDTABLE_NOTIFY_COLS_INSERTED
:
4006 case wxGRIDTABLE_NOTIFY_COLS_APPENDED
:
4007 case wxGRIDTABLE_NOTIFY_COLS_DELETED
:
4008 return Redimension( msg
);
4017 // ------ Grid location functions
4019 // (see also inline functions in grid.h)
4022 // check to see if a cell location is wholly visible
4024 bool wxGrid::IsVisible( const wxGridCellCoords
& coords
)
4026 return ( coords
.GetRow() >= m_scrollPosY
&&
4027 coords
.GetRow() < m_scrollPosY
+ m_wholeRowsVisible
&&
4028 coords
.GetCol() >= m_scrollPosX
&&
4029 coords
.GetCol() < m_scrollPosX
+ m_wholeColsVisible
);
4033 // make the specified cell location visible by doing a minimal amount
4036 void wxGrid::MakeCellVisible( int row
, int col
)
4038 int lastX
= m_scrollPosX
;
4039 int lastY
= m_scrollPosY
;
4041 if ( row
>= 0 && row
< m_numRows
&&
4042 col
>= 0 && col
< m_numCols
)
4044 if ( row
< m_scrollPosY
)
4046 SetVerticalScrollPos( row
);
4048 else if ( row
>= m_scrollPosY
+ m_wholeRowsVisible
)
4051 int h
= m_rowBottoms
[row
];
4052 for ( i
= m_scrollPosY
; i
< m_numRows
&& h
> m_bottom
; i
++ )
4054 h
-= m_rowHeights
[i
];
4056 SetVerticalScrollPos( i
);
4059 if ( col
< m_scrollPosX
)
4061 SetHorizontalScrollPos( col
);
4063 else if ( col
>= m_scrollPosX
+ m_wholeColsVisible
)
4066 int w
= m_colRights
[col
];
4067 for ( i
= m_scrollPosX
; i
< m_numCols
&& w
> m_right
; i
++ )
4069 w
-= m_colWidths
[i
];
4071 SetHorizontalScrollPos( i
);
4074 if ( m_scrollPosX
!= lastX
|| m_scrollPosY
!= lastY
)
4076 // The cell was not visible before but not it is
4078 ShowCellEditControl();
4083 // TODO: log an error
4088 void wxGrid::SetVerticalScrollPos( int topMostRow
)
4090 if ( m_vertScrollBar
&& topMostRow
!= m_scrollPosY
)
4092 m_scrollPosY
= topMostRow
;
4100 void wxGrid::SetHorizontalScrollPos( int leftMostCol
)
4102 if ( m_horizScrollBar
&& leftMostCol
!= m_scrollPosX
)
4104 m_scrollPosX
= leftMostCol
;
4113 // ------ block, row and col selection
4116 void wxGrid::SelectRow( int row
, bool addToSelected
)
4118 if ( IsSelection() && addToSelected
)
4120 if ( m_selectedTopLeft
.GetRow() > row
)
4121 m_selectedTopLeft
.SetRow( row
);
4123 m_selectedTopLeft
.SetCol( 0 );
4125 if ( m_selectedBottomRight
.GetRow() < row
)
4126 m_selectedBottomRight
.SetRow( row
);
4128 m_selectedBottomRight
.SetCol( m_numCols
- 1 );
4133 m_selectedTopLeft
.Set( row
, 0 );
4134 m_selectedBottomRight
.Set( row
, m_numCols
-1 );
4137 if ( !GetBatchCount() )
4139 wxRect
rect( SelectionToRect() );
4140 if ( rect
!= wxGridNoCellRect
) Refresh( TRUE
, &rect
);
4143 wxGridRangeSelectEvent
gridEvt( GetId(),
4144 EVT_GRID_RANGE_SELECT
,
4147 m_selectedBottomRight
);
4149 GetEventHandler()->ProcessEvent(gridEvt
);
4153 void wxGrid::SelectCol( int col
, bool addToSelected
)
4155 if ( addToSelected
&& m_selectedTopLeft
!= wxGridNoCellCoords
)
4157 if ( m_selectedTopLeft
.GetCol() > col
)
4158 m_selectedTopLeft
.SetCol( col
);
4160 m_selectedTopLeft
.SetRow( 0 );
4162 if ( m_selectedBottomRight
.GetCol() < col
)
4163 m_selectedBottomRight
.SetCol( col
);
4165 m_selectedBottomRight
.SetRow( m_numRows
- 1 );
4170 m_selectedTopLeft
.Set( 0, col
);
4171 m_selectedBottomRight
.Set( m_numRows
-1, col
);
4174 if ( !GetBatchCount() )
4176 wxRect
rect( SelectionToRect() );
4177 if ( rect
!= wxGridNoCellRect
) Refresh( TRUE
, &rect
);
4180 wxGridRangeSelectEvent
gridEvt( GetId(),
4181 EVT_GRID_RANGE_SELECT
,
4184 m_selectedBottomRight
);
4186 GetEventHandler()->ProcessEvent(gridEvt
);
4190 void wxGrid::SelectBlock( int topRow
, int leftCol
, int bottomRow
, int rightCol
)
4194 if ( topRow
> bottomRow
)
4201 if ( leftCol
> rightCol
)
4208 m_selectedTopLeft
.Set( topRow
, leftCol
);
4209 m_selectedBottomRight
.Set( bottomRow
, rightCol
);
4211 if ( !GetBatchCount() )
4213 wxRect
rect( SelectionToRect() );
4214 if ( rect
!= wxGridNoCellRect
) Refresh( TRUE
, &rect
);
4217 // only generate an event if the block is not being selected by
4218 // dragging the mouse (in which case the event will be generated in
4220 if ( !m_isDragging
)
4222 wxGridRangeSelectEvent
gridEvt( GetId(),
4223 EVT_GRID_RANGE_SELECT
,
4226 m_selectedBottomRight
);
4228 GetEventHandler()->ProcessEvent(gridEvt
);
4232 void wxGrid::SelectAll()
4234 m_selectedTopLeft
.Set( 0, 0 );
4235 m_selectedBottomRight
.Set( m_numRows
-1, m_numCols
-1 );
4237 if ( !GetBatchCount() ) Refresh();
4241 void wxGrid::ClearSelection()
4243 if ( IsSelection() )
4245 wxRect
rect( SelectionToRect() );
4246 if ( rect
!= wxGridNoCellRect
)
4248 Refresh( TRUE
, &rect
);
4251 m_selectedTopLeft
= wxGridNoCellCoords
;
4252 m_selectedBottomRight
= wxGridNoCellCoords
;
4257 wxRect
wxGrid::SelectionToRect()
4262 if ( IsSelection() )
4264 cellRect
= CellToRect( m_selectedTopLeft
);
4265 if ( cellRect
!= wxGridNoCellRect
)
4271 rect
= wxRect( m_left
, m_top
, 0, 0 );
4274 cellRect
= CellToRect( m_selectedBottomRight
);
4275 if ( cellRect
!= wxGridNoCellRect
)
4281 return wxGridNoCellRect
;
4292 // ------ Grid event classes
4295 IMPLEMENT_DYNAMIC_CLASS( wxGridEvent
, wxEvent
)
4297 wxGridEvent::wxGridEvent( int id
, wxEventType type
, wxObject
* obj
,
4298 int row
, int col
, int x
, int y
,
4299 bool control
, bool shift
, bool alt
, bool meta
)
4300 : wxNotifyEvent( type
, id
)
4306 m_control
= control
;
4311 SetEventObject(obj
);
4315 IMPLEMENT_DYNAMIC_CLASS( wxGridSizeEvent
, wxEvent
)
4317 wxGridSizeEvent::wxGridSizeEvent( int id
, wxEventType type
, wxObject
* obj
,
4318 int rowOrCol
, int x
, int y
,
4319 bool control
, bool shift
, bool alt
, bool meta
)
4320 : wxNotifyEvent( type
, id
)
4322 m_rowOrCol
= rowOrCol
;
4325 m_control
= control
;
4330 SetEventObject(obj
);
4334 IMPLEMENT_DYNAMIC_CLASS( wxGridRangeSelectEvent
, wxEvent
)
4336 wxGridRangeSelectEvent::wxGridRangeSelectEvent(int id
, wxEventType type
, wxObject
* obj
,
4337 const wxGridCellCoords
& topLeft
,
4338 const wxGridCellCoords
& bottomRight
,
4339 bool control
, bool shift
, bool alt
, bool meta
)
4340 : wxNotifyEvent( type
, id
)
4342 m_topLeft
= topLeft
;
4343 m_bottomRight
= bottomRight
;
4344 m_control
= control
;
4349 SetEventObject(obj
);
4353 #endif // ifndef wxUSE_NEW_GRID