]> git.saurik.com Git - wxWidgets.git/blob - src/generic/grid.cpp
a9232d42ed3746bde9a979470cb2740dddf4d244
[wxWidgets.git] / src / generic / grid.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: grid.cpp
3 // Purpose: wxGrid and related classes
4 // Author: Michael Bedward (based on code by Julian Smart, Robin Dunn)
5 // Modified by:
6 // Created: 1/08/1999
7 // RCS-ID: $Id$
8 // Copyright: (c) Michael Bedward (mbedward@ozemail.com.au)
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
11
12 #ifdef __GNUG__
13 #pragma implementation "grid.h"
14 #endif
15
16 // For compilers that support precompilation, includes "wx/wx.h".
17 #include "wx/wxprec.h"
18
19 #include "wx/defs.h"
20
21 #ifdef __BORLANDC__
22 #pragma hdrstop
23 #endif
24
25 #if !defined(wxUSE_NEW_GRID) || !(wxUSE_NEW_GRID)
26 #include "gridg.cpp"
27 #else
28
29 #ifndef WX_PRECOMP
30 #include "wx/utils.h"
31 #include "wx/dcclient.h"
32 #include "wx/settings.h"
33 #include "wx/log.h"
34 #include "wx/sizer.h"
35 #endif
36
37 #include "wx/generic/grid.h"
38
39 //////////////////////////////////////////////////////////////////////
40
41 wxGridCellCoords wxGridNoCellCoords( -1, -1 );
42 wxRect wxGridNoCellRect( -1, -1, -1, -1 );
43
44 // this is a magic incantation which must be done!
45 #include "wx/arrimpl.cpp"
46
47 WX_DEFINE_OBJARRAY(wxGridCellCoordsArray)
48
49 // scroll line size
50 // TODO: fixed so far - make configurable later (and also different for x/y)
51 static const size_t GRID_SCROLL_LINE = 10;
52
53 //////////////////////////////////////////////////////////////////////
54 //
55 // Abstract base class for grid data (the model)
56 //
57 IMPLEMENT_ABSTRACT_CLASS( wxGridTableBase, wxObject )
58
59
60 wxGridTableBase::wxGridTableBase()
61 : wxObject()
62 {
63 m_view = (wxGrid *) NULL;
64 }
65
66 wxGridTableBase::~wxGridTableBase()
67 {
68 }
69
70
71 bool wxGridTableBase::InsertRows( size_t pos, size_t numRows )
72 {
73 wxLogWarning( wxT("Called grid table class function InsertRows(pos=%d, N=%d)\n"
74 "but your derived table class does not override this function"),
75 pos, numRows );
76
77 return FALSE;
78 }
79
80 bool wxGridTableBase::AppendRows( size_t numRows )
81 {
82 wxLogWarning( wxT("Called grid table class function AppendRows(N=%d)\n"
83 "but your derived table class does not override this function"),
84 numRows );
85
86 return FALSE;
87 }
88
89 bool wxGridTableBase::DeleteRows( size_t pos, size_t numRows )
90 {
91 wxLogWarning( wxT("Called grid table class function DeleteRows(pos=%d, N=%d)\n"
92 "but your derived table class does not override this function"),
93 pos, numRows );
94
95 return FALSE;
96 }
97
98 bool wxGridTableBase::InsertCols( size_t pos, size_t numCols )
99 {
100 wxLogWarning( wxT("Called grid table class function InsertCols(pos=%d, N=%d)\n"
101 "but your derived table class does not override this function"),
102 pos, numCols );
103
104 return FALSE;
105 }
106
107 bool wxGridTableBase::AppendCols( size_t numCols )
108 {
109 wxLogWarning( wxT("Called grid table class function AppendCols(N=%d)\n"
110 "but your derived table class does not override this function"),
111 numCols );
112
113 return FALSE;
114 }
115
116 bool wxGridTableBase::DeleteCols( size_t pos, size_t numCols )
117 {
118 wxLogWarning( wxT("Called grid table class function DeleteCols(pos=%d, N=%d)\n"
119 "but your derived table class does not override this function"),
120 pos, numCols );
121
122 return FALSE;
123 }
124
125
126 wxString wxGridTableBase::GetRowLabelValue( int row )
127 {
128 wxString s;
129 s << row;
130 return s;
131 }
132
133 wxString wxGridTableBase::GetColLabelValue( int col )
134 {
135 // default col labels are:
136 // cols 0 to 25 : A-Z
137 // cols 26 to 675 : AA-ZZ
138 // etc.
139
140 wxString s;
141 unsigned int i, n;
142 for ( n = 1; ; n++ )
143 {
144 s += (_T('A') + (wxChar)( col%26 ));
145 col = col/26 - 1;
146 if ( col < 0 ) break;
147 }
148
149 // reverse the string...
150 wxString s2;
151 for ( i = 0; i < n; i++ )
152 {
153 s2 += s[n-i-1];
154 }
155
156 return s2;
157 }
158
159
160
161 //////////////////////////////////////////////////////////////////////
162 //
163 // Message class for the grid table to send requests and notifications
164 // to the grid view
165 //
166
167 wxGridTableMessage::wxGridTableMessage()
168 {
169 m_table = (wxGridTableBase *) NULL;
170 m_id = -1;
171 m_comInt1 = -1;
172 m_comInt2 = -1;
173 }
174
175 wxGridTableMessage::wxGridTableMessage( wxGridTableBase *table, int id,
176 int commandInt1, int commandInt2 )
177 {
178 m_table = table;
179 m_id = id;
180 m_comInt1 = commandInt1;
181 m_comInt2 = commandInt2;
182 }
183
184
185
186 //////////////////////////////////////////////////////////////////////
187 //
188 // A basic grid table for string data. An object of this class will
189 // created by wxGrid if you don't specify an alternative table class.
190 //
191
192 WX_DEFINE_OBJARRAY(wxGridStringArray)
193
194 IMPLEMENT_DYNAMIC_CLASS( wxGridStringTable, wxGridTableBase )
195
196 wxGridStringTable::wxGridStringTable()
197 : wxGridTableBase()
198 {
199 }
200
201 wxGridStringTable::wxGridStringTable( int numRows, int numCols )
202 : wxGridTableBase()
203 {
204 int row, col;
205
206 m_data.Alloc( numRows );
207
208 wxArrayString sa;
209 sa.Alloc( numCols );
210 for ( col = 0; col < numCols; col++ )
211 {
212 sa.Add( wxEmptyString );
213 }
214
215 for ( row = 0; row < numRows; row++ )
216 {
217 m_data.Add( sa );
218 }
219 }
220
221 wxGridStringTable::~wxGridStringTable()
222 {
223 }
224
225 long wxGridStringTable::GetNumberRows()
226 {
227 return m_data.GetCount();
228 }
229
230 long wxGridStringTable::GetNumberCols()
231 {
232 if ( m_data.GetCount() > 0 )
233 return m_data[0].GetCount();
234 else
235 return 0;
236 }
237
238 wxString wxGridStringTable::GetValue( int row, int col )
239 {
240 // TODO: bounds checking
241 //
242 return m_data[row][col];
243 }
244
245 void wxGridStringTable::SetValue( int row, int col, const wxString& s )
246 {
247 // TODO: bounds checking
248 //
249 m_data[row][col] = s;
250 }
251
252 bool wxGridStringTable::IsEmptyCell( int row, int col )
253 {
254 // TODO: bounds checking
255 //
256 return (m_data[row][col] == wxEmptyString);
257 }
258
259
260 void wxGridStringTable::Clear()
261 {
262 int row, col;
263 int numRows, numCols;
264
265 numRows = m_data.GetCount();
266 if ( numRows > 0 )
267 {
268 numCols = m_data[0].GetCount();
269
270 for ( row = 0; row < numRows; row++ )
271 {
272 for ( col = 0; col < numCols; col++ )
273 {
274 m_data[row][col] = wxEmptyString;
275 }
276 }
277 }
278 }
279
280
281 bool wxGridStringTable::InsertRows( size_t pos, size_t numRows )
282 {
283 size_t row, col;
284
285 size_t curNumRows = m_data.GetCount();
286 size_t curNumCols = ( curNumRows > 0 ? m_data[0].GetCount() : 0 );
287
288 if ( pos >= curNumRows )
289 {
290 return AppendRows( numRows );
291 }
292
293 wxArrayString sa;
294 sa.Alloc( curNumCols );
295 for ( col = 0; col < curNumCols; col++ )
296 {
297 sa.Add( wxEmptyString );
298 }
299
300 for ( row = pos; row < pos + numRows; row++ )
301 {
302 m_data.Insert( sa, row );
303 }
304
305 if ( GetView() )
306 {
307 wxGridTableMessage msg( this,
308 wxGRIDTABLE_NOTIFY_ROWS_INSERTED,
309 pos,
310 numRows );
311
312 GetView()->ProcessTableMessage( msg );
313 }
314
315 return TRUE;
316 }
317
318 bool wxGridStringTable::AppendRows( size_t numRows )
319 {
320 size_t row, col;
321
322 size_t curNumRows = m_data.GetCount();
323 size_t curNumCols = ( curNumRows > 0 ? m_data[0].GetCount() : 0 );
324
325 wxArrayString sa;
326 if ( curNumCols > 0 )
327 {
328 sa.Alloc( curNumCols );
329 for ( col = 0; col < curNumCols; col++ )
330 {
331 sa.Add( wxEmptyString );
332 }
333 }
334
335 for ( row = 0; row < numRows; row++ )
336 {
337 m_data.Add( sa );
338 }
339
340 if ( GetView() )
341 {
342 wxGridTableMessage msg( this,
343 wxGRIDTABLE_NOTIFY_ROWS_APPENDED,
344 numRows );
345
346 GetView()->ProcessTableMessage( msg );
347 }
348
349 return TRUE;
350 }
351
352 bool wxGridStringTable::DeleteRows( size_t pos, size_t numRows )
353 {
354 size_t n;
355
356 size_t curNumRows = m_data.GetCount();
357
358 if ( pos >= curNumRows )
359 {
360 wxLogError( wxT("Called wxGridStringTable::DeleteRows(pos=%d, N=%d)...\n"
361 "Pos value is invalid for present table with %d rows"),
362 pos, numRows, curNumRows );
363 return FALSE;
364 }
365
366 if ( numRows > curNumRows - pos )
367 {
368 numRows = curNumRows - pos;
369 }
370
371 if ( numRows >= curNumRows )
372 {
373 m_data.Empty(); // don't release memory just yet
374 }
375 else
376 {
377 for ( n = 0; n < numRows; n++ )
378 {
379 m_data.Remove( pos );
380 }
381 }
382
383 if ( GetView() )
384 {
385 wxGridTableMessage msg( this,
386 wxGRIDTABLE_NOTIFY_ROWS_DELETED,
387 pos,
388 numRows );
389
390 GetView()->ProcessTableMessage( msg );
391 }
392
393 return TRUE;
394 }
395
396 bool wxGridStringTable::InsertCols( size_t pos, size_t numCols )
397 {
398 size_t row, col;
399
400 size_t curNumRows = m_data.GetCount();
401 size_t curNumCols = ( curNumRows > 0 ? m_data[0].GetCount() : 0 );
402
403 if ( pos >= curNumCols )
404 {
405 return AppendCols( numCols );
406 }
407
408 for ( row = 0; row < curNumRows; row++ )
409 {
410 for ( col = pos; col < pos + numCols; col++ )
411 {
412 m_data[row].Insert( wxEmptyString, col );
413 }
414 }
415
416 if ( GetView() )
417 {
418 wxGridTableMessage msg( this,
419 wxGRIDTABLE_NOTIFY_COLS_INSERTED,
420 pos,
421 numCols );
422
423 GetView()->ProcessTableMessage( msg );
424 }
425
426 return TRUE;
427 }
428
429 bool wxGridStringTable::AppendCols( size_t numCols )
430 {
431 size_t row, n;
432
433 size_t curNumRows = m_data.GetCount();
434 if ( !curNumRows )
435 {
436 // TODO: something better than this ?
437 //
438 wxLogError( wxT("Unable to append cols to a grid table with no rows.\n"
439 "Call AppendRows() first") );
440 return FALSE;
441 }
442
443 for ( row = 0; row < curNumRows; row++ )
444 {
445 for ( n = 0; n < numCols; n++ )
446 {
447 m_data[row].Add( wxEmptyString );
448 }
449 }
450
451 if ( GetView() )
452 {
453 wxGridTableMessage msg( this,
454 wxGRIDTABLE_NOTIFY_COLS_APPENDED,
455 numCols );
456
457 GetView()->ProcessTableMessage( msg );
458 }
459
460 return TRUE;
461 }
462
463 bool wxGridStringTable::DeleteCols( size_t pos, size_t numCols )
464 {
465 size_t row, n;
466
467 size_t curNumRows = m_data.GetCount();
468 size_t curNumCols = ( curNumRows > 0 ? m_data[0].GetCount() : 0 );
469
470 if ( pos >= curNumCols )
471 {
472 wxLogError( wxT("Called wxGridStringTable::DeleteCols(pos=%d, N=%d)...\n"
473 "Pos value is invalid for present table with %d cols"),
474 pos, numCols, curNumCols );
475 return FALSE;
476 }
477
478 if ( numCols > curNumCols - pos )
479 {
480 numCols = curNumCols - pos;
481 }
482
483 for ( row = 0; row < curNumRows; row++ )
484 {
485 if ( numCols >= curNumCols )
486 {
487 m_data[row].Clear();
488 }
489 else
490 {
491 for ( n = 0; n < numCols; n++ )
492 {
493 m_data[row].Remove( pos );
494 }
495 }
496 }
497
498 if ( GetView() )
499 {
500 wxGridTableMessage msg( this,
501 wxGRIDTABLE_NOTIFY_COLS_DELETED,
502 pos,
503 numCols );
504
505 GetView()->ProcessTableMessage( msg );
506 }
507
508 return TRUE;
509 }
510
511 wxString wxGridStringTable::GetRowLabelValue( int row )
512 {
513 if ( row > (int)(m_rowLabels.GetCount()) - 1 )
514 {
515 // using default label
516 //
517 return wxGridTableBase::GetRowLabelValue( row );
518 }
519 else
520 {
521 return m_rowLabels[ row ];
522 }
523 }
524
525 wxString wxGridStringTable::GetColLabelValue( int col )
526 {
527 if ( col > (int)(m_colLabels.GetCount()) - 1 )
528 {
529 // using default label
530 //
531 return wxGridTableBase::GetColLabelValue( col );
532 }
533 else
534 {
535 return m_colLabels[ col ];
536 }
537 }
538
539 void wxGridStringTable::SetRowLabelValue( int row, const wxString& value )
540 {
541 if ( row > (int)(m_rowLabels.GetCount()) - 1 )
542 {
543 int n = m_rowLabels.GetCount();
544 int i;
545 for ( i = n; i <= row; i++ )
546 {
547 m_rowLabels.Add( wxGridTableBase::GetRowLabelValue(i) );
548 }
549 }
550
551 m_rowLabels[row] = value;
552 }
553
554 void wxGridStringTable::SetColLabelValue( int col, const wxString& value )
555 {
556 if ( col > (int)(m_colLabels.GetCount()) - 1 )
557 {
558 int n = m_colLabels.GetCount();
559 int i;
560 for ( i = n; i <= col; i++ )
561 {
562 m_colLabels.Add( wxGridTableBase::GetColLabelValue(i) );
563 }
564 }
565
566 m_colLabels[col] = value;
567 }
568
569
570
571
572 //////////////////////////////////////////////////////////////////////
573
574 IMPLEMENT_DYNAMIC_CLASS( wxGridTextCtrl, wxTextCtrl )
575
576 BEGIN_EVENT_TABLE( wxGridTextCtrl, wxTextCtrl )
577 EVT_KEY_DOWN( wxGridTextCtrl::OnKeyDown )
578 END_EVENT_TABLE()
579
580
581 wxGridTextCtrl::wxGridTextCtrl( wxWindow *par,
582 wxGrid *grid,
583 bool isCellControl,
584 wxWindowID id,
585 const wxString& value,
586 const wxPoint& pos,
587 const wxSize& size,
588 long style )
589 : wxTextCtrl( par, id, value, pos, size, style )
590 {
591 m_grid = grid;
592 m_isCellControl = isCellControl;
593 }
594
595
596 void wxGridTextCtrl::OnKeyDown( wxKeyEvent& event )
597 {
598 switch ( event.KeyCode() )
599 {
600 case WXK_ESCAPE:
601 m_grid->SetEditControlValue( startValue );
602 SetInsertionPointEnd();
603 break;
604
605 case WXK_UP:
606 case WXK_DOWN:
607 case WXK_LEFT:
608 case WXK_RIGHT:
609 case WXK_PRIOR:
610 case WXK_NEXT:
611 if ( m_isCellControl )
612 {
613 // send the event to the parent grid, skipping the
614 // event if nothing happens
615 //
616 event.Skip( m_grid->ProcessEvent( event ) );
617 }
618 else
619 {
620 // default text control response within the top edit
621 // control
622 //
623 event.Skip();
624 }
625 break;
626
627 case WXK_RETURN:
628 if ( m_isCellControl )
629 {
630 if ( !m_grid->ProcessEvent( event ) )
631 {
632 #ifdef __WXMOTIF__
633 // wxMotif needs a little extra help...
634 //
635 int pos = GetInsertionPoint();
636 wxString s( GetValue() );
637 s = s.Left(pos) + "\n" + s.Mid(pos);
638 SetValue(s);
639 SetInsertionPoint( pos );
640 #else
641 // the other ports can handle a Return key press
642 //
643 event.Skip();
644 #endif
645 }
646 }
647 break;
648
649 case WXK_HOME:
650 case WXK_END:
651 if ( m_isCellControl )
652 {
653 // send the event to the parent grid, skipping the
654 // event if nothing happens
655 //
656 event.Skip( m_grid->ProcessEvent( event ) );
657 }
658 else
659 {
660 // default text control response within the top edit
661 // control
662 //
663 event.Skip();
664 }
665 break;
666
667 default:
668 event.Skip();
669 }
670 }
671
672 void wxGridTextCtrl::SetStartValue( const wxString& s )
673 {
674 startValue = s;
675 wxTextCtrl::SetValue(s);
676 }
677
678
679
680 //////////////////////////////////////////////////////////////////////
681
682 IMPLEMENT_DYNAMIC_CLASS( wxGridRowLabelWindow, wxWindow )
683
684 BEGIN_EVENT_TABLE( wxGridRowLabelWindow, wxWindow )
685 EVT_PAINT( wxGridRowLabelWindow::OnPaint )
686 EVT_MOUSE_EVENTS( wxGridRowLabelWindow::OnMouseEvent )
687 EVT_KEY_DOWN( wxGridRowLabelWindow::OnKeyDown )
688 END_EVENT_TABLE()
689
690 wxGridRowLabelWindow::wxGridRowLabelWindow( wxGrid *parent,
691 wxWindowID id,
692 const wxPoint &pos, const wxSize &size )
693 : wxWindow( parent, id, pos, size )
694 {
695 m_owner = parent;
696 }
697
698 void wxGridRowLabelWindow::OnPaint( wxPaintEvent &event )
699 {
700 wxPaintDC dc(this);
701
702 // NO - don't do this because it will set both the x and y origin
703 // coords to match the parent scrolled window and we just want to
704 // set the y coord - MB
705 //
706 // m_owner->PrepareDC( dc );
707
708 wxCoord x, y;
709 m_owner->CalcUnscrolledPosition( 0, 0, &x, &y );
710 dc.SetDeviceOrigin( 0, -y );
711
712 m_owner->CalcRowLabelsExposed( GetUpdateRegion() );
713 m_owner->DrawRowLabels( dc );
714 }
715
716
717 void wxGridRowLabelWindow::OnMouseEvent( wxMouseEvent& event )
718 {
719 m_owner->ProcessRowLabelMouseEvent( event );
720 }
721
722
723 // This seems to be required for wxMotif otherwise the mouse
724 // cursor must be in the cell edit control to get key events
725 //
726 void wxGridRowLabelWindow::OnKeyDown( wxKeyEvent& event )
727 {
728 if ( !m_owner->ProcessEvent( event ) ) event.Skip();
729 }
730
731
732
733 //////////////////////////////////////////////////////////////////////
734
735 IMPLEMENT_DYNAMIC_CLASS( wxGridColLabelWindow, wxWindow )
736
737 BEGIN_EVENT_TABLE( wxGridColLabelWindow, wxWindow )
738 EVT_PAINT( wxGridColLabelWindow::OnPaint )
739 EVT_MOUSE_EVENTS( wxGridColLabelWindow::OnMouseEvent )
740 EVT_KEY_DOWN( wxGridColLabelWindow::OnKeyDown )
741 END_EVENT_TABLE()
742
743 wxGridColLabelWindow::wxGridColLabelWindow( wxGrid *parent,
744 wxWindowID id,
745 const wxPoint &pos, const wxSize &size )
746 : wxWindow( parent, id, pos, size )
747 {
748 m_owner = parent;
749 }
750
751 void wxGridColLabelWindow::OnPaint( wxPaintEvent &event )
752 {
753 wxPaintDC dc(this);
754
755 // NO - don't do this because it will set both the x and y origin
756 // coords to match the parent scrolled window and we just want to
757 // set the x coord - MB
758 //
759 // m_owner->PrepareDC( dc );
760
761 wxCoord x, y;
762 m_owner->CalcUnscrolledPosition( 0, 0, &x, &y );
763 dc.SetDeviceOrigin( -x, 0 );
764
765 m_owner->CalcColLabelsExposed( GetUpdateRegion() );
766 m_owner->DrawColLabels( dc );
767 }
768
769
770 void wxGridColLabelWindow::OnMouseEvent( wxMouseEvent& event )
771 {
772 m_owner->ProcessColLabelMouseEvent( event );
773 }
774
775
776 // This seems to be required for wxMotif otherwise the mouse
777 // cursor must be in the cell edit control to get key events
778 //
779 void wxGridColLabelWindow::OnKeyDown( wxKeyEvent& event )
780 {
781 if ( !m_owner->ProcessEvent( event ) ) event.Skip();
782 }
783
784
785
786 //////////////////////////////////////////////////////////////////////
787
788 IMPLEMENT_DYNAMIC_CLASS( wxGridCornerLabelWindow, wxWindow )
789
790 BEGIN_EVENT_TABLE( wxGridCornerLabelWindow, wxWindow )
791 EVT_MOUSE_EVENTS( wxGridCornerLabelWindow::OnMouseEvent )
792 EVT_PAINT( wxGridCornerLabelWindow::OnPaint)
793 EVT_KEY_DOWN( wxGridCornerLabelWindow::OnKeyDown )
794 END_EVENT_TABLE()
795
796 wxGridCornerLabelWindow::wxGridCornerLabelWindow( wxGrid *parent,
797 wxWindowID id,
798 const wxPoint &pos, const wxSize &size )
799 : wxWindow( parent, id, pos, size )
800 {
801 m_owner = parent;
802 }
803
804 void wxGridCornerLabelWindow::OnPaint( wxPaintEvent& WXUNUSED(event) )
805 {
806 wxPaintDC dc(this);
807
808 int client_height = 0;
809 int client_width = 0;
810 GetClientSize( &client_width, &client_height );
811
812 dc.SetPen( *wxBLACK_PEN );
813 dc.DrawLine( client_width-1, client_height-1, client_width-1, 0 );
814 dc.DrawLine( client_width-1, client_height-1, 0, client_height-1 );
815
816 dc.SetPen( *wxWHITE_PEN );
817 dc.DrawLine( 0, 0, client_width, 0 );
818 dc.DrawLine( 0, 0, 0, client_height );
819 }
820
821
822 void wxGridCornerLabelWindow::OnMouseEvent( wxMouseEvent& event )
823 {
824 m_owner->ProcessCornerLabelMouseEvent( event );
825 }
826
827
828 // This seems to be required for wxMotif otherwise the mouse
829 // cursor must be in the cell edit control to get key events
830 //
831 void wxGridCornerLabelWindow::OnKeyDown( wxKeyEvent& event )
832 {
833 if ( !m_owner->ProcessEvent( event ) ) event.Skip();
834 }
835
836
837
838 //////////////////////////////////////////////////////////////////////
839
840 IMPLEMENT_DYNAMIC_CLASS( wxGridWindow, wxPanel )
841
842 BEGIN_EVENT_TABLE( wxGridWindow, wxPanel )
843 EVT_PAINT( wxGridWindow::OnPaint )
844 EVT_MOUSE_EVENTS( wxGridWindow::OnMouseEvent )
845 EVT_KEY_DOWN( wxGridWindow::OnKeyDown )
846 END_EVENT_TABLE()
847
848 wxGridWindow::wxGridWindow( wxGrid *parent,
849 wxGridRowLabelWindow *rowLblWin,
850 wxGridColLabelWindow *colLblWin,
851 wxWindowID id, const wxPoint &pos, const wxSize &size )
852 : wxPanel( parent, id, pos, size, wxSUNKEN_BORDER, "grid window" )
853 {
854 m_owner = parent;
855 m_rowLabelWin = rowLblWin;
856 m_colLabelWin = colLblWin;
857
858 SetBackgroundColour( "WHITE" );
859 }
860
861
862 wxGridWindow::~wxGridWindow()
863 {
864 }
865
866
867 void wxGridWindow::OnPaint( wxPaintEvent &WXUNUSED(event) )
868 {
869 wxPaintDC dc( this );
870 m_owner->PrepareDC( dc );
871
872 m_owner->CalcCellsExposed( GetUpdateRegion() );
873 m_owner->DrawGridCellArea( dc );
874 }
875
876
877 void wxGridWindow::ScrollWindow( int dx, int dy, const wxRect *rect )
878 {
879 wxPanel::ScrollWindow( dx, dy, rect );
880 m_rowLabelWin->ScrollWindow( 0, dy, rect );
881 m_colLabelWin->ScrollWindow( dx, 0, rect );
882 }
883
884
885 void wxGridWindow::OnMouseEvent( wxMouseEvent& event )
886 {
887 m_owner->ProcessGridCellMouseEvent( event );
888 }
889
890
891 // This seems to be required for wxMotif otherwise the mouse
892 // cursor must be in the cell edit control to get key events
893 //
894 void wxGridWindow::OnKeyDown( wxKeyEvent& event )
895 {
896 if ( !m_owner->ProcessEvent( event ) ) event.Skip();
897 }
898
899
900
901 //////////////////////////////////////////////////////////////////////
902
903 IMPLEMENT_DYNAMIC_CLASS( wxGrid, wxScrolledWindow )
904
905 BEGIN_EVENT_TABLE( wxGrid, wxScrolledWindow )
906 EVT_PAINT( wxGrid::OnPaint )
907 EVT_SIZE( wxGrid::OnSize )
908 EVT_KEY_DOWN( wxGrid::OnKeyDown )
909 END_EVENT_TABLE()
910
911 wxGrid::wxGrid( wxWindow *parent,
912 wxWindowID id,
913 const wxPoint& pos,
914 const wxSize& size,
915 long style,
916 const wxString& name )
917 : wxScrolledWindow( parent, id, pos, size, style, name )
918 {
919 Create();
920 }
921
922
923 wxGrid::~wxGrid()
924 {
925 delete m_table;
926 }
927
928
929 //
930 // ----- internal init and update functions
931 //
932
933 void wxGrid::Create()
934 {
935 int colLblH = WXGRID_DEFAULT_COL_LABEL_HEIGHT;
936 int rowLblW = WXGRID_DEFAULT_ROW_LABEL_WIDTH;
937
938 m_rowLabelWin = new wxGridRowLabelWindow( this,
939 -1,
940 wxDefaultPosition,
941 wxSize(rowLblW,-1) );
942
943 m_colLabelWin = new wxGridColLabelWindow( this,
944 -1,
945 wxDefaultPosition,
946 wxSize(-1, colLblH ) );
947
948 m_cornerLabelWin = new wxGridCornerLabelWindow( this,
949 -1,
950 wxDefaultPosition,
951 wxSize(rowLblW, colLblH ) );
952
953 m_gridWin = new wxGridWindow( this,
954 m_rowLabelWin,
955 m_colLabelWin,
956 -1,
957 wxDefaultPosition,
958 wxDefaultSize );
959
960 SetTargetWindow( m_gridWin );
961
962 m_mainSizer = new wxBoxSizer( wxVERTICAL );
963
964 m_topSizer = new wxBoxSizer( wxHORIZONTAL );
965 m_topSizer->Add( m_cornerLabelWin, 0 );
966 m_topSizer->Add( m_colLabelWin, 1 );
967
968 m_mainSizer->Add( m_topSizer, 0, wxEXPAND );
969
970 m_middleSizer = new wxBoxSizer( wxHORIZONTAL );
971 m_middleSizer->Add( m_rowLabelWin, 0, wxEXPAND );
972 m_middleSizer->Add( m_gridWin, 1, wxEXPAND );
973
974 m_mainSizer->Add( m_middleSizer, 1, wxEXPAND );
975
976 SetAutoLayout( TRUE );
977 SetSizer( m_mainSizer );
978 }
979
980
981 bool wxGrid::CreateGrid( int numRows, int numCols )
982 {
983 if ( m_created )
984 {
985 wxLogError( wxT("wxGrid::CreateGrid(numRows, numCols) called more than once") );
986 return FALSE;
987 }
988 else
989 {
990 m_numRows = numRows;
991 m_numCols = numCols;
992
993 m_table = new wxGridStringTable( m_numRows, m_numCols );
994 m_table->SetView( this );
995 Init();
996 m_created = TRUE;
997 }
998
999 return m_created;
1000 }
1001
1002
1003 void wxGrid::Init()
1004 {
1005 int i;
1006
1007 if ( m_numRows <= 0 )
1008 m_numRows = WXGRID_DEFAULT_NUMBER_ROWS;
1009
1010 if ( m_numCols <= 0 )
1011 m_numCols = WXGRID_DEFAULT_NUMBER_COLS;
1012
1013 m_rowLabelWidth = WXGRID_DEFAULT_ROW_LABEL_WIDTH;
1014 m_colLabelHeight = WXGRID_DEFAULT_COL_LABEL_HEIGHT;
1015
1016 if ( m_rowLabelWin )
1017 {
1018 m_labelBackgroundColour = m_rowLabelWin->GetBackgroundColour();
1019 }
1020 else
1021 {
1022 m_labelBackgroundColour = wxColour( _T("WHITE") );
1023 }
1024
1025 m_labelTextColour = wxColour( _T("BLACK") );
1026
1027 // TODO: something better than this ?
1028 //
1029 m_labelFont = this->GetFont();
1030 m_labelFont.SetWeight( m_labelFont.GetWeight() + 2 );
1031
1032 m_rowLabelHorizAlign = wxLEFT;
1033 m_rowLabelVertAlign = wxCENTRE;
1034
1035 m_colLabelHorizAlign = wxCENTRE;
1036 m_colLabelVertAlign = wxTOP;
1037
1038 m_defaultColWidth = WXGRID_DEFAULT_COL_WIDTH;
1039 m_defaultRowHeight = m_gridWin->GetCharHeight();
1040
1041 #if defined(__WXMOTIF__) || defined(__WXGTK__) // see also text ctrl sizing in ShowCellEditControl()
1042 m_defaultRowHeight += 8;
1043 #else
1044 m_defaultRowHeight += 4;
1045 #endif
1046
1047 m_rowHeights.Alloc( m_numRows );
1048 m_rowBottoms.Alloc( m_numRows );
1049 int rowBottom = 0;
1050 for ( i = 0; i < m_numRows; i++ )
1051 {
1052 m_rowHeights.Add( m_defaultRowHeight );
1053 rowBottom += m_defaultRowHeight;
1054 m_rowBottoms.Add( rowBottom );
1055 }
1056
1057 m_colWidths.Alloc( m_numCols );
1058 m_colRights.Alloc( m_numCols );
1059 int colRight = 0;
1060 for ( i = 0; i < m_numCols; i++ )
1061 {
1062 m_colWidths.Add( m_defaultColWidth );
1063 colRight += m_defaultColWidth;
1064 m_colRights.Add( colRight );
1065 }
1066
1067 // TODO: improve this ?
1068 //
1069 m_defaultCellFont = this->GetFont();
1070
1071 m_gridLineColour = wxColour( 128, 128, 255 );
1072 m_gridLinesEnabled = TRUE;
1073
1074 m_cursorMode = WXGRID_CURSOR_SELECT_CELL;
1075 m_dragLastPos = -1;
1076 m_dragRowOrCol = -1;
1077 m_isDragging = FALSE;
1078
1079 m_rowResizeCursor = wxCursor( wxCURSOR_SIZENS );
1080 m_colResizeCursor = wxCursor( wxCURSOR_SIZEWE );
1081
1082 m_currentCellCoords = wxGridNoCellCoords;
1083
1084 m_selectedTopLeft = wxGridNoCellCoords;
1085 m_selectedBottomRight = wxGridNoCellCoords;
1086
1087 m_editable = TRUE; // default for whole grid
1088
1089 m_inOnKeyDown = FALSE;
1090 m_batchCount = 0;
1091
1092 // TODO: extend this to other types of controls
1093 //
1094 m_cellEditCtrl = new wxGridTextCtrl( m_gridWin,
1095 this,
1096 TRUE,
1097 wxGRID_CELLCTRL,
1098 "",
1099 wxPoint(1,1),
1100 wxSize(1,1)
1101 #if defined(__WXMSW__)
1102 , wxTE_MULTILINE | wxTE_NO_VSCROLL
1103 #endif
1104 );
1105
1106 m_cellEditCtrl->Show( FALSE );
1107 m_cellEditCtrlEnabled = TRUE;
1108 m_editCtrlType = wxGRID_TEXTCTRL;
1109 }
1110
1111
1112 void wxGrid::CalcDimensions()
1113 {
1114 int cw, ch;
1115 GetClientSize( &cw, &ch );
1116
1117 if ( m_numRows > 0 && m_numCols > 0 )
1118 {
1119 int right = m_colRights[ m_numCols-1 ] + 20;
1120 int bottom = m_rowBottoms[ m_numRows-1 ] + 20;
1121
1122 // TODO: restore the scroll position that we had before sizing
1123 //
1124 int x, y;
1125 GetViewStart( &x, &y );
1126 SetScrollbars( GRID_SCROLL_LINE, GRID_SCROLL_LINE,
1127 right/GRID_SCROLL_LINE, bottom/GRID_SCROLL_LINE,
1128 x, y );
1129 }
1130 }
1131
1132
1133 // this is called when the grid table sends a message to say that it
1134 // has been redimensioned
1135 //
1136 bool wxGrid::Redimension( wxGridTableMessage& msg )
1137 {
1138 int i;
1139
1140 switch ( msg.GetId() )
1141 {
1142 case wxGRIDTABLE_NOTIFY_ROWS_INSERTED:
1143 {
1144 size_t pos = msg.GetCommandInt();
1145 int numRows = msg.GetCommandInt2();
1146 for ( i = 0; i < numRows; i++ )
1147 {
1148 m_rowHeights.Insert( m_defaultRowHeight, pos );
1149 m_rowBottoms.Insert( 0, pos );
1150 }
1151 m_numRows += numRows;
1152
1153 int bottom = 0;
1154 if ( pos > 0 ) bottom = m_rowBottoms[pos-1];
1155
1156 for ( i = pos; i < m_numRows; i++ )
1157 {
1158 bottom += m_rowHeights[i];
1159 m_rowBottoms[i] = bottom;
1160 }
1161 CalcDimensions();
1162 }
1163 return TRUE;
1164
1165 case wxGRIDTABLE_NOTIFY_ROWS_APPENDED:
1166 {
1167 int numRows = msg.GetCommandInt();
1168 for ( i = 0; i < numRows; i++ )
1169 {
1170 m_rowHeights.Add( m_defaultRowHeight );
1171 m_rowBottoms.Add( 0 );
1172 }
1173
1174 int oldNumRows = m_numRows;
1175 m_numRows += numRows;
1176
1177 int bottom = 0;
1178 if ( oldNumRows > 0 ) bottom = m_rowBottoms[oldNumRows-1];
1179
1180 for ( i = oldNumRows; i < m_numRows; i++ )
1181 {
1182 bottom += m_rowHeights[i];
1183 m_rowBottoms[i] = bottom;
1184 }
1185 CalcDimensions();
1186 }
1187 return TRUE;
1188
1189 case wxGRIDTABLE_NOTIFY_ROWS_DELETED:
1190 {
1191 size_t pos = msg.GetCommandInt();
1192 int numRows = msg.GetCommandInt2();
1193 for ( i = 0; i < numRows; i++ )
1194 {
1195 m_rowHeights.Remove( pos );
1196 m_rowBottoms.Remove( pos );
1197 }
1198 m_numRows -= numRows;
1199
1200 if ( !m_numRows )
1201 {
1202 m_numCols = 0;
1203 m_colWidths.Clear();
1204 m_colRights.Clear();
1205 m_currentCellCoords = wxGridNoCellCoords;
1206 }
1207 else
1208 {
1209 if ( m_currentCellCoords.GetRow() >= m_numRows )
1210 m_currentCellCoords.Set( 0, 0 );
1211
1212 int h = 0;
1213 for ( i = 0; i < m_numRows; i++ )
1214 {
1215 h += m_rowHeights[i];
1216 m_rowBottoms[i] = h;
1217 }
1218 }
1219
1220 CalcDimensions();
1221 }
1222 return TRUE;
1223
1224 case wxGRIDTABLE_NOTIFY_COLS_INSERTED:
1225 {
1226 size_t pos = msg.GetCommandInt();
1227 int numCols = msg.GetCommandInt2();
1228 for ( i = 0; i < numCols; i++ )
1229 {
1230 m_colWidths.Insert( m_defaultColWidth, pos );
1231 m_colRights.Insert( 0, pos );
1232 }
1233 m_numCols += numCols;
1234
1235 int right = 0;
1236 if ( pos > 0 ) right = m_colRights[pos-1];
1237
1238 for ( i = pos; i < m_numCols; i++ )
1239 {
1240 right += m_colWidths[i];
1241 m_colRights[i] = right;
1242 }
1243 CalcDimensions();
1244 }
1245 return TRUE;
1246
1247 case wxGRIDTABLE_NOTIFY_COLS_APPENDED:
1248 {
1249 int numCols = msg.GetCommandInt();
1250 for ( i = 0; i < numCols; i++ )
1251 {
1252 m_colWidths.Add( m_defaultColWidth );
1253 m_colRights.Add( 0 );
1254 }
1255
1256 int oldNumCols = m_numCols;
1257 m_numCols += numCols;
1258
1259 int right = 0;
1260 if ( oldNumCols > 0 ) right = m_colRights[oldNumCols-1];
1261
1262 for ( i = oldNumCols; i < m_numCols; i++ )
1263 {
1264 right += m_colWidths[i];
1265 m_colRights[i] = right;
1266 }
1267 CalcDimensions();
1268 }
1269 return TRUE;
1270
1271 case wxGRIDTABLE_NOTIFY_COLS_DELETED:
1272 {
1273 size_t pos = msg.GetCommandInt();
1274 int numCols = msg.GetCommandInt2();
1275 for ( i = 0; i < numCols; i++ )
1276 {
1277 m_colWidths.Remove( pos );
1278 m_colRights.Remove( pos );
1279 }
1280 m_numCols -= numCols;
1281
1282 if ( !m_numCols )
1283 {
1284 #if 0 // leave the row alone here so that AppendCols will work subsequently
1285 m_numRows = 0;
1286 m_rowHeights.Clear();
1287 m_rowBottoms.Clear();
1288 #endif
1289 m_currentCellCoords = wxGridNoCellCoords;
1290 }
1291 else
1292 {
1293 if ( m_currentCellCoords.GetCol() >= m_numCols )
1294 m_currentCellCoords.Set( 0, 0 );
1295
1296 int w = 0;
1297 for ( i = 0; i < m_numCols; i++ )
1298 {
1299 w += m_colWidths[i];
1300 m_colRights[i] = w;
1301 }
1302 }
1303 CalcDimensions();
1304 }
1305 return TRUE;
1306 }
1307
1308 return FALSE;
1309 }
1310
1311
1312 void wxGrid::CalcRowLabelsExposed( wxRegion& reg )
1313 {
1314 wxRegionIterator iter( reg );
1315 wxRect r;
1316
1317 m_rowLabelsExposed.Empty();
1318
1319 int top, bottom;
1320 while ( iter )
1321 {
1322 r = iter.GetRect();
1323
1324 // TODO: remove this when we can...
1325 // There is a bug in wxMotif that gives garbage update
1326 // rectangles if you jump-scroll a long way by clicking the
1327 // scrollbar with middle button. This is a work-around
1328 //
1329 #if defined(__WXMOTIF__)
1330 int cw, ch;
1331 m_gridWin->GetClientSize( &cw, &ch );
1332 if ( r.GetTop() > ch ) r.SetTop( 0 );
1333 r.SetBottom( wxMin( r.GetBottom(), ch ) );
1334 #endif
1335
1336 // logical bounds of update region
1337 //
1338 int dummy;
1339 CalcUnscrolledPosition( 0, r.GetTop(), &dummy, &top );
1340 CalcUnscrolledPosition( 0, r.GetBottom(), &dummy, &bottom );
1341
1342 // find the row labels within these bounds
1343 //
1344 int row;
1345 int rowTop;
1346 for ( row = 0; row < m_numRows; row++ )
1347 {
1348 if ( m_rowBottoms[row] < top ) continue;
1349
1350 rowTop = m_rowBottoms[row] - m_rowHeights[row];
1351 if ( rowTop > bottom ) break;
1352
1353 m_rowLabelsExposed.Add( row );
1354 }
1355
1356 iter++ ;
1357 }
1358 }
1359
1360
1361 void wxGrid::CalcColLabelsExposed( wxRegion& reg )
1362 {
1363 wxRegionIterator iter( reg );
1364 wxRect r;
1365
1366 m_colLabelsExposed.Empty();
1367
1368 int left, right;
1369 while ( iter )
1370 {
1371 r = iter.GetRect();
1372
1373 // TODO: remove this when we can...
1374 // There is a bug in wxMotif that gives garbage update
1375 // rectangles if you jump-scroll a long way by clicking the
1376 // scrollbar with middle button. This is a work-around
1377 //
1378 #if defined(__WXMOTIF__)
1379 int cw, ch;
1380 m_gridWin->GetClientSize( &cw, &ch );
1381 if ( r.GetLeft() > cw ) r.SetLeft( 0 );
1382 r.SetRight( wxMin( r.GetRight(), cw ) );
1383 #endif
1384
1385 // logical bounds of update region
1386 //
1387 int dummy;
1388 CalcUnscrolledPosition( r.GetLeft(), 0, &left, &dummy );
1389 CalcUnscrolledPosition( r.GetRight(), 0, &right, &dummy );
1390
1391 // find the cells within these bounds
1392 //
1393 int col;
1394 int colLeft;
1395 for ( col = 0; col < m_numCols; col++ )
1396 {
1397 if ( m_colRights[col] < left ) continue;
1398
1399 colLeft = m_colRights[col] - m_colWidths[col];
1400 if ( colLeft > right ) break;
1401
1402 m_colLabelsExposed.Add( col );
1403 }
1404
1405 iter++ ;
1406 }
1407 }
1408
1409
1410 void wxGrid::CalcCellsExposed( wxRegion& reg )
1411 {
1412 wxRegionIterator iter( reg );
1413 wxRect r;
1414
1415 m_cellsExposed.Empty();
1416 m_rowsExposed.Empty();
1417 m_colsExposed.Empty();
1418
1419 int left, top, right, bottom;
1420 while ( iter )
1421 {
1422 r = iter.GetRect();
1423
1424 // TODO: remove this when we can...
1425 // There is a bug in wxMotif that gives garbage update
1426 // rectangles if you jump-scroll a long way by clicking the
1427 // scrollbar with middle button. This is a work-around
1428 //
1429 #if defined(__WXMOTIF__)
1430 int cw, ch;
1431 m_gridWin->GetClientSize( &cw, &ch );
1432 if ( r.GetTop() > ch ) r.SetTop( 0 );
1433 if ( r.GetLeft() > cw ) r.SetLeft( 0 );
1434 r.SetRight( wxMin( r.GetRight(), cw ) );
1435 r.SetBottom( wxMin( r.GetBottom(), ch ) );
1436 #endif
1437
1438 // logical bounds of update region
1439 //
1440 CalcUnscrolledPosition( r.GetLeft(), r.GetTop(), &left, &top );
1441 CalcUnscrolledPosition( r.GetRight(), r.GetBottom(), &right, &bottom );
1442
1443 // find the cells within these bounds
1444 //
1445 int row, col;
1446 int colLeft, rowTop;
1447 for ( row = 0; row < m_numRows; row++ )
1448 {
1449 if ( m_rowBottoms[row] < top ) continue;
1450
1451 rowTop = m_rowBottoms[row] - m_rowHeights[row];
1452 if ( rowTop > bottom ) break;
1453
1454 m_rowsExposed.Add( row );
1455
1456 for ( col = 0; col < m_numCols; col++ )
1457 {
1458 if ( m_colRights[col] < left ) continue;
1459
1460 colLeft = m_colRights[col] - m_colWidths[col];
1461 if ( colLeft > right ) break;
1462
1463 if ( m_colsExposed.Index( col ) == wxNOT_FOUND ) m_colsExposed.Add( col );
1464 m_cellsExposed.Add( wxGridCellCoords( row, col ) );
1465 }
1466 }
1467
1468 iter++ ;
1469 }
1470 }
1471
1472
1473 void wxGrid::ProcessRowLabelMouseEvent( wxMouseEvent& event )
1474 {
1475 int x, y, row;
1476 wxPoint pos( event.GetPosition() );
1477 CalcUnscrolledPosition( pos.x, pos.y, &x, &y );
1478
1479 if ( event.Dragging() )
1480 {
1481 m_isDragging = TRUE;
1482
1483 if ( event.LeftIsDown() )
1484 {
1485 switch( m_cursorMode )
1486 {
1487 case WXGRID_CURSOR_RESIZE_ROW:
1488 {
1489 int cw, ch, left, dummy;
1490 m_gridWin->GetClientSize( &cw, &ch );
1491 CalcUnscrolledPosition( 0, 0, &left, &dummy );
1492
1493 wxClientDC dc( m_gridWin );
1494 PrepareDC( dc );
1495 dc.SetLogicalFunction(wxINVERT);
1496 if ( m_dragLastPos >= 0 )
1497 {
1498 dc.DrawLine( left, m_dragLastPos, left+cw, m_dragLastPos );
1499 }
1500 dc.DrawLine( left, y, left+cw, y );
1501 m_dragLastPos = y;
1502 }
1503 break;
1504
1505 case WXGRID_CURSOR_SELECT_ROW:
1506 {
1507 if ( (row = YToRow( y )) >= 0 &&
1508 !IsInSelection( row, 0 ) )
1509 {
1510 SelectRow( row, TRUE );
1511 }
1512 }
1513 break;
1514 }
1515 }
1516 return;
1517 }
1518
1519 m_isDragging = FALSE;
1520
1521
1522 // ------------ Left button pressed
1523 //
1524 if ( event.LeftDown() )
1525 {
1526 // don't send a label click event for a hit on the
1527 // edge of the row label - this is probably the user
1528 // wanting to resize the row
1529 //
1530 if ( YToEdgeOfRow(y) < 0 )
1531 {
1532 row = YToRow(y);
1533 if ( row >= 0 &&
1534 !SendEvent( EVT_GRID_LABEL_LEFT_CLICK, row, -1, event ) )
1535 {
1536 SelectRow( row, event.ShiftDown() );
1537 m_cursorMode = WXGRID_CURSOR_SELECT_ROW;
1538 }
1539 }
1540 else
1541 {
1542 // starting to drag-resize a row
1543 //
1544 m_rowLabelWin->CaptureMouse();
1545 }
1546 }
1547
1548
1549 // ------------ Left double click
1550 //
1551 else if (event.LeftDClick() )
1552 {
1553 if ( YToEdgeOfRow(y) < 0 )
1554 {
1555 row = YToRow(y);
1556 SendEvent( EVT_GRID_LABEL_LEFT_DCLICK, row, -1, event );
1557 }
1558 }
1559
1560
1561 // ------------ Left button released
1562 //
1563 else if ( event.LeftUp() )
1564 {
1565 if ( m_cursorMode == WXGRID_CURSOR_RESIZE_ROW )
1566 {
1567 m_rowLabelWin->ReleaseMouse();
1568
1569 if ( m_dragLastPos >= 0 )
1570 {
1571 // erase the last line and resize the row
1572 //
1573 int cw, ch, left, dummy;
1574 m_gridWin->GetClientSize( &cw, &ch );
1575 CalcUnscrolledPosition( 0, 0, &left, &dummy );
1576
1577 wxClientDC dc( m_gridWin );
1578 PrepareDC( dc );
1579 dc.SetLogicalFunction( wxINVERT );
1580 dc.DrawLine( left, m_dragLastPos, left+cw, m_dragLastPos );
1581 HideCellEditControl();
1582
1583 int rowTop = m_rowBottoms[m_dragRowOrCol] - m_rowHeights[m_dragRowOrCol];
1584 SetRowSize( m_dragRowOrCol, wxMax( y - rowTop, WXGRID_MIN_ROW_HEIGHT ) );
1585 if ( !GetBatchCount() )
1586 {
1587 // TODO: optimize this
1588 m_rowLabelWin->Refresh();
1589 m_gridWin->Refresh();
1590 }
1591
1592 ShowCellEditControl();
1593
1594 // Note: we are ending the event *after* doing
1595 // default processing in this case
1596 //
1597 SendEvent( EVT_GRID_ROW_SIZE, m_dragRowOrCol, -1, event );
1598 }
1599 }
1600
1601 m_dragLastPos = -1;
1602 }
1603
1604
1605 // ------------ Right button down
1606 //
1607 else if ( event.RightDown() )
1608 {
1609 row = YToRow(y);
1610 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_CLICK, row, -1, event ) )
1611 {
1612 // no default action at the moment
1613 }
1614 }
1615
1616
1617 // ------------ Right double click
1618 //
1619 else if ( event.RightDClick() )
1620 {
1621 row = YToRow(y);
1622 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_DCLICK, row, -1, event ) )
1623 {
1624 // no default action at the moment
1625 }
1626 }
1627
1628
1629 // ------------ No buttons down and mouse moving
1630 //
1631 else if ( event.Moving() )
1632 {
1633 m_dragRowOrCol = YToEdgeOfRow( y );
1634 if ( m_dragRowOrCol >= 0 )
1635 {
1636 if ( m_cursorMode == WXGRID_CURSOR_SELECT_CELL )
1637 {
1638 m_cursorMode = WXGRID_CURSOR_RESIZE_ROW;
1639 m_rowLabelWin->SetCursor( m_rowResizeCursor );
1640 }
1641 }
1642 else
1643 {
1644 if ( m_cursorMode != WXGRID_CURSOR_SELECT_CELL )
1645 {
1646 m_cursorMode = WXGRID_CURSOR_SELECT_CELL;
1647 m_rowLabelWin->SetCursor( *wxSTANDARD_CURSOR );
1648 }
1649 }
1650 }
1651 }
1652
1653
1654 void wxGrid::ProcessColLabelMouseEvent( wxMouseEvent& event )
1655 {
1656 int x, y, col;
1657 wxPoint pos( event.GetPosition() );
1658 CalcUnscrolledPosition( pos.x, pos.y, &x, &y );
1659
1660 if ( event.Dragging() )
1661 {
1662 m_isDragging = TRUE;
1663
1664 if ( event.LeftIsDown() )
1665 {
1666 switch( m_cursorMode )
1667 {
1668 case WXGRID_CURSOR_RESIZE_COL:
1669 {
1670 int cw, ch, dummy, top;
1671 m_gridWin->GetClientSize( &cw, &ch );
1672 CalcUnscrolledPosition( 0, 0, &dummy, &top );
1673
1674 wxClientDC dc( m_gridWin );
1675 PrepareDC( dc );
1676 dc.SetLogicalFunction(wxINVERT);
1677 if ( m_dragLastPos >= 0 )
1678 {
1679 dc.DrawLine( m_dragLastPos, top, m_dragLastPos, top+ch );
1680 }
1681 dc.DrawLine( x, top, x, top+ch );
1682 m_dragLastPos = x;
1683 }
1684 break;
1685
1686 case WXGRID_CURSOR_SELECT_COL:
1687 {
1688 if ( (col = XToCol( x )) >= 0 &&
1689 !IsInSelection( 0, col ) )
1690 {
1691 SelectCol( col, TRUE );
1692 }
1693 }
1694 break;
1695 }
1696 }
1697 return;
1698 }
1699
1700 m_isDragging = FALSE;
1701
1702
1703 // ------------ Left button pressed
1704 //
1705 if ( event.LeftDown() )
1706 {
1707 // don't send a label click event for a hit on the
1708 // edge of the col label - this is probably the user
1709 // wanting to resize the col
1710 //
1711 if ( XToEdgeOfCol(x) < 0 )
1712 {
1713 col = XToCol(x);
1714 if ( col >= 0 &&
1715 !SendEvent( EVT_GRID_LABEL_LEFT_CLICK, -1, col, event ) )
1716 {
1717 SelectCol( col, event.ShiftDown() );
1718 m_cursorMode = WXGRID_CURSOR_SELECT_COL;
1719 }
1720 }
1721 else
1722 {
1723 // starting to drag-resize a col
1724 //
1725 m_colLabelWin->CaptureMouse();
1726 }
1727 }
1728
1729
1730 // ------------ Left double click
1731 //
1732 if ( event.LeftDClick() )
1733 {
1734 if ( XToEdgeOfCol(x) < 0 )
1735 {
1736 col = XToCol(x);
1737 SendEvent( EVT_GRID_LABEL_LEFT_DCLICK, -1, col, event );
1738 }
1739 }
1740
1741
1742 // ------------ Left button released
1743 //
1744 else if ( event.LeftUp() )
1745 {
1746 if ( m_cursorMode == WXGRID_CURSOR_RESIZE_COL )
1747 {
1748 m_colLabelWin->ReleaseMouse();
1749
1750 if ( m_dragLastPos >= 0 )
1751 {
1752 // erase the last line and resize the col
1753 //
1754 int cw, ch, dummy, top;
1755 m_gridWin->GetClientSize( &cw, &ch );
1756 CalcUnscrolledPosition( 0, 0, &dummy, &top );
1757
1758 wxClientDC dc( m_gridWin );
1759 PrepareDC( dc );
1760 dc.SetLogicalFunction( wxINVERT );
1761 dc.DrawLine( m_dragLastPos, top, m_dragLastPos, top+ch );
1762 HideCellEditControl();
1763
1764 int colLeft = m_colRights[m_dragRowOrCol] - m_colWidths[m_dragRowOrCol];
1765 SetColSize( m_dragRowOrCol, wxMax( x - colLeft, WXGRID_MIN_COL_WIDTH ) );
1766
1767 if ( !GetBatchCount() )
1768 {
1769 // TODO: optimize this
1770 m_colLabelWin->Refresh();
1771 m_gridWin->Refresh();
1772 }
1773
1774 ShowCellEditControl();
1775
1776 // Note: we are ending the event *after* doing
1777 // default processing in this case
1778 //
1779 SendEvent( EVT_GRID_COL_SIZE, -1, m_dragRowOrCol, event );
1780 }
1781 }
1782
1783 m_dragLastPos = -1;
1784 }
1785
1786
1787 // ------------ Right button down
1788 //
1789 else if ( event.RightDown() )
1790 {
1791 col = XToCol(x);
1792 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_CLICK, -1, col, event ) )
1793 {
1794 // no default action at the moment
1795 }
1796 }
1797
1798
1799 // ------------ Right double click
1800 //
1801 else if ( event.RightDClick() )
1802 {
1803 col = XToCol(x);
1804 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_DCLICK, -1, col, event ) )
1805 {
1806 // no default action at the moment
1807 }
1808 }
1809
1810
1811 // ------------ No buttons down and mouse moving
1812 //
1813 else if ( event.Moving() )
1814 {
1815 m_dragRowOrCol = XToEdgeOfCol( x );
1816 if ( m_dragRowOrCol >= 0 )
1817 {
1818 if ( m_cursorMode == WXGRID_CURSOR_SELECT_CELL )
1819 {
1820 m_cursorMode = WXGRID_CURSOR_RESIZE_COL;
1821 m_colLabelWin->SetCursor( m_colResizeCursor );
1822 }
1823 }
1824 else
1825 {
1826 if ( m_cursorMode != WXGRID_CURSOR_SELECT_CELL )
1827 {
1828 m_cursorMode = WXGRID_CURSOR_SELECT_CELL;
1829 m_colLabelWin->SetCursor( *wxSTANDARD_CURSOR );
1830 }
1831 }
1832 }
1833 }
1834
1835
1836 void wxGrid::ProcessCornerLabelMouseEvent( wxMouseEvent& event )
1837 {
1838 if ( event.LeftDown() )
1839 {
1840 // indicate corner label by having both row and
1841 // col args == -1
1842 //
1843 if ( !SendEvent( EVT_GRID_LABEL_LEFT_CLICK, -1, -1, event ) )
1844 {
1845 SelectAll();
1846 }
1847 }
1848
1849 else if ( event.LeftDClick() )
1850 {
1851 SendEvent( EVT_GRID_LABEL_LEFT_DCLICK, -1, -1, event );
1852 }
1853
1854 else if ( event.RightDown() )
1855 {
1856 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_CLICK, -1, -1, event ) )
1857 {
1858 // no default action at the moment
1859 }
1860 }
1861
1862 else if ( event.RightDClick() )
1863 {
1864 if ( !SendEvent( EVT_GRID_LABEL_RIGHT_DCLICK, -1, -1, event ) )
1865 {
1866 // no default action at the moment
1867 }
1868 }
1869 }
1870
1871
1872 void wxGrid::ProcessGridCellMouseEvent( wxMouseEvent& event )
1873 {
1874 int x, y;
1875 wxPoint pos( event.GetPosition() );
1876 CalcUnscrolledPosition( pos.x, pos.y, &x, &y );
1877
1878 wxGridCellCoords coords;
1879 XYToCell( x, y, coords );
1880
1881 if ( event.Dragging() )
1882 {
1883 m_isDragging = TRUE;
1884 if ( m_cursorMode == WXGRID_CURSOR_SELECT_CELL )
1885 {
1886 // Hide the edit control, so it
1887 // won't interfer with drag-shrinking.
1888 if ( IsCellEditControlEnabled() )
1889 HideCellEditControl();
1890 if ( coords != wxGridNoCellCoords )
1891 {
1892 if ( !IsSelection() )
1893 {
1894 SelectBlock( coords, coords );
1895 }
1896 else
1897 {
1898 SelectBlock( m_currentCellCoords, coords );
1899 }
1900 }
1901 }
1902
1903 return;
1904 }
1905
1906 m_isDragging = FALSE;
1907
1908 if ( coords != wxGridNoCellCoords )
1909 {
1910 if ( event.LeftDown() )
1911 {
1912 if ( event.ShiftDown() )
1913 {
1914 SelectBlock( m_currentCellCoords, coords );
1915 }
1916 else
1917 {
1918 if ( !SendEvent( EVT_GRID_CELL_LEFT_CLICK,
1919 coords.GetRow(),
1920 coords.GetCol(),
1921 event ) )
1922 {
1923 MakeCellVisible( coords );
1924 SetCurrentCell( coords );
1925 }
1926 }
1927 }
1928
1929
1930 // ------------ Left double click
1931 //
1932 else if ( event.LeftDClick() )
1933 {
1934 SendEvent( EVT_GRID_CELL_LEFT_DCLICK,
1935 coords.GetRow(),
1936 coords.GetCol(),
1937 event );
1938 }
1939
1940
1941 // ------------ Left button released
1942 //
1943 else if ( event.LeftUp() )
1944 {
1945 if ( m_cursorMode == WXGRID_CURSOR_SELECT_CELL )
1946 {
1947 if ( IsSelection() )
1948 {
1949 SendEvent( EVT_GRID_RANGE_SELECT, -1, -1, event );
1950 }
1951 }
1952
1953 // Show the edit control, if it has
1954 // been hidden for drag-shrinking.
1955 if ( IsCellEditControlEnabled() )
1956 ShowCellEditControl();
1957
1958 m_dragLastPos = -1;
1959 }
1960
1961
1962 // ------------ Right button down
1963 //
1964 else if ( event.RightDown() )
1965 {
1966 if ( !SendEvent( EVT_GRID_CELL_RIGHT_CLICK,
1967 coords.GetRow(),
1968 coords.GetCol(),
1969 event ) )
1970 {
1971 // no default action at the moment
1972 }
1973 }
1974
1975
1976 // ------------ Right double click
1977 //
1978 else if ( event.RightDClick() )
1979 {
1980 if ( !SendEvent( EVT_GRID_CELL_RIGHT_DCLICK,
1981 coords.GetRow(),
1982 coords.GetCol(),
1983 event ) )
1984 {
1985 // no default action at the moment
1986 }
1987 }
1988 }
1989 }
1990
1991
1992 //
1993 // ------ interaction with data model
1994 //
1995 bool wxGrid::ProcessTableMessage( wxGridTableMessage& msg )
1996 {
1997 switch ( msg.GetId() )
1998 {
1999 case wxGRIDTABLE_REQUEST_VIEW_GET_VALUES:
2000 return GetModelValues();
2001
2002 case wxGRIDTABLE_REQUEST_VIEW_SEND_VALUES:
2003 return SetModelValues();
2004
2005 case wxGRIDTABLE_NOTIFY_ROWS_INSERTED:
2006 case wxGRIDTABLE_NOTIFY_ROWS_APPENDED:
2007 case wxGRIDTABLE_NOTIFY_ROWS_DELETED:
2008 case wxGRIDTABLE_NOTIFY_COLS_INSERTED:
2009 case wxGRIDTABLE_NOTIFY_COLS_APPENDED:
2010 case wxGRIDTABLE_NOTIFY_COLS_DELETED:
2011 return Redimension( msg );
2012
2013 default:
2014 return FALSE;
2015 }
2016 }
2017
2018
2019
2020 // The behaviour of this function depends on the grid table class
2021 // Clear() function. For the default wxGridStringTable class the
2022 // behavious is to replace all cell contents with wxEmptyString but
2023 // not to change the number of rows or cols.
2024 //
2025 void wxGrid::ClearGrid()
2026 {
2027 if ( m_table )
2028 {
2029 m_table->Clear();
2030 SetEditControlValue();
2031 if ( !GetBatchCount() ) m_gridWin->Refresh();
2032 }
2033 }
2034
2035
2036 bool wxGrid::InsertRows( int pos, int numRows, bool WXUNUSED(updateLabels) )
2037 {
2038 // TODO: something with updateLabels flag
2039
2040 if ( !m_created )
2041 {
2042 wxLogError( wxT("Called wxGrid::InsertRows() before calling CreateGrid()") );
2043 return FALSE;
2044 }
2045
2046 if ( m_table )
2047 {
2048 bool ok = m_table->InsertRows( pos, numRows );
2049
2050 // the table will have sent the results of the insert row
2051 // operation to this view object as a grid table message
2052 //
2053 if ( ok )
2054 {
2055 if ( m_numCols == 0 )
2056 {
2057 m_table->AppendCols( WXGRID_DEFAULT_NUMBER_COLS );
2058 //
2059 // TODO: perhaps instead of appending the default number of cols
2060 // we should remember what the last non-zero number of cols was ?
2061 //
2062 }
2063
2064 if ( m_currentCellCoords == wxGridNoCellCoords )
2065 {
2066 // if we have just inserted cols into an empty grid the current
2067 // cell will be undefined...
2068 //
2069 SetCurrentCell( 0, 0 );
2070 }
2071
2072 ClearSelection();
2073 if ( !GetBatchCount() ) Refresh();
2074 }
2075
2076 SetEditControlValue();
2077 return ok;
2078 }
2079 else
2080 {
2081 return FALSE;
2082 }
2083 }
2084
2085
2086 bool wxGrid::AppendRows( int numRows, bool WXUNUSED(updateLabels) )
2087 {
2088 // TODO: something with updateLabels flag
2089
2090 if ( !m_created )
2091 {
2092 wxLogError( wxT("Called wxGrid::AppendRows() before calling CreateGrid()") );
2093 return FALSE;
2094 }
2095
2096 if ( m_table && m_table->AppendRows( numRows ) )
2097 {
2098 if ( m_currentCellCoords == wxGridNoCellCoords )
2099 {
2100 // if we have just inserted cols into an empty grid the current
2101 // cell will be undefined...
2102 //
2103 SetCurrentCell( 0, 0 );
2104 }
2105
2106 // the table will have sent the results of the append row
2107 // operation to this view object as a grid table message
2108 //
2109 ClearSelection();
2110 if ( !GetBatchCount() ) Refresh();
2111 return TRUE;
2112 }
2113 else
2114 {
2115 return FALSE;
2116 }
2117 }
2118
2119
2120 bool wxGrid::DeleteRows( int pos, int numRows, bool WXUNUSED(updateLabels) )
2121 {
2122 // TODO: something with updateLabels flag
2123
2124 if ( !m_created )
2125 {
2126 wxLogError( wxT("Called wxGrid::DeleteRows() before calling CreateGrid()") );
2127 return FALSE;
2128 }
2129
2130 if ( m_table && m_table->DeleteRows( pos, numRows ) )
2131 {
2132 // the table will have sent the results of the delete row
2133 // operation to this view object as a grid table message
2134 //
2135 if ( m_numRows > 0 )
2136 SetEditControlValue();
2137 else
2138 HideCellEditControl();
2139
2140 ClearSelection();
2141 if ( !GetBatchCount() ) Refresh();
2142 return TRUE;
2143 }
2144 else
2145 {
2146 return FALSE;
2147 }
2148 }
2149
2150
2151 bool wxGrid::InsertCols( int pos, int numCols, bool WXUNUSED(updateLabels) )
2152 {
2153 // TODO: something with updateLabels flag
2154
2155 if ( !m_created )
2156 {
2157 wxLogError( wxT("Called wxGrid::InsertCols() before calling CreateGrid()") );
2158 return FALSE;
2159 }
2160
2161 if ( m_table )
2162 {
2163 HideCellEditControl();
2164 bool ok = m_table->InsertCols( pos, numCols );
2165
2166 // the table will have sent the results of the insert col
2167 // operation to this view object as a grid table message
2168 //
2169 if ( ok )
2170 {
2171 if ( m_currentCellCoords == wxGridNoCellCoords )
2172 {
2173 // if we have just inserted cols into an empty grid the current
2174 // cell will be undefined...
2175 //
2176 SetCurrentCell( 0, 0 );
2177 }
2178
2179 ClearSelection();
2180 if ( !GetBatchCount() ) Refresh();
2181 }
2182
2183 SetEditControlValue();
2184 return ok;
2185 }
2186 else
2187 {
2188 return FALSE;
2189 }
2190 }
2191
2192
2193 bool wxGrid::AppendCols( int numCols, bool WXUNUSED(updateLabels) )
2194 {
2195 // TODO: something with updateLabels flag
2196
2197 if ( !m_created )
2198 {
2199 wxLogError( wxT("Called wxGrid::AppendCols() before calling CreateGrid()") );
2200 return FALSE;
2201 }
2202
2203 if ( m_table && m_table->AppendCols( numCols ) )
2204 {
2205 // the table will have sent the results of the append col
2206 // operation to this view object as a grid table message
2207 //
2208 if ( m_currentCellCoords == wxGridNoCellCoords )
2209 {
2210 // if we have just inserted cols into an empty grid the current
2211 // cell will be undefined...
2212 //
2213 SetCurrentCell( 0, 0 );
2214 }
2215
2216 ClearSelection();
2217 if ( !GetBatchCount() ) Refresh();
2218 return TRUE;
2219 }
2220 else
2221 {
2222 return FALSE;
2223 }
2224 }
2225
2226
2227 bool wxGrid::DeleteCols( int pos, int numCols, bool WXUNUSED(updateLabels) )
2228 {
2229 // TODO: something with updateLabels flag
2230
2231 if ( !m_created )
2232 {
2233 wxLogError( wxT("Called wxGrid::DeleteCols() before calling CreateGrid()") );
2234 return FALSE;
2235 }
2236
2237 if ( m_table && m_table->DeleteCols( pos, numCols ) )
2238 {
2239 // the table will have sent the results of the delete col
2240 // operation to this view object as a grid table message
2241 //
2242 if ( m_numCols > 0 )
2243 SetEditControlValue();
2244 else
2245 HideCellEditControl();
2246
2247 ClearSelection();
2248 if ( !GetBatchCount() ) Refresh();
2249 return TRUE;
2250 }
2251 else
2252 {
2253 return FALSE;
2254 }
2255 }
2256
2257
2258
2259 //
2260 // ----- event handlers
2261 //
2262
2263 // Generate a grid event based on a mouse event and
2264 // return the result of ProcessEvent()
2265 //
2266 bool wxGrid::SendEvent( const wxEventType type,
2267 int row, int col,
2268 wxMouseEvent& mouseEv )
2269 {
2270 if ( type == EVT_GRID_ROW_SIZE ||
2271 type == EVT_GRID_COL_SIZE )
2272 {
2273 int rowOrCol = (row == -1 ? col : row);
2274
2275 wxGridSizeEvent gridEvt( GetId(),
2276 type,
2277 this,
2278 rowOrCol,
2279 mouseEv.GetX(), mouseEv.GetY(),
2280 mouseEv.ControlDown(),
2281 mouseEv.ShiftDown(),
2282 mouseEv.AltDown(),
2283 mouseEv.MetaDown() );
2284
2285 return GetEventHandler()->ProcessEvent(gridEvt);
2286 }
2287 else if ( type == EVT_GRID_RANGE_SELECT )
2288 {
2289 wxGridRangeSelectEvent gridEvt( GetId(),
2290 type,
2291 this,
2292 m_selectedTopLeft,
2293 m_selectedBottomRight,
2294 mouseEv.ControlDown(),
2295 mouseEv.ShiftDown(),
2296 mouseEv.AltDown(),
2297 mouseEv.MetaDown() );
2298
2299 return GetEventHandler()->ProcessEvent(gridEvt);
2300 }
2301 else
2302 {
2303 wxGridEvent gridEvt( GetId(),
2304 type,
2305 this,
2306 row, col,
2307 mouseEv.GetX(), mouseEv.GetY(),
2308 mouseEv.ControlDown(),
2309 mouseEv.ShiftDown(),
2310 mouseEv.AltDown(),
2311 mouseEv.MetaDown() );
2312
2313 return GetEventHandler()->ProcessEvent(gridEvt);
2314 }
2315 }
2316
2317
2318 // Generate a grid event of specified type and return the result
2319 // of ProcessEvent().
2320 //
2321 bool wxGrid::SendEvent( const wxEventType type,
2322 int row, int col )
2323 {
2324 if ( type == EVT_GRID_ROW_SIZE ||
2325 type == EVT_GRID_COL_SIZE )
2326 {
2327 int rowOrCol = (row == -1 ? col : row);
2328
2329 wxGridSizeEvent gridEvt( GetId(),
2330 type,
2331 this,
2332 rowOrCol );
2333
2334 return GetEventHandler()->ProcessEvent(gridEvt);
2335 }
2336 else
2337 {
2338 wxGridEvent gridEvt( GetId(),
2339 type,
2340 this,
2341 row, col );
2342
2343 return GetEventHandler()->ProcessEvent(gridEvt);
2344 }
2345 }
2346
2347
2348 void wxGrid::OnPaint( wxPaintEvent& WXUNUSED(event) )
2349 {
2350 wxPaintDC dc( this );
2351
2352 if ( m_currentCellCoords == wxGridNoCellCoords &&
2353 m_numRows && m_numCols )
2354 {
2355 m_currentCellCoords.Set(0, 0);
2356 SetEditControlValue();
2357 ShowCellEditControl();
2358 }
2359 }
2360
2361
2362 // This is just here to make sure that CalcDimensions gets called when
2363 // the grid view is resized... then the size event is skipped to allow
2364 // the box sizers to handle everything
2365 //
2366 void wxGrid::OnSize( wxSizeEvent& event )
2367 {
2368 CalcDimensions();
2369 event.Skip();
2370 }
2371
2372
2373 void wxGrid::OnKeyDown( wxKeyEvent& event )
2374 {
2375 if ( m_inOnKeyDown )
2376 {
2377 // shouldn't be here - we are going round in circles...
2378 //
2379 wxLogFatalError( wxT("wxGrid::OnKeyDown called while alread active") );
2380 }
2381
2382 m_inOnKeyDown = TRUE;
2383
2384 // propagate the event up and see if it gets processed
2385 //
2386 wxWindow *parent = GetParent();
2387 wxKeyEvent keyEvt( event );
2388 keyEvt.SetEventObject( parent );
2389
2390 if ( !parent->GetEventHandler()->ProcessEvent( keyEvt ) )
2391 {
2392 // try local handlers
2393 //
2394 switch ( event.KeyCode() )
2395 {
2396 case WXK_UP:
2397 if ( event.ControlDown() )
2398 {
2399 MoveCursorUpBlock();
2400 }
2401 else
2402 {
2403 MoveCursorUp();
2404 }
2405 break;
2406
2407 case WXK_DOWN:
2408 if ( event.ControlDown() )
2409 {
2410 MoveCursorDownBlock();
2411 }
2412 else
2413 {
2414 MoveCursorDown();
2415 }
2416 break;
2417
2418 case WXK_LEFT:
2419 if ( event.ControlDown() )
2420 {
2421 MoveCursorLeftBlock();
2422 }
2423 else
2424 {
2425 MoveCursorLeft();
2426 }
2427 break;
2428
2429 case WXK_RIGHT:
2430 if ( event.ControlDown() )
2431 {
2432 MoveCursorRightBlock();
2433 }
2434 else
2435 {
2436 MoveCursorRight();
2437 }
2438 break;
2439
2440 case WXK_RETURN:
2441 if ( event.ControlDown() )
2442 {
2443 event.Skip(); // to let the edit control have the return
2444 }
2445 else
2446 {
2447 MoveCursorDown();
2448 }
2449 break;
2450
2451 case WXK_HOME:
2452 if ( event.ControlDown() )
2453 {
2454 MakeCellVisible( 0, 0 );
2455 SetCurrentCell( 0, 0 );
2456 }
2457 else
2458 {
2459 event.Skip();
2460 }
2461 break;
2462
2463 case WXK_END:
2464 if ( event.ControlDown() )
2465 {
2466 MakeCellVisible( m_numRows-1, m_numCols-1 );
2467 SetCurrentCell( m_numRows-1, m_numCols-1 );
2468 }
2469 else
2470 {
2471 event.Skip();
2472 }
2473 break;
2474
2475 case WXK_PRIOR:
2476 MovePageUp();
2477 break;
2478
2479 case WXK_NEXT:
2480 MovePageDown();
2481 break;
2482
2483 default:
2484 // now try the cell edit control
2485 //
2486 if ( IsCellEditControlEnabled() )
2487 {
2488 event.SetEventObject( m_cellEditCtrl );
2489 m_cellEditCtrl->GetEventHandler()->ProcessEvent( event );
2490 }
2491 break;
2492 }
2493 }
2494
2495 m_inOnKeyDown = FALSE;
2496 }
2497
2498
2499 void wxGrid::SetCurrentCell( const wxGridCellCoords& coords )
2500 {
2501 if ( SendEvent( EVT_GRID_SELECT_CELL, coords.GetRow(), coords.GetCol() ) )
2502 {
2503 // the event has been intercepted - do nothing
2504 return;
2505 }
2506
2507 wxClientDC dc( m_gridWin );
2508 PrepareDC( dc );
2509
2510 if ( m_currentCellCoords != wxGridNoCellCoords )
2511 {
2512 HideCellEditControl();
2513 SaveEditControlValue();
2514 }
2515
2516 m_currentCellCoords = coords;
2517
2518 SetEditControlValue();
2519 ShowCellEditControl();
2520
2521 if ( IsSelection() )
2522 {
2523 wxRect r( SelectionToDeviceRect() );
2524 ClearSelection();
2525 if ( !GetBatchCount() ) m_gridWin->Refresh( TRUE, &r );
2526 }
2527 }
2528
2529
2530 //
2531 // ------ functions to get/send data (see also public functions)
2532 //
2533
2534 bool wxGrid::GetModelValues()
2535 {
2536 if ( m_table )
2537 {
2538 // all we need to do is repaint the grid
2539 //
2540 m_gridWin->Refresh();
2541 return TRUE;
2542 }
2543
2544 return FALSE;
2545 }
2546
2547
2548 bool wxGrid::SetModelValues()
2549 {
2550 int row, col;
2551
2552 if ( m_table )
2553 {
2554 for ( row = 0; row < m_numRows; row++ )
2555 {
2556 for ( col = 0; col < m_numCols; col++ )
2557 {
2558 m_table->SetValue( row, col, GetCellValue(row, col) );
2559 }
2560 }
2561
2562 return TRUE;
2563 }
2564
2565 return FALSE;
2566 }
2567
2568
2569
2570 // Note - this function only draws cells that are in the list of
2571 // exposed cells (usually set from the update region by
2572 // CalcExposedCells)
2573 //
2574 void wxGrid::DrawGridCellArea( wxDC& dc )
2575 {
2576 if ( !m_numRows || !m_numCols ) return;
2577
2578 size_t i;
2579 size_t numCells = m_cellsExposed.GetCount();
2580
2581 for ( i = 0; i < numCells; i++ )
2582 {
2583 DrawCell( dc, m_cellsExposed[i] );
2584 }
2585 }
2586
2587
2588 void wxGrid::DrawCell( wxDC& dc, const wxGridCellCoords& coords )
2589 {
2590 if ( m_colWidths[coords.GetCol()] <=0 ||
2591 m_rowHeights[coords.GetRow()] <= 0 ) return;
2592
2593 if ( m_gridLinesEnabled )
2594 DrawCellBorder( dc, coords );
2595
2596 DrawCellBackground( dc, coords );
2597
2598 // TODO: separate functions here for different kinds of cells ?
2599 // e.g. text, image
2600 //
2601 DrawCellValue( dc, coords );
2602 }
2603
2604
2605 void wxGrid::DrawCellBorder( wxDC& dc, const wxGridCellCoords& coords )
2606 {
2607 if ( m_colWidths[coords.GetCol()] <=0 ||
2608 m_rowHeights[coords.GetRow()] <= 0 ) return;
2609
2610 dc.SetPen( wxPen(GetGridLineColour(), 1, wxSOLID) );
2611 int row = coords.GetRow();
2612 int col = coords.GetCol();
2613
2614 // right hand border
2615 //
2616 dc.DrawLine( m_colRights[col], m_rowBottoms[row] - m_rowHeights[row],
2617 m_colRights[col], m_rowBottoms[row] );
2618
2619 // bottom border
2620 //
2621 dc.DrawLine( m_colRights[col] - m_colWidths[col], m_rowBottoms[row],
2622 m_colRights[col], m_rowBottoms[row] );
2623 }
2624
2625
2626 void wxGrid::DrawCellBackground( wxDC& dc, const wxGridCellCoords& coords )
2627 {
2628 if ( m_colWidths[coords.GetCol()] <=0 ||
2629 m_rowHeights[coords.GetRow()] <= 0 ) return;
2630
2631 int row = coords.GetRow();
2632 int col = coords.GetCol();
2633
2634 dc.SetBackgroundMode( wxSOLID );
2635
2636 if ( IsInSelection( coords ) )
2637 {
2638 // TODO: improve this
2639 //
2640 dc.SetBrush( *wxBLACK_BRUSH );
2641 }
2642 else
2643 {
2644 dc.SetBrush( wxBrush(GetCellBackgroundColour(row, col), wxSOLID) );
2645 }
2646
2647 dc.SetPen( *wxTRANSPARENT_PEN );
2648
2649 dc.DrawRectangle( m_colRights[col] - m_colWidths[col] + 1,
2650 m_rowBottoms[row] - m_rowHeights[row] + 1,
2651 m_colWidths[col]-1,
2652 m_rowHeights[row]-1 );
2653 }
2654
2655
2656 void wxGrid::DrawCellValue( wxDC& dc, const wxGridCellCoords& coords )
2657 {
2658 if ( m_colWidths[coords.GetCol()] <=0 ||
2659 m_rowHeights[coords.GetRow()] <= 0 ) return;
2660
2661 int row = coords.GetRow();
2662 int col = coords.GetCol();
2663
2664 dc.SetBackgroundMode( wxTRANSPARENT );
2665
2666 if ( IsInSelection( row, col ) )
2667 {
2668 // TODO: improve this
2669 //
2670 dc.SetTextBackground( wxColour(0, 0, 0) );
2671 dc.SetTextForeground( wxColour(255, 255, 255) );
2672 }
2673 else
2674 {
2675 dc.SetTextBackground( GetCellBackgroundColour(row, col) );
2676 dc.SetTextForeground( GetCellTextColour(row, col) );
2677 }
2678 dc.SetFont( GetCellFont(row, col) );
2679
2680 int hAlign, vAlign;
2681 GetCellAlignment( row, col, &hAlign, &vAlign );
2682
2683 wxRect rect;
2684 rect.SetX( m_colRights[col] - m_colWidths[col] + 2 );
2685 rect.SetY( m_rowBottoms[row] - m_rowHeights[row] + 2 );
2686 rect.SetWidth( m_colWidths[col] - 4 );
2687 rect.SetHeight( m_rowHeights[row] - 4 );
2688
2689 DrawTextRectangle( dc, GetCellValue( row, col ), rect, hAlign, vAlign );
2690 }
2691
2692
2693
2694 // TODO: remove this ???
2695 // This is used to redraw all grid lines e.g. when the grid line colour
2696 // has been changed
2697 //
2698 void wxGrid::DrawAllGridLines( wxDC& dc )
2699 {
2700 if ( !m_gridLinesEnabled ||
2701 !m_numRows ||
2702 !m_numCols ) return;
2703
2704 int cw, ch;
2705 m_gridWin->GetClientSize(&cw, &ch);
2706
2707 // virtual coords of visible area
2708 //
2709 int top, bottom, left, right;
2710 CalcUnscrolledPosition( 0, 0, &left, &top );
2711 CalcUnscrolledPosition( cw, ch, &right, &bottom );
2712
2713 dc.SetPen( wxPen(GetGridLineColour(), 1, wxSOLID) );
2714
2715 // horizontal grid lines
2716 //
2717 int i;
2718 for ( i = 0; i <= m_numRows; i++ )
2719 {
2720 if ( m_rowBottoms[i] > bottom )
2721 {
2722 break;
2723 }
2724 else if ( m_rowBottoms[i] >= top )
2725 {
2726 dc.DrawLine( left, m_rowBottoms[i], right, m_rowBottoms[i] );
2727 }
2728 }
2729
2730
2731 // vertical grid lines
2732 //
2733 for ( i = 0; i <= m_numCols; i++ )
2734 {
2735 if ( m_colRights[i] > right )
2736 {
2737 break;
2738 }
2739 else if ( m_colRights[i] >= left )
2740 {
2741 dc.DrawLine( m_colRights[i], top, m_colRights[i], bottom );
2742 }
2743 }
2744 }
2745
2746
2747 void wxGrid::DrawRowLabels( wxDC& dc )
2748 {
2749 if ( !m_numRows || !m_numCols ) return;
2750
2751 size_t i;
2752 size_t numLabels = m_rowLabelsExposed.GetCount();
2753
2754 for ( i = 0; i < numLabels; i++ )
2755 {
2756 DrawRowLabel( dc, m_rowLabelsExposed[i] );
2757 }
2758 }
2759
2760
2761 void wxGrid::DrawRowLabel( wxDC& dc, int row )
2762 {
2763 if ( m_rowHeights[row] <= 0 ) return;
2764
2765 // draw the label's horizontal border (the vertical border is
2766 // provided by the cell area window margin)
2767 //
2768 dc.SetPen( *wxBLACK_PEN );
2769
2770 dc.DrawLine( 0, m_rowBottoms[row]+1,
2771 m_rowLabelWidth, m_rowBottoms[row]+1 );
2772
2773 dc.SetPen( *wxWHITE_PEN );
2774
2775 dc.DrawLine( 0, m_rowBottoms[row]+2,
2776 m_rowLabelWidth, m_rowBottoms[row]+2 );
2777
2778 dc.SetBackgroundMode( wxTRANSPARENT );
2779 dc.SetTextForeground( GetLabelTextColour() );
2780 dc.SetFont( GetLabelFont() );
2781
2782 int hAlign, vAlign;
2783 GetRowLabelAlignment( &hAlign, &vAlign );
2784
2785 wxRect rect;
2786 rect.SetX( 2 );
2787 rect.SetY( m_rowBottoms[row] - m_rowHeights[row] + 2 );
2788 rect.SetWidth( m_rowLabelWidth - 4 );
2789 rect.SetHeight( m_rowHeights[row] - 4 );
2790 DrawTextRectangle( dc, GetRowLabelValue( row ), rect, hAlign, vAlign );
2791 }
2792
2793
2794 void wxGrid::DrawColLabels( wxDC& dc )
2795 {
2796 if ( !m_numRows || !m_numCols ) return;
2797
2798 size_t i;
2799 size_t numLabels = m_colLabelsExposed.GetCount();
2800
2801 for ( i = 0; i < numLabels; i++ )
2802 {
2803 DrawColLabel( dc, m_colLabelsExposed[i] );
2804 }
2805 }
2806
2807
2808 void wxGrid::DrawColLabel( wxDC& dc, int col )
2809 {
2810 if ( m_colWidths[col] <= 0 ) return;
2811
2812 // draw the label's vertical border (the horizontal border is
2813 // provided by the cell area window margin)
2814 //
2815 dc.SetPen( *wxBLACK_PEN );
2816
2817 dc.DrawLine( m_colRights[col]+1, 0,
2818 m_colRights[col]+1, m_colLabelHeight );
2819
2820 dc.SetPen( *wxWHITE_PEN );
2821
2822 dc.DrawLine( m_colRights[col]+2, 0,
2823 m_colRights[col]+2, m_colLabelHeight );
2824
2825 dc.SetBackgroundMode( wxTRANSPARENT );
2826 dc.SetTextForeground( GetLabelTextColour() );
2827 dc.SetFont( GetLabelFont() );
2828
2829 int hAlign, vAlign;
2830 GetColLabelAlignment( &hAlign, &vAlign );
2831
2832 wxRect rect;
2833 rect.SetX( m_colRights[col] - m_colWidths[col] + 2 );
2834 rect.SetY( 2 );
2835 rect.SetWidth( m_colWidths[col] - 4 );
2836 rect.SetHeight( m_colLabelHeight - 4 );
2837 DrawTextRectangle( dc, GetColLabelValue( col ), rect, hAlign, vAlign );
2838 }
2839
2840
2841 void wxGrid::DrawTextRectangle( wxDC& dc,
2842 const wxString& value,
2843 const wxRect& rect,
2844 int horizAlign,
2845 int vertAlign )
2846 {
2847 long textWidth, textHeight;
2848 long lineWidth, lineHeight;
2849 wxArrayString lines;
2850
2851 dc.SetClippingRegion( rect );
2852 StringToLines( value, lines );
2853 if ( lines.GetCount() )
2854 {
2855 GetTextBoxSize( dc, lines, &textWidth, &textHeight );
2856 dc.GetTextExtent( lines[0], &lineWidth, &lineHeight );
2857
2858 float x, y;
2859 switch ( horizAlign )
2860 {
2861 case wxRIGHT:
2862 x = rect.x + (rect.width - textWidth - 1);
2863 break;
2864
2865 case wxCENTRE:
2866 x = rect.x + ((rect.width - textWidth)/2);
2867 break;
2868
2869 case wxLEFT:
2870 default:
2871 x = rect.x + 1;
2872 break;
2873 }
2874
2875 switch ( vertAlign )
2876 {
2877 case wxBOTTOM:
2878 y = rect.y + (rect.height - textHeight - 1);
2879 break;
2880
2881 case wxCENTRE:
2882 y = rect.y + ((rect.height - textHeight)/2);
2883 break;
2884
2885 case wxTOP:
2886 default:
2887 y = rect.y + 1;
2888 break;
2889 }
2890
2891 for ( size_t i = 0; i < lines.GetCount(); i++ )
2892 {
2893 dc.DrawText( lines[i], (long)x, (long)y );
2894 y += lineHeight;
2895 }
2896 }
2897
2898 dc.DestroyClippingRegion();
2899 }
2900
2901
2902 // Split multi line text up into an array of strings. Any existing
2903 // contents of the string array are preserved.
2904 //
2905 void wxGrid::StringToLines( const wxString& value, wxArrayString& lines )
2906 {
2907 // TODO: this won't work for WXMAC ? (lines end with '\r')
2908 // => use wxTextFile functions then (VZ)
2909 int startPos = 0;
2910 int pos;
2911 while ( startPos < (int)value.Length() )
2912 {
2913 pos = value.Mid(startPos).Find( '\n' );
2914 if ( pos < 0 )
2915 {
2916 break;
2917 }
2918 else if ( pos == 0 )
2919 {
2920 lines.Add( wxEmptyString );
2921 }
2922 else
2923 {
2924 if ( value[startPos+pos-1] == '\r' )
2925 {
2926 lines.Add( value.Mid(startPos, pos-1) );
2927 }
2928 else
2929 {
2930 lines.Add( value.Mid(startPos, pos) );
2931 }
2932 }
2933 startPos += pos+1;
2934 }
2935 if ( startPos < (int)value.Length() )
2936 {
2937 lines.Add( value.Mid( startPos ) );
2938 }
2939 }
2940
2941
2942 void wxGrid::GetTextBoxSize( wxDC& dc,
2943 wxArrayString& lines,
2944 long *width, long *height )
2945 {
2946 long w = 0;
2947 long h = 0;
2948 long lineW, lineH;
2949
2950 size_t i;
2951 for ( i = 0; i < lines.GetCount(); i++ )
2952 {
2953 dc.GetTextExtent( lines[i], &lineW, &lineH );
2954 w = wxMax( w, lineW );
2955 h += lineH;
2956 }
2957
2958 *width = w;
2959 *height = h;
2960 }
2961
2962
2963 //
2964 // ------ Edit control functions
2965 //
2966
2967
2968 void wxGrid::EnableEditing( bool edit )
2969 {
2970 // TODO: improve this ?
2971 //
2972 if ( edit != m_editable )
2973 {
2974 m_editable = edit;
2975
2976 // TODO: extend this for other edit control types
2977 //
2978 if ( m_editCtrlType == wxGRID_TEXTCTRL )
2979 {
2980 ((wxTextCtrl *)m_cellEditCtrl)->SetEditable( m_editable );
2981 }
2982 }
2983 }
2984
2985
2986 #if 0 // disabled for the moment - the cell control is always active
2987 void wxGrid::EnableCellEditControl( bool enable )
2988 {
2989 if ( m_cellEditCtrl &&
2990 enable != m_cellEditCtrlEnabled )
2991 {
2992 m_cellEditCtrlEnabled = enable;
2993
2994 if ( m_cellEditCtrlEnabled )
2995 {
2996 SetEditControlValue();
2997 ShowCellEditControl();
2998 }
2999 else
3000 {
3001 HideCellEditControl();
3002 SaveEditControlValue();
3003 }
3004 }
3005 }
3006 #endif
3007
3008
3009 void wxGrid::ShowCellEditControl()
3010 {
3011 wxRect rect;
3012
3013 if ( IsCellEditControlEnabled() )
3014 {
3015 if ( !IsVisible( m_currentCellCoords ) )
3016 {
3017 return;
3018 }
3019 else
3020 {
3021 rect = CellToRect( m_currentCellCoords );
3022
3023 // convert to scrolled coords
3024 //
3025 int left, top, right, bottom;
3026 CalcScrolledPosition( rect.GetLeft(), rect.GetTop(), &left, &top );
3027 CalcScrolledPosition( rect.GetRight(), rect.GetBottom(), &right, &bottom );
3028
3029 int cw, ch;
3030 m_gridWin->GetClientSize( &cw, &ch );
3031
3032 // Make the edit control large enough to allow for internal margins
3033 // TODO: remove this if the text ctrl sizing is improved esp. for unix
3034 //
3035 int extra;
3036 #if defined(__WXMOTIF__)
3037 if ( m_currentCellCoords.GetRow() == 0 ||
3038 m_currentCellCoords.GetCol() == 0 )
3039 {
3040 extra = 2;
3041 }
3042 else
3043 {
3044 extra = 4;
3045 }
3046 #else
3047 if ( m_currentCellCoords.GetRow() == 0 ||
3048 m_currentCellCoords.GetCol() == 0 )
3049 {
3050 extra = 1;
3051 }
3052 else
3053 {
3054 extra = 2;
3055 }
3056 #endif
3057
3058 #if defined(__WXGTK__)
3059 int top_diff = 0;
3060 int left_diff = 0;
3061 if (left != 0) left_diff++;
3062 if (top != 0) top_diff++;
3063 rect.SetLeft( left + left_diff );
3064 rect.SetTop( top + top_diff );
3065 rect.SetRight( rect.GetRight() - left_diff );
3066 rect.SetBottom( rect.GetBottom() - top_diff );
3067 #else
3068 rect.SetLeft( wxMax(0, left - extra) );
3069 rect.SetTop( wxMax(0, top - extra) );
3070 rect.SetRight( rect.GetRight() + 2*extra );
3071 rect.SetBottom( rect.GetBottom() + 2*extra );
3072 #endif
3073
3074 m_cellEditCtrl->SetSize( rect );
3075 m_cellEditCtrl->Show( TRUE );
3076
3077 switch ( m_editCtrlType )
3078 {
3079 case wxGRID_TEXTCTRL:
3080 ((wxTextCtrl *) m_cellEditCtrl)->SetInsertionPointEnd();
3081 break;
3082
3083 case wxGRID_CHECKBOX:
3084 // TODO: anything ???
3085 //
3086 break;
3087
3088 case wxGRID_CHOICE:
3089 // TODO: anything ???
3090 //
3091 break;
3092
3093 case wxGRID_COMBOBOX:
3094 // TODO: anything ???
3095 //
3096 break;
3097 }
3098
3099 m_cellEditCtrl->SetFocus();
3100 }
3101 }
3102 }
3103
3104
3105 void wxGrid::HideCellEditControl()
3106 {
3107 if ( IsCellEditControlEnabled() )
3108 {
3109 m_cellEditCtrl->Show( FALSE );
3110 }
3111 }
3112
3113
3114 void wxGrid::SetEditControlValue( const wxString& value )
3115 {
3116 if ( m_table )
3117 {
3118 wxString s;
3119 if ( !value )
3120 s = GetCellValue(m_currentCellCoords);
3121 else
3122 s = value;
3123
3124 if ( IsCellEditControlEnabled() )
3125 {
3126 switch ( m_editCtrlType )
3127 {
3128 case wxGRID_TEXTCTRL:
3129 ((wxGridTextCtrl *)m_cellEditCtrl)->SetStartValue(s);
3130 break;
3131
3132 case wxGRID_CHECKBOX:
3133 // TODO: implement this
3134 //
3135 break;
3136
3137 case wxGRID_CHOICE:
3138 // TODO: implement this
3139 //
3140 break;
3141
3142 case wxGRID_COMBOBOX:
3143 // TODO: implement this
3144 //
3145 break;
3146 }
3147 }
3148 }
3149 }
3150
3151
3152 void wxGrid::SaveEditControlValue()
3153 {
3154 if ( m_table )
3155 {
3156 wxWindow *ctrl = (wxWindow *)NULL;
3157
3158 if ( IsCellEditControlEnabled() )
3159 {
3160 ctrl = m_cellEditCtrl;
3161 }
3162 else
3163 {
3164 return;
3165 }
3166
3167 bool valueChanged = FALSE;
3168
3169 switch ( m_editCtrlType )
3170 {
3171 case wxGRID_TEXTCTRL:
3172 valueChanged = (((wxGridTextCtrl *)ctrl)->GetValue() !=
3173 ((wxGridTextCtrl *)ctrl)->GetStartValue());
3174 SetCellValue( m_currentCellCoords,
3175 ((wxTextCtrl *) ctrl)->GetValue() );
3176 break;
3177
3178 case wxGRID_CHECKBOX:
3179 // TODO: implement this
3180 //
3181 break;
3182
3183 case wxGRID_CHOICE:
3184 // TODO: implement this
3185 //
3186 break;
3187
3188 case wxGRID_COMBOBOX:
3189 // TODO: implement this
3190 //
3191 break;
3192 }
3193
3194 if ( valueChanged )
3195 {
3196 SendEvent( EVT_GRID_CELL_CHANGE,
3197 m_currentCellCoords.GetRow(),
3198 m_currentCellCoords.GetCol() );
3199 }
3200 }
3201 }
3202
3203
3204 //
3205 // ------ Grid location functions
3206 // Note that all of these functions work with the logical coordinates of
3207 // grid cells and labels so you will need to convert from device
3208 // coordinates for mouse events etc.
3209 //
3210
3211 void wxGrid::XYToCell( int x, int y, wxGridCellCoords& coords )
3212 {
3213 int row = YToRow(y);
3214 int col = XToCol(x);
3215
3216 if ( row == -1 || col == -1 )
3217 {
3218 coords = wxGridNoCellCoords;
3219 }
3220 else
3221 {
3222 coords.Set( row, col );
3223 }
3224 }
3225
3226
3227 int wxGrid::YToRow( int y )
3228 {
3229 int i;
3230
3231 for ( i = 0; i < m_numRows; i++ )
3232 {
3233 if ( y < m_rowBottoms[i] ) return i;
3234 }
3235
3236 return -1;
3237 }
3238
3239
3240 int wxGrid::XToCol( int x )
3241 {
3242 int i;
3243
3244 for ( i = 0; i < m_numCols; i++ )
3245 {
3246 if ( x < m_colRights[i] ) return i;
3247 }
3248
3249 return -1;
3250 }
3251
3252
3253 // return the row number that that the y coord is near the edge of, or
3254 // -1 if not near an edge
3255 //
3256 int wxGrid::YToEdgeOfRow( int y )
3257 {
3258 int i, d;
3259
3260 for ( i = 0; i < m_numRows; i++ )
3261 {
3262 if ( m_rowHeights[i] > WXGRID_LABEL_EDGE_ZONE )
3263 {
3264 d = abs( y - m_rowBottoms[i] );
3265 {
3266 if ( d < WXGRID_LABEL_EDGE_ZONE ) return i;
3267 }
3268 }
3269 }
3270
3271 return -1;
3272 }
3273
3274
3275 // return the col number that that the x coord is near the edge of, or
3276 // -1 if not near an edge
3277 //
3278 int wxGrid::XToEdgeOfCol( int x )
3279 {
3280 int i, d;
3281
3282 for ( i = 0; i < m_numCols; i++ )
3283 {
3284 if ( m_colWidths[i] > WXGRID_LABEL_EDGE_ZONE )
3285 {
3286 d = abs( x - m_colRights[i] );
3287 {
3288 if ( d < WXGRID_LABEL_EDGE_ZONE ) return i;
3289 }
3290 }
3291 }
3292
3293 return -1;
3294 }
3295
3296
3297 wxRect wxGrid::CellToRect( int row, int col )
3298 {
3299 wxRect rect( -1, -1, -1, -1 );
3300
3301 if ( row >= 0 && row < m_numRows &&
3302 col >= 0 && col < m_numCols )
3303 {
3304 rect.x = m_colRights[col] - m_colWidths[col];
3305 rect.y = m_rowBottoms[row] - m_rowHeights[row];
3306 rect.width = m_colWidths[col];
3307 rect.height = m_rowHeights[ row ];
3308 }
3309
3310 return rect;
3311 }
3312
3313
3314 bool wxGrid::IsVisible( int row, int col, bool wholeCellVisible )
3315 {
3316 // get the cell rectangle in logical coords
3317 //
3318 wxRect r( CellToRect( row, col ) );
3319
3320 // convert to device coords
3321 //
3322 int left, top, right, bottom;
3323 CalcScrolledPosition( r.GetLeft(), r.GetTop(), &left, &top );
3324 CalcScrolledPosition( r.GetRight(), r.GetBottom(), &right, &bottom );
3325
3326 // check against the client area of the grid window
3327 //
3328 int cw, ch;
3329 m_gridWin->GetClientSize( &cw, &ch );
3330
3331 if ( wholeCellVisible )
3332 {
3333 // is the cell wholly visible ?
3334 //
3335 return ( left >= 0 && right <= cw &&
3336 top >= 0 && bottom <= ch );
3337 }
3338 else
3339 {
3340 // is the cell partly visible ?
3341 //
3342 return ( ((left >=0 && left < cw) || (right > 0 && right <= cw)) &&
3343 ((top >=0 && top < ch) || (bottom > 0 && bottom <= ch)) );
3344 }
3345 }
3346
3347
3348 // make the specified cell location visible by doing a minimal amount
3349 // of scrolling
3350 //
3351 void wxGrid::MakeCellVisible( int row, int col )
3352 {
3353 int i;
3354 int xpos = -1, ypos = -1;
3355
3356 if ( row >= 0 && row < m_numRows &&
3357 col >= 0 && col < m_numCols )
3358 {
3359 // get the cell rectangle in logical coords
3360 //
3361 wxRect r( CellToRect( row, col ) );
3362
3363 // convert to device coords
3364 //
3365 int left, top, right, bottom;
3366 CalcScrolledPosition( r.GetLeft(), r.GetTop(), &left, &top );
3367 CalcScrolledPosition( r.GetRight(), r.GetBottom(), &right, &bottom );
3368
3369 int cw, ch;
3370 m_gridWin->GetClientSize( &cw, &ch );
3371
3372 if ( top < 0 )
3373 {
3374 ypos = r.GetTop();
3375 }
3376 else if ( bottom > ch )
3377 {
3378 int h = r.GetHeight();
3379 ypos = r.GetTop();
3380 for ( i = row-1; i >= 0; i-- )
3381 {
3382 if ( h + m_rowHeights[i] > ch ) break;
3383
3384 h += m_rowHeights[i];
3385 ypos -= m_rowHeights[i];
3386 }
3387
3388 // we divide it later by GRID_SCROLL_LINE, make sure that we don't
3389 // have rounding errors (this is important, because if we do, we
3390 // might not scroll at all and some cells won't be redrawn)
3391 ypos += GRID_SCROLL_LINE / 2;
3392 }
3393
3394 if ( left < 0 )
3395 {
3396 xpos = r.GetLeft();
3397 }
3398 else if ( right > cw )
3399 {
3400 int w = r.GetWidth();
3401 xpos = r.GetLeft();
3402 for ( i = col-1; i >= 0; i-- )
3403 {
3404 if ( w + m_colWidths[i] > cw ) break;
3405
3406 w += m_colWidths[i];
3407 xpos -= m_colWidths[i];
3408 }
3409
3410 // see comment for ypos above
3411 xpos += GRID_SCROLL_LINE / 2;
3412 }
3413
3414 if ( xpos != -1 || ypos != -1 )
3415 {
3416 if ( xpos != -1 ) xpos /= GRID_SCROLL_LINE;
3417 if ( ypos != -1 ) ypos /= GRID_SCROLL_LINE;
3418 Scroll( xpos, ypos );
3419 AdjustScrollbars();
3420 }
3421 }
3422 }
3423
3424
3425 //
3426 // ------ Grid cursor movement functions
3427 //
3428
3429 bool wxGrid::MoveCursorUp()
3430 {
3431 if ( m_currentCellCoords != wxGridNoCellCoords &&
3432 m_currentCellCoords.GetRow() > 0 )
3433 {
3434 MakeCellVisible( m_currentCellCoords.GetRow() - 1,
3435 m_currentCellCoords.GetCol() );
3436
3437 SetCurrentCell( m_currentCellCoords.GetRow() - 1,
3438 m_currentCellCoords.GetCol() );
3439
3440 return TRUE;
3441 }
3442
3443 return FALSE;
3444 }
3445
3446
3447 bool wxGrid::MoveCursorDown()
3448 {
3449 // TODO: allow for scrolling
3450 //
3451 if ( m_currentCellCoords != wxGridNoCellCoords &&
3452 m_currentCellCoords.GetRow() < m_numRows-1 )
3453 {
3454 MakeCellVisible( m_currentCellCoords.GetRow() + 1,
3455 m_currentCellCoords.GetCol() );
3456
3457 SetCurrentCell( m_currentCellCoords.GetRow() + 1,
3458 m_currentCellCoords.GetCol() );
3459
3460 return TRUE;
3461 }
3462
3463 return FALSE;
3464 }
3465
3466
3467 bool wxGrid::MoveCursorLeft()
3468 {
3469 if ( m_currentCellCoords != wxGridNoCellCoords &&
3470 m_currentCellCoords.GetCol() > 0 )
3471 {
3472 MakeCellVisible( m_currentCellCoords.GetRow(),
3473 m_currentCellCoords.GetCol() - 1 );
3474
3475 SetCurrentCell( m_currentCellCoords.GetRow(),
3476 m_currentCellCoords.GetCol() - 1 );
3477
3478 return TRUE;
3479 }
3480
3481 return FALSE;
3482 }
3483
3484
3485 bool wxGrid::MoveCursorRight()
3486 {
3487 if ( m_currentCellCoords != wxGridNoCellCoords &&
3488 m_currentCellCoords.GetCol() < m_numCols - 1 )
3489 {
3490 MakeCellVisible( m_currentCellCoords.GetRow(),
3491 m_currentCellCoords.GetCol() + 1 );
3492
3493 SetCurrentCell( m_currentCellCoords.GetRow(),
3494 m_currentCellCoords.GetCol() + 1 );
3495
3496 return TRUE;
3497 }
3498
3499 return FALSE;
3500 }
3501
3502
3503 bool wxGrid::MovePageUp()
3504 {
3505 if ( m_currentCellCoords == wxGridNoCellCoords ) return FALSE;
3506
3507 int row = m_currentCellCoords.GetRow();
3508 if ( row > 0 )
3509 {
3510 int cw, ch;
3511 m_gridWin->GetClientSize( &cw, &ch );
3512
3513 int y = m_rowBottoms[ row ] - m_rowHeights[ row ];
3514 int newRow = YToRow( y - ch + 1 );
3515 if ( newRow == -1 )
3516 {
3517 newRow = 0;
3518 }
3519 else if ( newRow == row )
3520 {
3521 newRow = row - 1;
3522 }
3523
3524 MakeCellVisible( newRow, m_currentCellCoords.GetCol() );
3525 SetCurrentCell( newRow, m_currentCellCoords.GetCol() );
3526
3527 return TRUE;
3528 }
3529
3530 return FALSE;
3531 }
3532
3533 bool wxGrid::MovePageDown()
3534 {
3535 if ( m_currentCellCoords == wxGridNoCellCoords ) return FALSE;
3536
3537 int row = m_currentCellCoords.GetRow();
3538 if ( row < m_numRows )
3539 {
3540 int cw, ch;
3541 m_gridWin->GetClientSize( &cw, &ch );
3542
3543 int y = m_rowBottoms[ row ] - m_rowHeights[ row ];
3544 int newRow = YToRow( y + ch );
3545 if ( newRow == -1 )
3546 {
3547 newRow = m_numRows - 1;
3548 }
3549 else if ( newRow == row )
3550 {
3551 newRow = row + 1;
3552 }
3553
3554 MakeCellVisible( newRow, m_currentCellCoords.GetCol() );
3555 SetCurrentCell( newRow, m_currentCellCoords.GetCol() );
3556
3557 return TRUE;
3558 }
3559
3560 return FALSE;
3561 }
3562
3563 bool wxGrid::MoveCursorUpBlock()
3564 {
3565 if ( m_table &&
3566 m_currentCellCoords != wxGridNoCellCoords &&
3567 m_currentCellCoords.GetRow() > 0 )
3568 {
3569 int row = m_currentCellCoords.GetRow();
3570 int col = m_currentCellCoords.GetCol();
3571
3572 if ( m_table->IsEmptyCell(row, col) )
3573 {
3574 // starting in an empty cell: find the next block of
3575 // non-empty cells
3576 //
3577 while ( row > 0 )
3578 {
3579 row-- ;
3580 if ( !(m_table->IsEmptyCell(row, col)) ) break;
3581 }
3582 }
3583 else if ( m_table->IsEmptyCell(row-1, col) )
3584 {
3585 // starting at the top of a block: find the next block
3586 //
3587 row--;
3588 while ( row > 0 )
3589 {
3590 row-- ;
3591 if ( !(m_table->IsEmptyCell(row, col)) ) break;
3592 }
3593 }
3594 else
3595 {
3596 // starting within a block: find the top of the block
3597 //
3598 while ( row > 0 )
3599 {
3600 row-- ;
3601 if ( m_table->IsEmptyCell(row, col) )
3602 {
3603 row++ ;
3604 break;
3605 }
3606 }
3607 }
3608
3609 MakeCellVisible( row, col );
3610 SetCurrentCell( row, col );
3611
3612 return TRUE;
3613 }
3614
3615 return FALSE;
3616 }
3617
3618 bool wxGrid::MoveCursorDownBlock()
3619 {
3620 if ( m_table &&
3621 m_currentCellCoords != wxGridNoCellCoords &&
3622 m_currentCellCoords.GetRow() < m_numRows-1 )
3623 {
3624 int row = m_currentCellCoords.GetRow();
3625 int col = m_currentCellCoords.GetCol();
3626
3627 if ( m_table->IsEmptyCell(row, col) )
3628 {
3629 // starting in an empty cell: find the next block of
3630 // non-empty cells
3631 //
3632 while ( row < m_numRows-1 )
3633 {
3634 row++ ;
3635 if ( !(m_table->IsEmptyCell(row, col)) ) break;
3636 }
3637 }
3638 else if ( m_table->IsEmptyCell(row+1, col) )
3639 {
3640 // starting at the bottom of a block: find the next block
3641 //
3642 row++;
3643 while ( row < m_numRows-1 )
3644 {
3645 row++ ;
3646 if ( !(m_table->IsEmptyCell(row, col)) ) break;
3647 }
3648 }
3649 else
3650 {
3651 // starting within a block: find the bottom of the block
3652 //
3653 while ( row < m_numRows-1 )
3654 {
3655 row++ ;
3656 if ( m_table->IsEmptyCell(row, col) )
3657 {
3658 row-- ;
3659 break;
3660 }
3661 }
3662 }
3663
3664 MakeCellVisible( row, col );
3665 SetCurrentCell( row, col );
3666
3667 return TRUE;
3668 }
3669
3670 return FALSE;
3671 }
3672
3673 bool wxGrid::MoveCursorLeftBlock()
3674 {
3675 if ( m_table &&
3676 m_currentCellCoords != wxGridNoCellCoords &&
3677 m_currentCellCoords.GetCol() > 0 )
3678 {
3679 int row = m_currentCellCoords.GetRow();
3680 int col = m_currentCellCoords.GetCol();
3681
3682 if ( m_table->IsEmptyCell(row, col) )
3683 {
3684 // starting in an empty cell: find the next block of
3685 // non-empty cells
3686 //
3687 while ( col > 0 )
3688 {
3689 col-- ;
3690 if ( !(m_table->IsEmptyCell(row, col)) ) break;
3691 }
3692 }
3693 else if ( m_table->IsEmptyCell(row, col-1) )
3694 {
3695 // starting at the left of a block: find the next block
3696 //
3697 col--;
3698 while ( col > 0 )
3699 {
3700 col-- ;
3701 if ( !(m_table->IsEmptyCell(row, col)) ) break;
3702 }
3703 }
3704 else
3705 {
3706 // starting within a block: find the left of the block
3707 //
3708 while ( col > 0 )
3709 {
3710 col-- ;
3711 if ( m_table->IsEmptyCell(row, col) )
3712 {
3713 col++ ;
3714 break;
3715 }
3716 }
3717 }
3718
3719 MakeCellVisible( row, col );
3720 SetCurrentCell( row, col );
3721
3722 return TRUE;
3723 }
3724
3725 return FALSE;
3726 }
3727
3728 bool wxGrid::MoveCursorRightBlock()
3729 {
3730 if ( m_table &&
3731 m_currentCellCoords != wxGridNoCellCoords &&
3732 m_currentCellCoords.GetCol() < m_numCols-1 )
3733 {
3734 int row = m_currentCellCoords.GetRow();
3735 int col = m_currentCellCoords.GetCol();
3736
3737 if ( m_table->IsEmptyCell(row, col) )
3738 {
3739 // starting in an empty cell: find the next block of
3740 // non-empty cells
3741 //
3742 while ( col < m_numCols-1 )
3743 {
3744 col++ ;
3745 if ( !(m_table->IsEmptyCell(row, col)) ) break;
3746 }
3747 }
3748 else if ( m_table->IsEmptyCell(row, col+1) )
3749 {
3750 // starting at the right of a block: find the next block
3751 //
3752 col++;
3753 while ( col < m_numCols-1 )
3754 {
3755 col++ ;
3756 if ( !(m_table->IsEmptyCell(row, col)) ) break;
3757 }
3758 }
3759 else
3760 {
3761 // starting within a block: find the right of the block
3762 //
3763 while ( col < m_numCols-1 )
3764 {
3765 col++ ;
3766 if ( m_table->IsEmptyCell(row, col) )
3767 {
3768 col-- ;
3769 break;
3770 }
3771 }
3772 }
3773
3774 MakeCellVisible( row, col );
3775 SetCurrentCell( row, col );
3776
3777 return TRUE;
3778 }
3779
3780 return FALSE;
3781 }
3782
3783
3784
3785 //
3786 // ------ Label values and formatting
3787 //
3788
3789 void wxGrid::GetRowLabelAlignment( int *horiz, int *vert )
3790 {
3791 *horiz = m_rowLabelHorizAlign;
3792 *vert = m_rowLabelVertAlign;
3793 }
3794
3795 void wxGrid::GetColLabelAlignment( int *horiz, int *vert )
3796 {
3797 *horiz = m_colLabelHorizAlign;
3798 *vert = m_colLabelVertAlign;
3799 }
3800
3801 wxString wxGrid::GetRowLabelValue( int row )
3802 {
3803 if ( m_table )
3804 {
3805 return m_table->GetRowLabelValue( row );
3806 }
3807 else
3808 {
3809 wxString s;
3810 s << row;
3811 return s;
3812 }
3813 }
3814
3815 wxString wxGrid::GetColLabelValue( int col )
3816 {
3817 if ( m_table )
3818 {
3819 return m_table->GetColLabelValue( col );
3820 }
3821 else
3822 {
3823 wxString s;
3824 s << col;
3825 return s;
3826 }
3827 }
3828
3829 void wxGrid::SetRowLabelSize( int width )
3830 {
3831 // TODO: how to do this with the box sizers ?
3832 }
3833
3834 void wxGrid::SetColLabelSize( int height )
3835 {
3836 // TODO: how to do this with the box sizers ?
3837 }
3838
3839 void wxGrid::SetLabelBackgroundColour( const wxColour& colour )
3840 {
3841 if ( m_labelBackgroundColour != colour )
3842 {
3843 m_labelBackgroundColour = colour;
3844 m_rowLabelWin->SetBackgroundColour( colour );
3845 m_colLabelWin->SetBackgroundColour( colour );
3846 m_cornerLabelWin->SetBackgroundColour( colour );
3847
3848 if ( !GetBatchCount() )
3849 {
3850 m_rowLabelWin->Refresh();
3851 m_colLabelWin->Refresh();
3852 m_cornerLabelWin->Refresh();
3853 }
3854 }
3855 }
3856
3857 void wxGrid::SetLabelTextColour( const wxColour& colour )
3858 {
3859 if ( m_labelTextColour != colour )
3860 {
3861 m_labelTextColour = colour;
3862 if ( !GetBatchCount() )
3863 {
3864 m_rowLabelWin->Refresh();
3865 m_colLabelWin->Refresh();
3866 }
3867 }
3868 }
3869
3870 void wxGrid::SetLabelFont( const wxFont& font )
3871 {
3872 m_labelFont = font;
3873 if ( !GetBatchCount() )
3874 {
3875 m_rowLabelWin->Refresh();
3876 m_colLabelWin->Refresh();
3877 }
3878 }
3879
3880 void wxGrid::SetRowLabelAlignment( int horiz, int vert )
3881 {
3882 if ( horiz == wxLEFT || horiz == wxCENTRE || horiz == wxRIGHT )
3883 {
3884 m_rowLabelHorizAlign = horiz;
3885 }
3886
3887 if ( vert == wxTOP || vert == wxCENTRE || vert == wxBOTTOM )
3888 {
3889 m_rowLabelVertAlign = vert;
3890 }
3891
3892 if ( !GetBatchCount() )
3893 {
3894 m_rowLabelWin->Refresh();
3895 m_colLabelWin->Refresh();
3896 }
3897 }
3898
3899 void wxGrid::SetColLabelAlignment( int horiz, int vert )
3900 {
3901 if ( horiz == wxLEFT || horiz == wxCENTRE || horiz == wxRIGHT )
3902 {
3903 m_colLabelHorizAlign = horiz;
3904 }
3905
3906 if ( vert == wxTOP || vert == wxCENTRE || vert == wxBOTTOM )
3907 {
3908 m_colLabelVertAlign = vert;
3909 }
3910
3911 if ( !GetBatchCount() )
3912 {
3913 m_rowLabelWin->Refresh();
3914 m_colLabelWin->Refresh();
3915 }
3916 }
3917
3918 void wxGrid::SetRowLabelValue( int row, const wxString& s )
3919 {
3920 if ( m_table )
3921 {
3922 m_table->SetRowLabelValue( row, s );
3923 if ( !GetBatchCount() )
3924 {
3925 // TODO: Optimize this
3926 //
3927 m_rowLabelWin->Refresh();
3928 }
3929 }
3930 }
3931
3932 void wxGrid::SetColLabelValue( int col, const wxString& s )
3933 {
3934 if ( m_table )
3935 {
3936 m_table->SetColLabelValue( col, s );
3937 if ( !GetBatchCount() )
3938 {
3939 // TODO: optimize this
3940 //
3941 m_colLabelWin->Refresh();
3942 }
3943 }
3944 }
3945
3946 void wxGrid::SetGridLineColour( const wxColour& colour )
3947 {
3948 if ( m_gridLineColour != colour )
3949 {
3950 m_gridLineColour = colour;
3951
3952 wxClientDC dc( m_gridWin );
3953 PrepareDC( dc );
3954 DrawAllGridLines( dc );
3955 }
3956 }
3957
3958 void wxGrid::EnableGridLines( bool enable )
3959 {
3960 if ( enable != m_gridLinesEnabled )
3961 {
3962 m_gridLinesEnabled = enable;
3963
3964 if ( !GetBatchCount() )
3965 {
3966 if ( enable )
3967 {
3968 wxClientDC dc( m_gridWin );
3969 PrepareDC( dc );
3970 DrawAllGridLines( dc );
3971 }
3972 else
3973 {
3974 m_gridWin->Refresh();
3975 }
3976 }
3977 }
3978 }
3979
3980
3981 int wxGrid::GetDefaultRowSize()
3982 {
3983 return m_defaultRowHeight;
3984 }
3985
3986 int wxGrid::GetRowSize( int row )
3987 {
3988 if ( row >= 0 && row < m_numRows )
3989 return m_rowHeights[row];
3990 else
3991 return 0; // TODO: log an error here
3992 }
3993
3994 int wxGrid::GetDefaultColSize()
3995 {
3996 return m_defaultColWidth;
3997 }
3998
3999 int wxGrid::GetColSize( int col )
4000 {
4001 if ( col >= 0 && col < m_numCols )
4002 return m_colWidths[col];
4003 else
4004 return 0; // TODO: log an error here
4005 }
4006
4007 wxColour wxGrid::GetDefaultCellBackgroundColour()
4008 {
4009 // TODO: replace this temp test code
4010 //
4011 return wxColour( 255, 255, 255 );
4012 }
4013
4014 wxColour wxGrid::GetCellBackgroundColour( int WXUNUSED(row), int WXUNUSED(col) )
4015 {
4016 // TODO: replace this temp test code
4017 //
4018 return wxColour( 255, 255, 255 );
4019 }
4020
4021 wxColour wxGrid::GetDefaultCellTextColour()
4022 {
4023 // TODO: replace this temp test code
4024 //
4025 return wxColour( 0, 0, 0 );
4026 }
4027
4028 wxColour wxGrid::GetCellTextColour( int WXUNUSED(row), int WXUNUSED(col) )
4029 {
4030 // TODO: replace this temp test code
4031 //
4032 return wxColour( 0, 0, 0 );
4033 }
4034
4035
4036 wxFont wxGrid::GetDefaultCellFont()
4037 {
4038 return m_defaultCellFont;
4039 }
4040
4041 wxFont wxGrid::GetCellFont( int WXUNUSED(row), int WXUNUSED(col) )
4042 {
4043 // TODO: replace this temp test code
4044 //
4045 return m_defaultCellFont;
4046 }
4047
4048 void wxGrid::GetDefaultCellAlignment( int *horiz, int *vert )
4049 {
4050 // TODO: replace this temp test code
4051 //
4052 *horiz = wxLEFT;
4053 *vert = wxTOP;
4054 }
4055
4056 void wxGrid::GetCellAlignment( int WXUNUSED(row), int WXUNUSED(col), int *horiz, int *vert )
4057 {
4058 // TODO: replace this temp test code
4059 //
4060 *horiz = wxLEFT;
4061 *vert = wxTOP;
4062 }
4063
4064 void wxGrid::SetDefaultRowSize( int height, bool resizeExistingRows )
4065 {
4066 m_defaultRowHeight = wxMax( height, WXGRID_MIN_ROW_HEIGHT );
4067
4068 if ( resizeExistingRows )
4069 {
4070 int row;
4071 int bottom = 0;
4072 for ( row = 0; row < m_numRows; row++ )
4073 {
4074 m_rowHeights[row] = m_defaultRowHeight;
4075 bottom += m_defaultRowHeight;
4076 m_rowBottoms[row] = bottom;
4077 }
4078 CalcDimensions();
4079 }
4080 }
4081
4082 void wxGrid::SetRowSize( int row, int height )
4083 {
4084 int i;
4085
4086 if ( row >= 0 && row < m_numRows )
4087 {
4088 int h = wxMax( 0, height );
4089 int diff = h - m_rowHeights[row];
4090
4091 m_rowHeights[row] = h;
4092 for ( i = row; i < m_numRows; i++ )
4093 {
4094 m_rowBottoms[i] += diff;
4095 }
4096 CalcDimensions();
4097
4098 // Note: we are ending the event *after* doing
4099 // default processing in this case
4100 //
4101 SendEvent( EVT_GRID_ROW_SIZE,
4102 row, -1 );
4103 }
4104 else
4105 {
4106 // TODO: log an error here
4107 }
4108 }
4109
4110 void wxGrid::SetDefaultColSize( int width, bool resizeExistingCols )
4111 {
4112 m_defaultColWidth = wxMax( width, WXGRID_MIN_COL_WIDTH );
4113
4114 if ( resizeExistingCols )
4115 {
4116 int col;
4117 int right = 0;
4118 for ( col = 0; col < m_numCols; col++ )
4119 {
4120 m_colWidths[col] = m_defaultColWidth;
4121 right += m_defaultColWidth;
4122 m_colRights[col] = right;
4123 }
4124 CalcDimensions();
4125 }
4126 }
4127
4128 void wxGrid::SetColSize( int col, int width )
4129 {
4130 int i;
4131
4132 if ( col >= 0 && col < m_numCols )
4133 {
4134 int w = wxMax( 0, width );
4135 int diff = w - m_colWidths[col];
4136 m_colWidths[col] = w;
4137
4138 for ( i = col; i < m_numCols; i++ )
4139 {
4140 m_colRights[i] += diff;
4141 }
4142 CalcDimensions();
4143
4144 // Note: we are ending the event *after* doing
4145 // default processing in this case
4146 //
4147 SendEvent( EVT_GRID_COL_SIZE,
4148 -1, col );
4149 }
4150 else
4151 {
4152 // TODO: log an error here
4153 }
4154 }
4155
4156 void wxGrid::SetDefaultCellBackgroundColour( const wxColour& )
4157 {
4158 // TODO: everything !!!
4159 //
4160 }
4161
4162 void wxGrid::SetCellBackgroundColour( int WXUNUSED(row), int WXUNUSED(col), const wxColour& )
4163 {
4164 // TODO: everything !!!
4165 //
4166 }
4167
4168 void wxGrid::SetDefaultCellTextColour( const wxColour& )
4169 {
4170 // TODO: everything !!!
4171 //
4172 }
4173
4174 void wxGrid::SetCellTextColour( int WXUNUSED(row), int WXUNUSED(col), const wxColour& )
4175 {
4176 // TODO: everything !!!
4177 //
4178 }
4179
4180 void wxGrid::SetDefaultCellFont( const wxFont& )
4181 {
4182 // TODO: everything !!!
4183 //
4184 }
4185
4186 void wxGrid::SetCellFont( int WXUNUSED(row), int WXUNUSED(col), const wxFont& )
4187 {
4188 // TODO: everything !!!
4189 //
4190 }
4191
4192 void wxGrid::SetDefaultCellAlignment( int WXUNUSED(horiz), int WXUNUSED(vert) )
4193 {
4194 // TODO: everything !!!
4195 //
4196 }
4197
4198 void wxGrid::SetCellAlignment( int WXUNUSED(row), int WXUNUSED(col), int WXUNUSED(horiz), int WXUNUSED(vert) )
4199 {
4200 // TODO: everything !!!
4201 //
4202 }
4203
4204
4205
4206 //
4207 // ------ cell value accessor functions
4208 //
4209
4210 void wxGrid::SetCellValue( int row, int col, const wxString& s )
4211 {
4212 if ( m_table )
4213 {
4214 m_table->SetValue( row, col, s.c_str() );
4215 if ( !GetBatchCount() )
4216 {
4217 wxClientDC dc( m_gridWin );
4218 PrepareDC( dc );
4219 DrawCell( dc, wxGridCellCoords(row, col) );
4220 }
4221
4222 #if 0 // TODO: edit in place
4223
4224 if ( m_currentCellCoords.GetRow() == row &&
4225 m_currentCellCoords.GetCol() == col )
4226 {
4227 SetEditControlValue( s );
4228 }
4229 #endif
4230
4231 }
4232 }
4233
4234
4235 //
4236 // ------ Block, row and col selection
4237 //
4238
4239 void wxGrid::SelectRow( int row, bool addToSelected )
4240 {
4241 wxRect r;
4242
4243 if ( IsSelection() && addToSelected )
4244 {
4245 if ( m_selectedTopLeft.GetRow() > row )
4246 m_selectedTopLeft.SetRow( row );
4247
4248 m_selectedTopLeft.SetCol( 0 );
4249
4250 if ( m_selectedBottomRight.GetRow() < row )
4251 m_selectedBottomRight.SetRow( row );
4252
4253 m_selectedBottomRight.SetCol( m_numCols - 1 );
4254
4255 // TODO: optimize this so that we only refresh the newly
4256 // selected cells
4257 //
4258 r = SelectionToDeviceRect();
4259 if ( r != wxGridNoCellRect ) m_gridWin->Refresh( TRUE, &r );
4260 }
4261 else
4262 {
4263 r = SelectionToDeviceRect();
4264 ClearSelection();
4265 if ( r != wxGridNoCellRect ) m_gridWin->Refresh( TRUE, &r );
4266
4267 m_selectedTopLeft.Set( row, 0 );
4268 m_selectedBottomRight.Set( row, m_numCols-1 );
4269 r = SelectionToDeviceRect();
4270 m_gridWin->Refresh( TRUE, &r );
4271 }
4272
4273 wxGridRangeSelectEvent gridEvt( GetId(),
4274 EVT_GRID_RANGE_SELECT,
4275 this,
4276 m_selectedTopLeft,
4277 m_selectedBottomRight );
4278
4279 GetEventHandler()->ProcessEvent(gridEvt);
4280 }
4281
4282
4283 void wxGrid::SelectCol( int col, bool addToSelected )
4284 {
4285 wxRect r;
4286
4287 if ( IsSelection() && addToSelected )
4288 {
4289 if ( m_selectedTopLeft.GetCol() > col )
4290 m_selectedTopLeft.SetCol( col );
4291
4292 m_selectedTopLeft.SetRow( 0 );
4293
4294 if ( m_selectedBottomRight.GetCol() < col )
4295 m_selectedBottomRight.SetCol( col );
4296
4297 m_selectedBottomRight.SetRow( m_numRows - 1 );
4298
4299 // TODO: optimize this so that we only refresh the newly
4300 // selected cells
4301 //
4302 r = SelectionToDeviceRect();
4303 if ( r != wxGridNoCellRect ) m_gridWin->Refresh( TRUE, &r );
4304 }
4305 else
4306 {
4307 r = SelectionToDeviceRect();
4308 ClearSelection();
4309 if ( r != wxGridNoCellRect ) m_gridWin->Refresh( TRUE, &r );
4310
4311 m_selectedTopLeft.Set( 0, col );
4312 m_selectedBottomRight.Set( m_numRows-1, col );
4313 r = SelectionToDeviceRect();
4314 m_gridWin->Refresh( TRUE, &r );
4315 }
4316
4317 wxGridRangeSelectEvent gridEvt( GetId(),
4318 EVT_GRID_RANGE_SELECT,
4319 this,
4320 m_selectedTopLeft,
4321 m_selectedBottomRight );
4322
4323 GetEventHandler()->ProcessEvent(gridEvt);
4324 }
4325
4326
4327 void wxGrid::SelectBlock( int topRow, int leftCol, int bottomRow, int rightCol )
4328 {
4329 int temp;
4330 bool changed = false;
4331 wxGridCellCoords updateTopLeft, updateBottomRight;
4332
4333 if ( topRow > bottomRow )
4334 {
4335 temp = topRow;
4336 topRow = bottomRow;
4337 bottomRow = temp;
4338 }
4339
4340 if ( leftCol > rightCol )
4341 {
4342 temp = leftCol;
4343 leftCol = rightCol;
4344 rightCol = temp;
4345 }
4346
4347 updateTopLeft = m_selectedTopLeft;
4348 if (m_selectedTopLeft != wxGridCellCoords( topRow, leftCol ) )
4349 {
4350 m_selectedTopLeft = wxGridCellCoords( topRow, leftCol );
4351 if (updateTopLeft == wxGridNoCellCoords)
4352 {
4353 updateTopLeft = m_selectedTopLeft;
4354 }
4355 else
4356 {
4357 if(updateTopLeft.GetRow() > topRow)
4358 updateTopLeft.SetRow(topRow);
4359 if (updateTopLeft.GetCol() > leftCol)
4360 updateTopLeft.SetCol(leftCol);
4361 }
4362 changed = true;
4363 }
4364
4365 updateBottomRight = m_selectedBottomRight;
4366 if (m_selectedBottomRight != wxGridCellCoords( bottomRow, rightCol ) )
4367 {
4368 m_selectedBottomRight = wxGridCellCoords( bottomRow, rightCol );
4369 if (updateBottomRight == wxGridNoCellCoords)
4370 {
4371 updateBottomRight = m_selectedBottomRight;
4372 }
4373 else
4374 {
4375 if (updateBottomRight.GetRow() < bottomRow)
4376 updateBottomRight.SetRow(bottomRow);
4377 if (updateBottomRight.GetCol() < rightCol)
4378 updateBottomRight.SetCol(rightCol);
4379 }
4380 changed = true;
4381 }
4382
4383 if (changed)
4384 {
4385 wxRect r( BlockToDeviceRect( updateTopLeft, updateBottomRight ) );
4386 m_gridWin->Refresh( TRUE, &r );
4387 }
4388
4389 // only generate an event if the block is not being selected by
4390 // dragging the mouse (in which case the event will be generated in
4391 // the mouse event handler)
4392 if ( !m_isDragging )
4393 {
4394 wxGridRangeSelectEvent gridEvt( GetId(),
4395 EVT_GRID_RANGE_SELECT,
4396 this,
4397 m_selectedTopLeft,
4398 m_selectedBottomRight );
4399
4400 GetEventHandler()->ProcessEvent(gridEvt);
4401 }
4402 }
4403
4404 void wxGrid::SelectAll()
4405 {
4406 m_selectedTopLeft.Set( 0, 0 );
4407 m_selectedBottomRight.Set( m_numRows-1, m_numCols-1 );
4408
4409 m_gridWin->Refresh();
4410 }
4411
4412
4413 void wxGrid::ClearSelection()
4414 {
4415 m_selectedTopLeft = wxGridNoCellCoords;
4416 m_selectedBottomRight = wxGridNoCellCoords;
4417 }
4418
4419
4420 // This function returns the rectangle that encloses the given block
4421 // in device coords clipped to the client size of the grid window.
4422 //
4423 wxRect wxGrid::BlockToDeviceRect( const wxGridCellCoords &topLeft,
4424 const wxGridCellCoords &bottomRight )
4425 {
4426 wxRect rect( wxGridNoCellRect );
4427 wxRect cellRect;
4428
4429 cellRect = CellToRect( topLeft );
4430 if ( cellRect != wxGridNoCellRect )
4431 {
4432 rect = cellRect;
4433 }
4434 else
4435 {
4436 rect = wxRect( 0, 0, 0, 0 );
4437 }
4438
4439 cellRect = CellToRect( bottomRight );
4440 if ( cellRect != wxGridNoCellRect )
4441 {
4442 rect += cellRect;
4443 }
4444 else
4445 {
4446 return wxGridNoCellRect;
4447 }
4448
4449 // convert to scrolled coords
4450 //
4451 int left, top, right, bottom;
4452 CalcScrolledPosition( rect.GetLeft(), rect.GetTop(), &left, &top );
4453 CalcScrolledPosition( rect.GetRight(), rect.GetBottom(), &right, &bottom );
4454
4455 int cw, ch;
4456 m_gridWin->GetClientSize( &cw, &ch );
4457
4458 rect.SetLeft( wxMax(0, left) );
4459 rect.SetTop( wxMax(0, top) );
4460 rect.SetRight( wxMin(cw, right) );
4461 rect.SetBottom( wxMin(ch, bottom) );
4462
4463 return rect;
4464 }
4465
4466
4467
4468 //
4469 // ------ Grid event classes
4470 //
4471
4472 IMPLEMENT_DYNAMIC_CLASS( wxGridEvent, wxEvent )
4473
4474 wxGridEvent::wxGridEvent( int id, wxEventType type, wxObject* obj,
4475 int row, int col, int x, int y,
4476 bool control, bool shift, bool alt, bool meta )
4477 : wxNotifyEvent( type, id )
4478 {
4479 m_row = row;
4480 m_col = col;
4481 m_x = x;
4482 m_y = y;
4483 m_control = control;
4484 m_shift = shift;
4485 m_alt = alt;
4486 m_meta = meta;
4487
4488 SetEventObject(obj);
4489 }
4490
4491
4492 IMPLEMENT_DYNAMIC_CLASS( wxGridSizeEvent, wxEvent )
4493
4494 wxGridSizeEvent::wxGridSizeEvent( int id, wxEventType type, wxObject* obj,
4495 int rowOrCol, int x, int y,
4496 bool control, bool shift, bool alt, bool meta )
4497 : wxNotifyEvent( type, id )
4498 {
4499 m_rowOrCol = rowOrCol;
4500 m_x = x;
4501 m_y = y;
4502 m_control = control;
4503 m_shift = shift;
4504 m_alt = alt;
4505 m_meta = meta;
4506
4507 SetEventObject(obj);
4508 }
4509
4510
4511 IMPLEMENT_DYNAMIC_CLASS( wxGridRangeSelectEvent, wxEvent )
4512
4513 wxGridRangeSelectEvent::wxGridRangeSelectEvent(int id, wxEventType type, wxObject* obj,
4514 const wxGridCellCoords& topLeft,
4515 const wxGridCellCoords& bottomRight,
4516 bool control, bool shift, bool alt, bool meta )
4517 : wxNotifyEvent( type, id )
4518 {
4519 m_topLeft = topLeft;
4520 m_bottomRight = bottomRight;
4521 m_control = control;
4522 m_shift = shift;
4523 m_alt = alt;
4524 m_meta = meta;
4525
4526 SetEventObject(obj);
4527 }
4528
4529
4530 #endif // ifndef wxUSE_NEW_GRID
4531