1 /////////////////////////////////////////////////////////////////////////// 
   2 // Name:        generic/grid.cpp 
   3 // Purpose:     wxGrid and related classes 
   4 // Author:      Michael Bedward (based on code by Julian Smart, Robin Dunn) 
   5 // Modified by: Robin Dunn, Vadim Zeitlin 
   8 // Copyright:   (c) Michael Bedward (mbedward@ozemail.com.au) 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  21     #pragma implementation "grid.h" 
  24 // For compilers that support precompilatixon, includes "wx/wx.h". 
  25 #include "wx/wxprec.h" 
  35 #if !defined(wxUSE_NEW_GRID) || !(wxUSE_NEW_GRID) 
  37 #else // wxUSE_NEW_GRID 
  41     #include "wx/dcclient.h" 
  42     #include "wx/settings.h" 
  44     #include "wx/textctrl.h" 
  45     #include "wx/checkbox.h" 
  46     #include "wx/combobox.h" 
  47     #include "wx/valtext.h" 
  50 #include "wx/textfile.h" 
  51 #include "wx/spinctrl.h" 
  52 #include "wx/tokenzr.h" 
  55 #include "wx/generic/gridsel.h" 
  57 #if defined(__WXMOTIF__) 
  58     #define WXUNUSED_MOTIF(identifier)  WXUNUSED(identifier) 
  60     #define WXUNUSED_MOTIF(identifier)  identifier 
  63 #if defined(__WXGTK__) 
  64     #define WXUNUSED_GTK(identifier)    WXUNUSED(identifier) 
  66     #define WXUNUSED_GTK(identifier)    identifier 
  69 // Required for wxIs... functions 
  72 // ---------------------------------------------------------------------------- 
  74 // ---------------------------------------------------------------------------- 
  76 WX_DEFINE_EXPORTED_ARRAY(wxGridCellAttr 
*, wxArrayAttrs
); 
  78 struct wxGridCellWithAttr
 
  80     wxGridCellWithAttr(int row
, int col
, wxGridCellAttr 
*attr_
) 
  81         : coords(row
, col
), attr(attr_
) 
  90     wxGridCellCoords coords
; 
  94 WX_DECLARE_EXPORTED_OBJARRAY(wxGridCellWithAttr
, wxGridCellWithAttrArray
); 
  96 #include "wx/arrimpl.cpp" 
  98 WX_DEFINE_OBJARRAY(wxGridCellCoordsArray
) 
  99 WX_DEFINE_OBJARRAY(wxGridCellWithAttrArray
) 
 101 // ---------------------------------------------------------------------------- 
 103 // ---------------------------------------------------------------------------- 
 105 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_LEFT_CLICK
) 
 106 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_RIGHT_CLICK
) 
 107 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_LEFT_DCLICK
) 
 108 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_RIGHT_DCLICK
) 
 109 DEFINE_EVENT_TYPE(wxEVT_GRID_LABEL_LEFT_CLICK
) 
 110 DEFINE_EVENT_TYPE(wxEVT_GRID_LABEL_RIGHT_CLICK
) 
 111 DEFINE_EVENT_TYPE(wxEVT_GRID_LABEL_LEFT_DCLICK
) 
 112 DEFINE_EVENT_TYPE(wxEVT_GRID_LABEL_RIGHT_DCLICK
) 
 113 DEFINE_EVENT_TYPE(wxEVT_GRID_ROW_SIZE
) 
 114 DEFINE_EVENT_TYPE(wxEVT_GRID_COL_SIZE
) 
 115 DEFINE_EVENT_TYPE(wxEVT_GRID_RANGE_SELECT
) 
 116 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_CHANGE
) 
 117 DEFINE_EVENT_TYPE(wxEVT_GRID_SELECT_CELL
) 
 118 DEFINE_EVENT_TYPE(wxEVT_GRID_EDITOR_SHOWN
) 
 119 DEFINE_EVENT_TYPE(wxEVT_GRID_EDITOR_HIDDEN
) 
 120 DEFINE_EVENT_TYPE(wxEVT_GRID_EDITOR_CREATED
) 
 122 // ---------------------------------------------------------------------------- 
 124 // ---------------------------------------------------------------------------- 
 126 class WXDLLEXPORT wxGridRowLabelWindow 
: public wxWindow
 
 129     wxGridRowLabelWindow() { m_owner 
= (wxGrid 
*)NULL
; } 
 130     wxGridRowLabelWindow( wxGrid 
*parent
, wxWindowID id
, 
 131                           const wxPoint 
&pos
, const wxSize 
&size 
); 
 136     void OnPaint( wxPaintEvent
& event 
); 
 137     void OnMouseEvent( wxMouseEvent
& event 
); 
 138     void OnMouseWheel( wxMouseEvent
& event 
); 
 139     void OnKeyDown( wxKeyEvent
& event 
); 
 140     void OnKeyUp( wxKeyEvent
& ); 
 142     DECLARE_DYNAMIC_CLASS(wxGridRowLabelWindow
) 
 143     DECLARE_EVENT_TABLE() 
 147 class WXDLLEXPORT wxGridColLabelWindow 
: public wxWindow
 
 150     wxGridColLabelWindow() { m_owner 
= (wxGrid 
*)NULL
; } 
 151     wxGridColLabelWindow( wxGrid 
*parent
, wxWindowID id
, 
 152                           const wxPoint 
&pos
, const wxSize 
&size 
); 
 157     void OnPaint( wxPaintEvent 
&event 
); 
 158     void OnMouseEvent( wxMouseEvent
& event 
); 
 159     void OnMouseWheel( wxMouseEvent
& event 
); 
 160     void OnKeyDown( wxKeyEvent
& event 
); 
 161     void OnKeyUp( wxKeyEvent
& ); 
 163     DECLARE_DYNAMIC_CLASS(wxGridColLabelWindow
) 
 164     DECLARE_EVENT_TABLE() 
 168 class WXDLLEXPORT wxGridCornerLabelWindow 
: public wxWindow
 
 171     wxGridCornerLabelWindow() { m_owner 
= (wxGrid 
*)NULL
; } 
 172     wxGridCornerLabelWindow( wxGrid 
*parent
, wxWindowID id
, 
 173                              const wxPoint 
&pos
, const wxSize 
&size 
); 
 178     void OnMouseEvent( wxMouseEvent
& event 
); 
 179     void OnMouseWheel( wxMouseEvent
& event 
); 
 180     void OnKeyDown( wxKeyEvent
& event 
); 
 181     void OnKeyUp( wxKeyEvent
& ); 
 182     void OnPaint( wxPaintEvent
& event 
); 
 184     DECLARE_DYNAMIC_CLASS(wxGridCornerLabelWindow
) 
 185     DECLARE_EVENT_TABLE() 
 188 class WXDLLEXPORT wxGridWindow 
: public wxWindow
 
 193         m_owner 
= (wxGrid 
*)NULL
; 
 194         m_rowLabelWin 
= (wxGridRowLabelWindow 
*)NULL
; 
 195         m_colLabelWin 
= (wxGridColLabelWindow 
*)NULL
; 
 198     wxGridWindow( wxGrid 
*parent
, 
 199                   wxGridRowLabelWindow 
*rowLblWin
, 
 200                   wxGridColLabelWindow 
*colLblWin
, 
 201                   wxWindowID id
, const wxPoint 
&pos
, const wxSize 
&size 
); 
 204     void ScrollWindow( int dx
, int dy
, const wxRect 
*rect 
); 
 208     wxGridRowLabelWindow     
*m_rowLabelWin
; 
 209     wxGridColLabelWindow     
*m_colLabelWin
; 
 211     void OnPaint( wxPaintEvent 
&event 
); 
 212     void OnMouseWheel( wxMouseEvent
& event 
); 
 213     void OnMouseEvent( wxMouseEvent
& event 
); 
 214     void OnKeyDown( wxKeyEvent
& ); 
 215     void OnKeyUp( wxKeyEvent
& ); 
 216     void OnEraseBackground( wxEraseEvent
& ); 
 219     DECLARE_DYNAMIC_CLASS(wxGridWindow
) 
 220     DECLARE_EVENT_TABLE() 
 225 class wxGridCellEditorEvtHandler 
: public wxEvtHandler
 
 228     wxGridCellEditorEvtHandler() 
 229         : m_grid(0), m_editor(0) 
 231     wxGridCellEditorEvtHandler(wxGrid
* grid
, wxGridCellEditor
* editor
) 
 232         : m_grid(grid
), m_editor(editor
) 
 235     void OnKeyDown(wxKeyEvent
& event
); 
 236     void OnChar(wxKeyEvent
& event
); 
 240     wxGridCellEditor
*   m_editor
; 
 241     DECLARE_DYNAMIC_CLASS(wxGridCellEditorEvtHandler
) 
 242     DECLARE_EVENT_TABLE() 
 246 IMPLEMENT_DYNAMIC_CLASS( wxGridCellEditorEvtHandler
, wxEvtHandler 
) 
 247 BEGIN_EVENT_TABLE( wxGridCellEditorEvtHandler
, wxEvtHandler 
) 
 248     EVT_KEY_DOWN( wxGridCellEditorEvtHandler::OnKeyDown 
) 
 249     EVT_CHAR( wxGridCellEditorEvtHandler::OnChar 
) 
 254 // ---------------------------------------------------------------------------- 
 255 // the internal data representation used by wxGridCellAttrProvider 
 256 // ---------------------------------------------------------------------------- 
 258 // this class stores attributes set for cells 
 259 class WXDLLEXPORT wxGridCellAttrData
 
 262     void SetAttr(wxGridCellAttr 
*attr
, int row
, int col
); 
 263     wxGridCellAttr 
*GetAttr(int row
, int col
) const; 
 264     void UpdateAttrRows( size_t pos
, int numRows 
); 
 265     void UpdateAttrCols( size_t pos
, int numCols 
); 
 268     // searches for the attr for given cell, returns wxNOT_FOUND if not found 
 269     int FindIndex(int row
, int col
) const; 
 271     wxGridCellWithAttrArray m_attrs
; 
 274 // this class stores attributes set for rows or columns 
 275 class WXDLLEXPORT wxGridRowOrColAttrData
 
 278     // empty ctor to suppress warnings 
 279     wxGridRowOrColAttrData() { } 
 280     ~wxGridRowOrColAttrData(); 
 282     void SetAttr(wxGridCellAttr 
*attr
, int rowOrCol
); 
 283     wxGridCellAttr 
*GetAttr(int rowOrCol
) const; 
 284     void UpdateAttrRowsOrCols( size_t pos
, int numRowsOrCols 
); 
 287     wxArrayInt m_rowsOrCols
; 
 288     wxArrayAttrs m_attrs
; 
 291 // NB: this is just a wrapper around 3 objects: one which stores cell 
 292 //     attributes, and 2 others for row/col ones 
 293 class WXDLLEXPORT wxGridCellAttrProviderData
 
 296     wxGridCellAttrData m_cellAttrs
; 
 297     wxGridRowOrColAttrData m_rowAttrs
, 
 302 // ---------------------------------------------------------------------------- 
 303 // data structures used for the data type registry 
 304 // ---------------------------------------------------------------------------- 
 306 struct wxGridDataTypeInfo
 
 308     wxGridDataTypeInfo(const wxString
& typeName
, 
 309                        wxGridCellRenderer
* renderer
, 
 310                        wxGridCellEditor
* editor
) 
 311         : m_typeName(typeName
), m_renderer(renderer
), m_editor(editor
) 
 314     ~wxGridDataTypeInfo() 
 316         wxSafeDecRef(m_renderer
); 
 317         wxSafeDecRef(m_editor
); 
 321     wxGridCellRenderer
* m_renderer
; 
 322     wxGridCellEditor
*   m_editor
; 
 326 WX_DEFINE_EXPORTED_ARRAY(wxGridDataTypeInfo
*, wxGridDataTypeInfoArray
); 
 329 class WXDLLEXPORT wxGridTypeRegistry
 
 332   wxGridTypeRegistry() {} 
 333     ~wxGridTypeRegistry(); 
 335     void RegisterDataType(const wxString
& typeName
, 
 336                      wxGridCellRenderer
* renderer
, 
 337                      wxGridCellEditor
* editor
); 
 339     // find one of already registered data types 
 340     int FindRegisteredDataType(const wxString
& typeName
); 
 342     // try to FindRegisteredDataType(), if this fails and typeName is one of 
 343     // standard typenames, register it and return its index 
 344     int FindDataType(const wxString
& typeName
); 
 346     // try to FindDataType(), if it fails see if it is not one of already 
 347     // registered data types with some params in which case clone the 
 348     // registered data type and set params for it 
 349     int FindOrCloneDataType(const wxString
& typeName
); 
 351     wxGridCellRenderer
* GetRenderer(int index
); 
 352     wxGridCellEditor
*   GetEditor(int index
); 
 355     wxGridDataTypeInfoArray m_typeinfo
; 
 358 // ---------------------------------------------------------------------------- 
 359 // conditional compilation 
 360 // ---------------------------------------------------------------------------- 
 362 #ifndef WXGRID_DRAW_LINES 
 363 #define WXGRID_DRAW_LINES 1 
 366 // ---------------------------------------------------------------------------- 
 368 // ---------------------------------------------------------------------------- 
 370 //#define DEBUG_ATTR_CACHE 
 371 #ifdef DEBUG_ATTR_CACHE 
 372     static size_t gs_nAttrCacheHits 
= 0; 
 373     static size_t gs_nAttrCacheMisses 
= 0; 
 374 #endif // DEBUG_ATTR_CACHE 
 376 // ---------------------------------------------------------------------------- 
 378 // ---------------------------------------------------------------------------- 
 380 wxGridCellCoords 
wxGridNoCellCoords( -1, -1 ); 
 381 wxRect           
wxGridNoCellRect( -1, -1, -1, -1 ); 
 384 // TODO: this doesn't work at all, grid cells have different sizes and approx 
 385 //       calculations don't work as because of the size mismatch scrollbars 
 386 //       sometimes fail to be shown when they should be or vice versa 
 388 //       The scroll bars may be a little flakey once in a while, but that is 
 389 //       surely much less horrible than having scroll lines of only 1!!! 
 392 //       Well, it's still seriously broken so it might be better but needs 
 395 static const size_t GRID_SCROLL_LINE_X 
= 15;  // 1; 
 396 static const size_t GRID_SCROLL_LINE_Y 
= GRID_SCROLL_LINE_X
; 
 398 // the size of hash tables used a bit everywhere (the max number of elements 
 399 // in these hash tables is the number of rows/columns) 
 400 static const int GRID_HASH_SIZE 
= 100; 
 402 // ---------------------------------------------------------------------------- 
 404 // ---------------------------------------------------------------------------- 
 406 static inline int GetScrollX(int x
) 
 408     return (x 
+ GRID_SCROLL_LINE_X 
- 1) / GRID_SCROLL_LINE_X
; 
 411 static inline int GetScrollY(int y
) 
 413     return (y 
+ GRID_SCROLL_LINE_Y 
- 1) / GRID_SCROLL_LINE_Y
; 
 416 // ============================================================================ 
 418 // ============================================================================ 
 420 // ---------------------------------------------------------------------------- 
 422 // ---------------------------------------------------------------------------- 
 424 wxGridCellEditor::wxGridCellEditor() 
 430 wxGridCellEditor::~wxGridCellEditor() 
 435 void wxGridCellEditor::Create(wxWindow
* WXUNUSED(parent
), 
 436                               wxWindowID 
WXUNUSED(id
), 
 437                               wxEvtHandler
* evtHandler
) 
 440         m_control
->PushEventHandler(evtHandler
); 
 443 void wxGridCellEditor::PaintBackground(const wxRect
& rectCell
, 
 444                                        wxGridCellAttr 
*attr
) 
 446     // erase the background because we might not fill the cell 
 447     wxClientDC 
dc(m_control
->GetParent()); 
 448     dc
.SetPen(*wxTRANSPARENT_PEN
); 
 449     dc
.SetBrush(wxBrush(attr
->GetBackgroundColour(), wxSOLID
)); 
 450     dc
.DrawRectangle(rectCell
); 
 452     // redraw the control we just painted over 
 453     m_control
->Refresh(); 
 456 void wxGridCellEditor::Destroy() 
 460         m_control
->PopEventHandler(TRUE 
/* delete it*/); 
 462         m_control
->Destroy(); 
 467 void wxGridCellEditor::Show(bool show
, wxGridCellAttr 
*attr
) 
 469     wxASSERT_MSG(m_control
, 
 470                  wxT("The wxGridCellEditor must be Created first!")); 
 471     m_control
->Show(show
); 
 475         // set the colours/fonts if we have any 
 478             m_colFgOld 
= m_control
->GetForegroundColour(); 
 479             m_control
->SetForegroundColour(attr
->GetTextColour()); 
 481             m_colBgOld 
= m_control
->GetBackgroundColour(); 
 482             m_control
->SetBackgroundColour(attr
->GetBackgroundColour()); 
 484             m_fontOld 
= m_control
->GetFont(); 
 485             m_control
->SetFont(attr
->GetFont()); 
 487             // can't do anything more in the base class version, the other 
 488             // attributes may only be used by the derived classes 
 493         // restore the standard colours fonts 
 494         if ( m_colFgOld
.Ok() ) 
 496             m_control
->SetForegroundColour(m_colFgOld
); 
 497             m_colFgOld 
= wxNullColour
; 
 500         if ( m_colBgOld
.Ok() ) 
 502             m_control
->SetBackgroundColour(m_colBgOld
); 
 503             m_colBgOld 
= wxNullColour
; 
 506         if ( m_fontOld
.Ok() ) 
 508             m_control
->SetFont(m_fontOld
); 
 509             m_fontOld 
= wxNullFont
; 
 514 void wxGridCellEditor::SetSize(const wxRect
& rect
) 
 516     wxASSERT_MSG(m_control
, 
 517                  wxT("The wxGridCellEditor must be Created first!")); 
 518     m_control
->SetSize(rect
, wxSIZE_ALLOW_MINUS_ONE
); 
 521 void wxGridCellEditor::HandleReturn(wxKeyEvent
& event
) 
 526 bool wxGridCellEditor::IsAcceptedKey(wxKeyEvent
& event
) 
 528     // accept the simple key presses, not anything with Ctrl/Alt/Meta 
 529     return !(event
.ControlDown() || event
.AltDown()); 
 532 void wxGridCellEditor::StartingKey(wxKeyEvent
& event
) 
 537 void wxGridCellEditor::StartingClick() 
 543 // ---------------------------------------------------------------------------- 
 544 // wxGridCellTextEditor 
 545 // ---------------------------------------------------------------------------- 
 547 wxGridCellTextEditor::wxGridCellTextEditor() 
 552 void wxGridCellTextEditor::Create(wxWindow
* parent
, 
 554                                   wxEvtHandler
* evtHandler
) 
 556     m_control 
= new wxTextCtrl(parent
, id
, wxEmptyString
, 
 557                                wxDefaultPosition
, wxDefaultSize
 
 558 #if defined(__WXMSW__) 
 559                                , wxTE_PROCESS_TAB 
| wxTE_MULTILINE 
| 
 560                                  wxTE_NO_VSCROLL 
| wxTE_AUTO_SCROLL
 
 564     // TODO: use m_maxChars 
 566     wxGridCellEditor::Create(parent
, id
, evtHandler
); 
 569 void wxGridCellTextEditor::PaintBackground(const wxRect
& WXUNUSED(rectCell
), 
 570                                            wxGridCellAttr 
* WXUNUSED(attr
)) 
 572     // as we fill the entire client area, don't do anything here to minimize 
 576 void wxGridCellTextEditor::SetSize(const wxRect
& rectOrig
) 
 578     wxRect 
rect(rectOrig
); 
 580     // Make the edit control large enough to allow for internal 
 583     // TODO: remove this if the text ctrl sizing is improved esp. for 
 586 #if defined(__WXGTK__) 
 595     int extra_x 
= ( rect
.x 
> 2 )? 2 : 1; 
 597 // MB: treat MSW separately here otherwise the caret doesn't show 
 598 // when the editor is in the first row. 
 599 #if defined(__WXMSW__) 
 602     int extra_y 
= ( rect
.y 
> 2 )? 2 : 1; 
 605 #if defined(__WXMOTIF__) 
 609     rect
.SetLeft( wxMax(0, rect
.x 
- extra_x
) ); 
 610     rect
.SetTop( wxMax(0, rect
.y 
- extra_y
) ); 
 611     rect
.SetRight( rect
.GetRight() + 2*extra_x 
); 
 612     rect
.SetBottom( rect
.GetBottom() + 2*extra_y 
); 
 615     wxGridCellEditor::SetSize(rect
); 
 618 void wxGridCellTextEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
 620     wxASSERT_MSG(m_control
, 
 621                  wxT("The wxGridCellEditor must be Created first!")); 
 623     m_startValue 
= grid
->GetTable()->GetValue(row
, col
); 
 625     DoBeginEdit(m_startValue
); 
 628 void wxGridCellTextEditor::DoBeginEdit(const wxString
& startValue
) 
 630     Text()->SetValue(startValue
); 
 631     Text()->SetInsertionPointEnd(); 
 632     Text()->SetSelection(-1,-1); 
 636 bool wxGridCellTextEditor::EndEdit(int row
, int col
, 
 639     wxASSERT_MSG(m_control
, 
 640                  wxT("The wxGridCellEditor must be Created first!")); 
 642     bool changed 
= FALSE
; 
 643     wxString value 
= Text()->GetValue(); 
 644     if (value 
!= m_startValue
) 
 648         grid
->GetTable()->SetValue(row
, col
, value
); 
 650     m_startValue 
= wxEmptyString
; 
 651     Text()->SetValue(m_startValue
); 
 657 void wxGridCellTextEditor::Reset() 
 659     wxASSERT_MSG(m_control
, 
 660                  wxT("The wxGridCellEditor must be Created first!")); 
 662     DoReset(m_startValue
); 
 665 void wxGridCellTextEditor::DoReset(const wxString
& startValue
) 
 667     Text()->SetValue(startValue
); 
 668     Text()->SetInsertionPointEnd(); 
 671 bool wxGridCellTextEditor::IsAcceptedKey(wxKeyEvent
& event
) 
 673     if ( wxGridCellEditor::IsAcceptedKey(event
) ) 
 675         int keycode 
= event
.GetKeyCode(); 
 689             case WXK_NUMPAD_MULTIPLY
: 
 693             case WXK_NUMPAD_SUBTRACT
: 
 695             case WXK_NUMPAD_DECIMAL
: 
 697             case WXK_NUMPAD_DIVIDE
: 
 701                 // accept 8 bit chars too if isprint() agrees 
 702                 if ( (keycode 
< 255) && (isprint(keycode
)) ) 
 710 void wxGridCellTextEditor::StartingKey(wxKeyEvent
& event
) 
 712     if ( !Text()->EmulateKeyPress(event
) ) 
 718 void wxGridCellTextEditor::HandleReturn( wxKeyEvent
& 
 719                                          WXUNUSED_GTK(WXUNUSED_MOTIF(event
)) ) 
 721 #if defined(__WXMOTIF__) || defined(__WXGTK__) 
 722     // wxMotif needs a little extra help... 
 723     size_t pos 
= (size_t)( Text()->GetInsertionPoint() ); 
 724     wxString 
s( Text()->GetValue() ); 
 725     s 
= s
.Left(pos
) + wxT("\n") + s
.Mid(pos
); 
 727     Text()->SetInsertionPoint( pos 
); 
 729     // the other ports can handle a Return key press 
 735 void wxGridCellTextEditor::SetParameters(const wxString
& params
) 
 745         if ( !params
.ToLong(&tmp
) ) 
 747             wxLogDebug(_T("Invalid wxGridCellTextEditor parameter string '%s' ignored"), params
.c_str()); 
 751             m_maxChars 
= (size_t)tmp
; 
 756 // ---------------------------------------------------------------------------- 
 757 // wxGridCellNumberEditor 
 758 // ---------------------------------------------------------------------------- 
 760 wxGridCellNumberEditor::wxGridCellNumberEditor(int min
, int max
) 
 766 void wxGridCellNumberEditor::Create(wxWindow
* parent
, 
 768                                     wxEvtHandler
* evtHandler
) 
 772         // create a spin ctrl 
 773         m_control 
= new wxSpinCtrl(parent
, -1, wxEmptyString
, 
 774                                    wxDefaultPosition
, wxDefaultSize
, 
 778         wxGridCellEditor::Create(parent
, id
, evtHandler
); 
 782         // just a text control 
 783         wxGridCellTextEditor::Create(parent
, id
, evtHandler
); 
 786         Text()->SetValidator(wxTextValidator(wxFILTER_NUMERIC
)); 
 787 #endif // wxUSE_VALIDATORS 
 791 void wxGridCellNumberEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
 793     // first get the value 
 794     wxGridTableBase 
*table 
= grid
->GetTable(); 
 795     if ( table
->CanGetValueAs(row
, col
, wxGRID_VALUE_NUMBER
) ) 
 797         m_valueOld 
= table
->GetValueAsLong(row
, col
); 
 802         wxString sValue 
= table
->GetValue(row
, col
); 
 803         if (! sValue
.ToLong(&m_valueOld
) && ! sValue
.IsEmpty()) 
 805             wxFAIL_MSG( _T("this cell doesn't have numeric value") ); 
 812         Spin()->SetValue((int)m_valueOld
); 
 817         DoBeginEdit(GetString()); 
 821 bool wxGridCellNumberEditor::EndEdit(int row
, int col
, 
 830         value 
= Spin()->GetValue(); 
 831         changed 
= value 
!= m_valueOld
; 
 833             text 
= wxString::Format(wxT("%ld"), value
); 
 837         text 
= Text()->GetValue(); 
 838         changed 
= (text
.IsEmpty() || text
.ToLong(&value
)) && (value 
!= m_valueOld
); 
 843         if (grid
->GetTable()->CanSetValueAs(row
, col
, wxGRID_VALUE_NUMBER
)) 
 844             grid
->GetTable()->SetValueAsLong(row
, col
, value
); 
 846             grid
->GetTable()->SetValue(row
, col
, text
); 
 852 void wxGridCellNumberEditor::Reset() 
 856         Spin()->SetValue((int)m_valueOld
); 
 860         DoReset(GetString()); 
 864 bool wxGridCellNumberEditor::IsAcceptedKey(wxKeyEvent
& event
) 
 866     if ( wxGridCellEditor::IsAcceptedKey(event
) ) 
 868         int keycode 
= event
.GetKeyCode(); 
 884             case WXK_NUMPAD_SUBTRACT
: 
 890                 if ( (keycode 
< 128) && isdigit(keycode
) ) 
 898 void wxGridCellNumberEditor::StartingKey(wxKeyEvent
& event
) 
 902         int keycode 
= (int) event
.KeyCode(); 
 903         if ( isdigit(keycode
) || keycode 
== '+' || keycode 
== '-' 
 904             || keycode 
==  WXK_NUMPAD0
 
 905             || keycode 
==  WXK_NUMPAD1
 
 906             || keycode 
==  WXK_NUMPAD2
 
 907             || keycode 
==  WXK_NUMPAD3
 
 908             || keycode 
==  WXK_NUMPAD4
 
 909             || keycode 
==  WXK_NUMPAD5
 
 910             || keycode 
==  WXK_NUMPAD6
 
 911             || keycode 
==  WXK_NUMPAD7
 
 912             || keycode 
==  WXK_NUMPAD8
 
 913             || keycode 
==  WXK_NUMPAD9
 
 914             || keycode 
==  WXK_ADD
 
 915             || keycode 
==  WXK_NUMPAD_ADD
 
 916             || keycode 
==  WXK_SUBTRACT
 
 917             || keycode 
==  WXK_NUMPAD_SUBTRACT
) 
 919             wxGridCellTextEditor::StartingKey(event
); 
 929 void wxGridCellNumberEditor::SetParameters(const wxString
& params
) 
 940         if ( params
.BeforeFirst(_T(',')).ToLong(&tmp
) ) 
 944             if ( params
.AfterFirst(_T(',')).ToLong(&tmp
) ) 
 948                 // skip the error message below 
 953         wxLogDebug(_T("Invalid wxGridCellNumberEditor parameter string '%s' ignored"), params
.c_str()); 
 957 // ---------------------------------------------------------------------------- 
 958 // wxGridCellFloatEditor 
 959 // ---------------------------------------------------------------------------- 
 961 wxGridCellFloatEditor::wxGridCellFloatEditor(int width
, int precision
) 
 964     m_precision 
= precision
; 
 967 void wxGridCellFloatEditor::Create(wxWindow
* parent
, 
 969                                    wxEvtHandler
* evtHandler
) 
 971     wxGridCellTextEditor::Create(parent
, id
, evtHandler
); 
 974     Text()->SetValidator(wxTextValidator(wxFILTER_NUMERIC
)); 
 975 #endif // wxUSE_VALIDATORS 
 978 void wxGridCellFloatEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
 980     // first get the value 
 981     wxGridTableBase 
*table 
= grid
->GetTable(); 
 982     if ( table
->CanGetValueAs(row
, col
, wxGRID_VALUE_FLOAT
) ) 
 984         m_valueOld 
= table
->GetValueAsDouble(row
, col
); 
 989         wxString sValue 
= table
->GetValue(row
, col
); 
 990         if (! sValue
.ToDouble(&m_valueOld
) && ! sValue
.IsEmpty()) 
 992             wxFAIL_MSG( _T("this cell doesn't have float value") ); 
 997     DoBeginEdit(GetString()); 
1000 bool wxGridCellFloatEditor::EndEdit(int row
, int col
, 
1004     wxString 
text(Text()->GetValue()); 
1006     if ( (text
.IsEmpty() || text
.ToDouble(&value
)) && (value 
!= m_valueOld
) ) 
1008         if (grid
->GetTable()->CanSetValueAs(row
, col
, wxGRID_VALUE_FLOAT
)) 
1009             grid
->GetTable()->SetValueAsDouble(row
, col
, value
); 
1011             grid
->GetTable()->SetValue(row
, col
, text
); 
1018 void wxGridCellFloatEditor::Reset() 
1020     DoReset(GetString()); 
1023 void wxGridCellFloatEditor::StartingKey(wxKeyEvent
& event
) 
1025     int keycode 
= (int)event
.KeyCode(); 
1026         if ( isdigit(keycode
) || keycode 
== '+' || keycode 
== '-' || keycode 
== '.' 
1027             || keycode 
==  WXK_NUMPAD0
 
1028             || keycode 
==  WXK_NUMPAD1
 
1029             || keycode 
==  WXK_NUMPAD2
 
1030             || keycode 
==  WXK_NUMPAD3
 
1031             || keycode 
==  WXK_NUMPAD4
 
1032             || keycode 
==  WXK_NUMPAD5
 
1033             || keycode 
==  WXK_NUMPAD6
 
1034             || keycode 
==  WXK_NUMPAD7
 
1035             || keycode 
==  WXK_NUMPAD8
 
1036             || keycode 
==  WXK_NUMPAD9
 
1037             || keycode 
==  WXK_ADD
 
1038             || keycode 
==  WXK_NUMPAD_ADD
 
1039             || keycode 
==  WXK_SUBTRACT
 
1040             || keycode 
==  WXK_NUMPAD_SUBTRACT
) 
1042         wxGridCellTextEditor::StartingKey(event
); 
1044         // skip Skip() below 
1051 void wxGridCellFloatEditor::SetParameters(const wxString
& params
) 
1062         if ( params
.BeforeFirst(_T(',')).ToLong(&tmp
) ) 
1066             if ( params
.AfterFirst(_T(',')).ToLong(&tmp
) ) 
1068                 m_precision 
= (int)tmp
; 
1070                 // skip the error message below 
1075         wxLogDebug(_T("Invalid wxGridCellFloatEditor parameter string '%s' ignored"), params
.c_str()); 
1079 wxString 
wxGridCellFloatEditor::GetString() const 
1082     if ( m_width 
== -1 ) 
1084         // default width/precision 
1087     else if ( m_precision 
== -1 ) 
1089         // default precision 
1090         fmt
.Printf(_T("%%%d.f"), m_width
); 
1094         fmt
.Printf(_T("%%%d.%df"), m_width
, m_precision
); 
1097     return wxString::Format(fmt
, m_valueOld
); 
1100 bool wxGridCellFloatEditor::IsAcceptedKey(wxKeyEvent
& event
) 
1102     if ( wxGridCellEditor::IsAcceptedKey(event
) ) 
1104         int keycode 
= event
.GetKeyCode(); 
1118             case WXK_NUMPAD_ADD
: 
1120             case WXK_NUMPAD_SUBTRACT
: 
1122             case WXK_NUMPAD_DECIMAL
: 
1126                 // additionally accept 'e' as in '1e+6' 
1127                 if ( (keycode 
< 128) && 
1128                      (isdigit(keycode
) || tolower(keycode
) == 'e') ) 
1136 #endif // wxUSE_TEXTCTRL 
1140 // ---------------------------------------------------------------------------- 
1141 // wxGridCellBoolEditor 
1142 // ---------------------------------------------------------------------------- 
1144 void wxGridCellBoolEditor::Create(wxWindow
* parent
, 
1146                                   wxEvtHandler
* evtHandler
) 
1148     m_control 
= new wxCheckBox(parent
, id
, wxEmptyString
, 
1149                                wxDefaultPosition
, wxDefaultSize
, 
1152     wxGridCellEditor::Create(parent
, id
, evtHandler
); 
1155 void wxGridCellBoolEditor::SetSize(const wxRect
& r
) 
1157     bool resize 
= FALSE
; 
1158     wxSize size 
= m_control
->GetSize(); 
1159     wxCoord minSize 
= wxMin(r
.width
, r
.height
); 
1161     // check if the checkbox is not too big/small for this cell 
1162     wxSize sizeBest 
= m_control
->GetBestSize(); 
1163     if ( !(size 
== sizeBest
) ) 
1165         // reset to default size if it had been made smaller 
1171     if ( size
.x 
>= minSize 
|| size
.y 
>= minSize 
) 
1173         // leave 1 pixel margin 
1174         size
.x 
= size
.y 
= minSize 
- 2; 
1181         m_control
->SetSize(size
); 
1184     // position it in the centre of the rectangle (TODO: support alignment?) 
1186 #if defined(__WXGTK__) || defined (__WXMOTIF__) 
1187     // the checkbox without label still has some space to the right in wxGTK, 
1188     // so shift it to the right 
1190 #elif defined(__WXMSW__) 
1191     // here too, but in other way 
1196     m_control
->Move(r
.x 
+ r
.width
/2 - size
.x
/2, r
.y 
+ r
.height
/2 - size
.y
/2); 
1199 void wxGridCellBoolEditor::Show(bool show
, wxGridCellAttr 
*attr
) 
1201     m_control
->Show(show
); 
1205         wxColour colBg 
= attr 
? attr
->GetBackgroundColour() : *wxLIGHT_GREY
; 
1206         CBox()->SetBackgroundColour(colBg
); 
1210 void wxGridCellBoolEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
1212     wxASSERT_MSG(m_control
, 
1213                  wxT("The wxGridCellEditor must be Created first!")); 
1215     if (grid
->GetTable()->CanGetValueAs(row
, col
, wxGRID_VALUE_BOOL
)) 
1216         m_startValue 
= grid
->GetTable()->GetValueAsBool(row
, col
); 
1219         wxString 
cellval( grid
->GetTable()->GetValue(row
, col
) ); 
1220         m_startValue 
= !( !cellval 
|| (cellval 
== wxT("0")) ); 
1222     CBox()->SetValue(m_startValue
); 
1226 bool wxGridCellBoolEditor::EndEdit(int row
, int col
, 
1229     wxASSERT_MSG(m_control
, 
1230                  wxT("The wxGridCellEditor must be Created first!")); 
1232     bool changed 
= FALSE
; 
1233     bool value 
= CBox()->GetValue(); 
1234     if ( value 
!= m_startValue 
) 
1239         if (grid
->GetTable()->CanGetValueAs(row
, col
, wxGRID_VALUE_BOOL
)) 
1240             grid
->GetTable()->SetValueAsBool(row
, col
, value
); 
1242             grid
->GetTable()->SetValue(row
, col
, value 
? _T("1") : wxEmptyString
); 
1248 void wxGridCellBoolEditor::Reset() 
1250     wxASSERT_MSG(m_control
, 
1251                  wxT("The wxGridCellEditor must be Created first!")); 
1253     CBox()->SetValue(m_startValue
); 
1256 void wxGridCellBoolEditor::StartingClick() 
1258     CBox()->SetValue(!CBox()->GetValue()); 
1261 bool wxGridCellBoolEditor::IsAcceptedKey(wxKeyEvent
& event
) 
1263     if ( wxGridCellEditor::IsAcceptedKey(event
) ) 
1265         int keycode 
= event
.GetKeyCode(); 
1269             case WXK_NUMPAD_MULTIPLY
: 
1271             case WXK_NUMPAD_ADD
: 
1273             case WXK_NUMPAD_SUBTRACT
: 
1284 #endif // wxUSE_CHECKBOX 
1288 // ---------------------------------------------------------------------------- 
1289 // wxGridCellChoiceEditor 
1290 // ---------------------------------------------------------------------------- 
1292 wxGridCellChoiceEditor::wxGridCellChoiceEditor(size_t count
, 
1293                                                const wxString choices
[], 
1295                       : m_allowOthers(allowOthers
) 
1299         m_choices
.Alloc(count
); 
1300         for ( size_t n 
= 0; n 
< count
; n
++ ) 
1302             m_choices
.Add(choices
[n
]); 
1307 wxGridCellEditor 
*wxGridCellChoiceEditor::Clone() const 
1309     wxGridCellChoiceEditor 
*editor 
= new wxGridCellChoiceEditor
; 
1310     editor
->m_allowOthers 
= m_allowOthers
; 
1311     editor
->m_choices 
= m_choices
; 
1316 void wxGridCellChoiceEditor::Create(wxWindow
* parent
, 
1318                                     wxEvtHandler
* evtHandler
) 
1320     size_t count 
= m_choices
.GetCount(); 
1321     wxString 
*choices 
= new wxString
[count
]; 
1322     for ( size_t n 
= 0; n 
< count
; n
++ ) 
1324         choices
[n
] = m_choices
[n
]; 
1327     m_control 
= new wxComboBox(parent
, id
, wxEmptyString
, 
1328                                wxDefaultPosition
, wxDefaultSize
, 
1330                                m_allowOthers 
? 0 : wxCB_READONLY
); 
1334     wxGridCellEditor::Create(parent
, id
, evtHandler
); 
1337 void wxGridCellChoiceEditor::PaintBackground(const wxRect
& rectCell
, 
1338                                              wxGridCellAttr 
* attr
) 
1340     // as we fill the entire client area, don't do anything here to minimize 
1343     // TODO: It doesn't actually fill the client area since the height of a 
1344     // combo always defaults to the standard...  Until someone has time to 
1345     // figure out the right rectangle to paint, just do it the normal way... 
1346     wxGridCellEditor::PaintBackground(rectCell
, attr
); 
1349 void wxGridCellChoiceEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
1351     wxASSERT_MSG(m_control
, 
1352                  wxT("The wxGridCellEditor must be Created first!")); 
1354     m_startValue 
= grid
->GetTable()->GetValue(row
, col
); 
1357         Combo()->SetValue(m_startValue
); 
1360         // find the right position, or default to the first if not found 
1361         int pos 
= Combo()->FindString(m_startValue
); 
1364         Combo()->SetSelection(pos
); 
1366     Combo()->SetInsertionPointEnd(); 
1367     Combo()->SetFocus(); 
1370 bool wxGridCellChoiceEditor::EndEdit(int row
, int col
, 
1373     wxString value 
= Combo()->GetValue(); 
1374     bool changed 
= value 
!= m_startValue
; 
1377         grid
->GetTable()->SetValue(row
, col
, value
); 
1379     m_startValue 
= wxEmptyString
; 
1381         Combo()->SetValue(m_startValue
); 
1383         Combo()->SetSelection(0); 
1388 void wxGridCellChoiceEditor::Reset() 
1390     Combo()->SetValue(m_startValue
); 
1391     Combo()->SetInsertionPointEnd(); 
1394 void wxGridCellChoiceEditor::SetParameters(const wxString
& params
) 
1404     wxStringTokenizer 
tk(params
, _T(',')); 
1405     while ( tk
.HasMoreTokens() ) 
1407         m_choices
.Add(tk
.GetNextToken()); 
1411 #endif // wxUSE_COMBOBOX 
1413 // ---------------------------------------------------------------------------- 
1414 // wxGridCellEditorEvtHandler 
1415 // ---------------------------------------------------------------------------- 
1417 void wxGridCellEditorEvtHandler::OnKeyDown(wxKeyEvent
& event
) 
1419     switch ( event
.KeyCode() ) 
1423             m_grid
->DisableCellEditControl(); 
1427             m_grid
->GetEventHandler()->ProcessEvent( event 
); 
1431         case WXK_NUMPAD_ENTER
: 
1432             if (!m_grid
->GetEventHandler()->ProcessEvent(event
)) 
1433                 m_editor
->HandleReturn(event
); 
1442 void wxGridCellEditorEvtHandler::OnChar(wxKeyEvent
& event
) 
1444     switch ( event
.KeyCode() ) 
1449         case WXK_NUMPAD_ENTER
: 
1457 // ---------------------------------------------------------------------------- 
1458 // wxGridCellWorker is an (almost) empty common base class for 
1459 // wxGridCellRenderer and wxGridCellEditor managing ref counting 
1460 // ---------------------------------------------------------------------------- 
1462 void wxGridCellWorker::SetParameters(const wxString
& WXUNUSED(params
)) 
1467 wxGridCellWorker::~wxGridCellWorker() 
1471 // ============================================================================ 
1473 // ============================================================================ 
1475 // ---------------------------------------------------------------------------- 
1476 // wxGridCellRenderer 
1477 // ---------------------------------------------------------------------------- 
1479 void wxGridCellRenderer::Draw(wxGrid
& grid
, 
1480                               wxGridCellAttr
& attr
, 
1483                               int WXUNUSED(row
), int WXUNUSED(col
), 
1486     dc
.SetBackgroundMode( wxSOLID 
); 
1490         dc
.SetBrush( wxBrush(grid
.GetSelectionBackground(), wxSOLID
) ); 
1494         dc
.SetBrush( wxBrush(attr
.GetBackgroundColour(), wxSOLID
) ); 
1497     dc
.SetPen( *wxTRANSPARENT_PEN 
); 
1498     dc
.DrawRectangle(rect
); 
1501 // ---------------------------------------------------------------------------- 
1502 // wxGridCellStringRenderer 
1503 // ---------------------------------------------------------------------------- 
1505 void wxGridCellStringRenderer::SetTextColoursAndFont(wxGrid
& grid
, 
1506                                                      wxGridCellAttr
& attr
, 
1510     dc
.SetBackgroundMode( wxTRANSPARENT 
); 
1512     // TODO some special colours for attr.IsReadOnly() case? 
1516         dc
.SetTextBackground( grid
.GetSelectionBackground() ); 
1517         dc
.SetTextForeground( grid
.GetSelectionForeground() ); 
1521         dc
.SetTextBackground( attr
.GetBackgroundColour() ); 
1522         dc
.SetTextForeground( attr
.GetTextColour() ); 
1525     dc
.SetFont( attr
.GetFont() ); 
1528 wxSize 
wxGridCellStringRenderer::DoGetBestSize(wxGridCellAttr
& attr
, 
1530                                                const wxString
& text
) 
1532     wxCoord x 
= 0, y 
= 0, max_x 
= 0; 
1533     dc
.SetFont(attr
.GetFont()); 
1534     wxStringTokenizer 
tk(text
, _T('\n')); 
1535     while ( tk
.HasMoreTokens() ) 
1537         dc
.GetTextExtent(tk
.GetNextToken(), &x
, &y
); 
1538         max_x 
= wxMax(max_x
, x
); 
1541     y 
*= 1 + text
.Freq(wxT('\n')); // multiply by the number of lines. 
1543     return wxSize(max_x
, y
); 
1546 wxSize 
wxGridCellStringRenderer::GetBestSize(wxGrid
& grid
, 
1547                                              wxGridCellAttr
& attr
, 
1551     return DoGetBestSize(attr
, dc
, grid
.GetCellValue(row
, col
)); 
1554 void wxGridCellStringRenderer::Draw(wxGrid
& grid
, 
1555                                     wxGridCellAttr
& attr
, 
1557                                     const wxRect
& rectCell
, 
1561     wxRect rect 
= rectCell
; 
1564     // erase only this cells background, overflow cells should have been erased 
1565     wxGridCellRenderer::Draw(grid
, attr
, dc
, rectCell
, row
, col
, isSelected
); 
1568     attr
.GetAlignment(&hAlign
, &vAlign
); 
1570     int overflowCols 
= 0; 
1572     if (attr
.GetOverflow()) 
1574         int cols 
= grid
.GetNumberCols(); 
1575         int best_width 
= GetBestSize(grid
,attr
,dc
,row
,col
).GetWidth(); 
1576         int cell_rows
, cell_cols
; 
1577         attr
.GetSize( &cell_rows
, &cell_cols 
); // shouldn't get here if <=0 
1578         if ((best_width 
> rectCell
.width
) && (col 
< cols
) && grid
.GetTable()) 
1580             int i
, c_cols
, c_rows
; 
1581             for (i 
= col
+cell_cols
; i 
< cols
; i
++) 
1583                 bool is_empty 
= TRUE
; 
1584                 for (int j
=row
; j
<row
+cell_rows
; j
++) 
1586                     // check w/ anchor cell for multicell block 
1587                     grid
.GetCellSize(j
, i
, &c_rows
, &c_cols
); 
1588                     if (c_rows 
> 0) c_rows 
= 0; 
1589                     if (!grid
.GetTable()->IsEmptyCell(j
+c_rows
, i
)) 
1596                     rect
.width 
+= grid
.GetColSize(i
); 
1602                 if (rect
.width 
>= best_width
) break; 
1604             overflowCols 
= i 
- col 
- cell_cols 
+ 1; 
1605             if (overflowCols 
>= cols
) overflowCols 
= cols 
- 1; 
1608         if (overflowCols 
> 0) // redraw overflow cells w/ proper hilight 
1610             hAlign 
= wxALIGN_LEFT
; // if oveflowed then it's left aligned 
1612             clip
.x 
+= rectCell
.width
; 
1613             // draw each overflow cell individually 
1614             int col_end 
= col
+cell_cols
+overflowCols
; 
1615             if (col_end 
>= grid
.GetNumberCols()) 
1616                 col_end 
= grid
.GetNumberCols() - 1; 
1617             for (int i 
= col
+cell_cols
; i 
<= col_end
; i
++) 
1619                 clip
.width 
= grid
.GetColSize(i
) - 1; 
1620                 dc
.DestroyClippingRegion(); 
1621                 dc
.SetClippingRegion(clip
); 
1623                 SetTextColoursAndFont(grid
, attr
, dc
, 
1624                         grid
.IsInSelection(row
,i
)); 
1626                 grid
.DrawTextRectangle(dc
, grid
.GetCellValue(row
, col
), 
1627                         rect
, hAlign
, vAlign
); 
1628                 clip
.x 
+= grid
.GetColSize(i
) - 1; 
1634             dc
.DestroyClippingRegion(); 
1638     // now we only have to draw the text 
1639     SetTextColoursAndFont(grid
, attr
, dc
, isSelected
); 
1641     grid
.DrawTextRectangle(dc
, grid
.GetCellValue(row
, col
), 
1642                            rect
, hAlign
, vAlign
); 
1645 // ---------------------------------------------------------------------------- 
1646 // wxGridCellNumberRenderer 
1647 // ---------------------------------------------------------------------------- 
1649 wxString 
wxGridCellNumberRenderer::GetString(wxGrid
& grid
, int row
, int col
) 
1651     wxGridTableBase 
*table 
= grid
.GetTable(); 
1653     if ( table
->CanGetValueAs(row
, col
, wxGRID_VALUE_NUMBER
) ) 
1655         text
.Printf(_T("%ld"), table
->GetValueAsLong(row
, col
)); 
1659         text 
= table
->GetValue(row
, col
); 
1665 void wxGridCellNumberRenderer::Draw(wxGrid
& grid
, 
1666                                     wxGridCellAttr
& attr
, 
1668                                     const wxRect
& rectCell
, 
1672     wxGridCellRenderer::Draw(grid
, attr
, dc
, rectCell
, row
, col
, isSelected
); 
1674     SetTextColoursAndFont(grid
, attr
, dc
, isSelected
); 
1676     // draw the text right aligned by default 
1678     attr
.GetAlignment(&hAlign
, &vAlign
); 
1679     hAlign 
= wxALIGN_RIGHT
; 
1681     wxRect rect 
= rectCell
; 
1684     grid
.DrawTextRectangle(dc
, GetString(grid
, row
, col
), rect
, hAlign
, vAlign
); 
1687 wxSize 
wxGridCellNumberRenderer::GetBestSize(wxGrid
& grid
, 
1688                                              wxGridCellAttr
& attr
, 
1692     return DoGetBestSize(attr
, dc
, GetString(grid
, row
, col
)); 
1695 // ---------------------------------------------------------------------------- 
1696 // wxGridCellFloatRenderer 
1697 // ---------------------------------------------------------------------------- 
1699 wxGridCellFloatRenderer::wxGridCellFloatRenderer(int width
, int precision
) 
1702     SetPrecision(precision
); 
1705 wxGridCellRenderer 
*wxGridCellFloatRenderer::Clone() const 
1707     wxGridCellFloatRenderer 
*renderer 
= new wxGridCellFloatRenderer
; 
1708     renderer
->m_width 
= m_width
; 
1709     renderer
->m_precision 
= m_precision
; 
1710     renderer
->m_format 
= m_format
; 
1715 wxString 
wxGridCellFloatRenderer::GetString(wxGrid
& grid
, int row
, int col
) 
1717     wxGridTableBase 
*table 
= grid
.GetTable(); 
1722     if ( table
->CanGetValueAs(row
, col
, wxGRID_VALUE_FLOAT
) ) 
1724         val 
= table
->GetValueAsDouble(row
, col
); 
1729         text 
= table
->GetValue(row
, col
); 
1730         hasDouble 
= text
.ToDouble(&val
); 
1737             if ( m_width 
== -1 ) 
1739                 if ( m_precision 
== -1 ) 
1741                     // default width/precision 
1742                     m_format 
= _T("%f"); 
1746                     m_format
.Printf(_T("%%.%df"), m_precision
); 
1749             else if ( m_precision 
== -1 ) 
1751                 // default precision 
1752                 m_format
.Printf(_T("%%%d.f"), m_width
); 
1756                 m_format
.Printf(_T("%%%d.%df"), m_width
, m_precision
); 
1760         text
.Printf(m_format
, val
); 
1763     //else: text already contains the string 
1768 void wxGridCellFloatRenderer::Draw(wxGrid
& grid
, 
1769                                    wxGridCellAttr
& attr
, 
1771                                    const wxRect
& rectCell
, 
1775     wxGridCellRenderer::Draw(grid
, attr
, dc
, rectCell
, row
, col
, isSelected
); 
1777     SetTextColoursAndFont(grid
, attr
, dc
, isSelected
); 
1779     // draw the text right aligned by default 
1781     attr
.GetAlignment(&hAlign
, &vAlign
); 
1782     hAlign 
= wxALIGN_RIGHT
; 
1784     wxRect rect 
= rectCell
; 
1787     grid
.DrawTextRectangle(dc
, GetString(grid
, row
, col
), rect
, hAlign
, vAlign
); 
1790 wxSize 
wxGridCellFloatRenderer::GetBestSize(wxGrid
& grid
, 
1791                                             wxGridCellAttr
& attr
, 
1795     return DoGetBestSize(attr
, dc
, GetString(grid
, row
, col
)); 
1798 void wxGridCellFloatRenderer::SetParameters(const wxString
& params
) 
1802         // reset to defaults 
1808         wxString tmp 
= params
.BeforeFirst(_T(',')); 
1812             if ( tmp
.ToLong(&width
) ) 
1814                 SetWidth((int)width
); 
1818                 wxLogDebug(_T("Invalid wxGridCellFloatRenderer width parameter string '%s ignored"), params
.c_str()); 
1822                 tmp 
= params
.AfterFirst(_T(',')); 
1826             if ( tmp
.ToLong(&precision
) ) 
1828                 SetPrecision((int)precision
); 
1832                 wxLogDebug(_T("Invalid wxGridCellFloatRenderer precision parameter string '%s ignored"), params
.c_str()); 
1840 // ---------------------------------------------------------------------------- 
1841 // wxGridCellBoolRenderer 
1842 // ---------------------------------------------------------------------------- 
1844 wxSize 
wxGridCellBoolRenderer::ms_sizeCheckMark
; 
1846 // FIXME these checkbox size calculations are really ugly... 
1848 // between checkmark and box 
1849 static const wxCoord wxGRID_CHECKMARK_MARGIN 
= 2; 
1851 wxSize 
wxGridCellBoolRenderer::GetBestSize(wxGrid
& grid
, 
1852                                            wxGridCellAttr
& WXUNUSED(attr
), 
1857     // compute it only once (no locks for MT safeness in GUI thread...) 
1858     if ( !ms_sizeCheckMark
.x 
) 
1860         // get checkbox size 
1861         wxCoord checkSize 
= 0; 
1862         wxCheckBox 
*checkbox 
= new wxCheckBox(&grid
, -1, wxEmptyString
); 
1863         wxSize size 
= checkbox
->GetBestSize(); 
1864         checkSize 
= size
.y 
+ 2*wxGRID_CHECKMARK_MARGIN
; 
1866         // FIXME wxGTK::wxCheckBox::GetBestSize() gives "wrong" result 
1867 #if defined(__WXGTK__) || defined(__WXMOTIF__) 
1868         checkSize 
-= size
.y 
/ 2; 
1873         ms_sizeCheckMark
.x 
= ms_sizeCheckMark
.y 
= checkSize
; 
1876     return ms_sizeCheckMark
; 
1879 void wxGridCellBoolRenderer::Draw(wxGrid
& grid
, 
1880                                   wxGridCellAttr
& attr
, 
1886     wxGridCellRenderer::Draw(grid
, attr
, dc
, rect
, row
, col
, isSelected
); 
1888     // draw a check mark in the centre (ignoring alignment - TODO) 
1889     wxSize size 
= GetBestSize(grid
, attr
, dc
, row
, col
); 
1891     // don't draw outside the cell 
1892     wxCoord minSize 
= wxMin(rect
.width
, rect
.height
); 
1893     if ( size
.x 
>= minSize 
|| size
.y 
>= minSize 
) 
1895         // and even leave (at least) 1 pixel margin 
1896         size
.x 
= size
.y 
= minSize 
- 2; 
1899     // draw a border around checkmark 
1901     rectBorder
.x 
= rect
.x 
+ rect
.width
/2 - size
.x
/2; 
1902     rectBorder
.y 
= rect
.y 
+ rect
.height
/2 - size
.y
/2; 
1903     rectBorder
.width 
= size
.x
; 
1904     rectBorder
.height 
= size
.y
; 
1907     if ( grid
.GetTable()->CanGetValueAs(row
, col
, wxGRID_VALUE_BOOL
) ) 
1908         value 
= grid
.GetTable()->GetValueAsBool(row
, col
); 
1911         wxString 
cellval( grid
.GetTable()->GetValue(row
, col
) ); 
1912         value 
= !( !cellval 
|| (cellval 
== wxT("0")) ); 
1917         wxRect rectMark 
= rectBorder
; 
1919         // MSW DrawCheckMark() is weird (and should probably be changed...) 
1920         rectMark
.Inflate(-wxGRID_CHECKMARK_MARGIN
/2); 
1924         rectMark
.Inflate(-wxGRID_CHECKMARK_MARGIN
); 
1927         dc
.SetTextForeground(attr
.GetTextColour()); 
1928         dc
.DrawCheckMark(rectMark
); 
1931     dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
1932     dc
.SetPen(wxPen(attr
.GetTextColour(), 1, wxSOLID
)); 
1933     dc
.DrawRectangle(rectBorder
); 
1936 // ---------------------------------------------------------------------------- 
1938 // ---------------------------------------------------------------------------- 
1940 void wxGridCellAttr::Init(wxGridCellAttr 
*attrDefault
) 
1944     m_isReadOnly 
= Unset
; 
1949     m_attrkind 
= wxGridCellAttr::Cell
; 
1951     m_sizeRows 
= m_sizeCols 
= 1; 
1954     SetDefAttr(attrDefault
); 
1957 wxGridCellAttr 
*wxGridCellAttr::Clone() const 
1959     wxGridCellAttr 
*attr 
= new wxGridCellAttr(m_defGridAttr
); 
1961     if ( HasTextColour() ) 
1962         attr
->SetTextColour(GetTextColour()); 
1963     if ( HasBackgroundColour() ) 
1964         attr
->SetBackgroundColour(GetBackgroundColour()); 
1966         attr
->SetFont(GetFont()); 
1967     if ( HasAlignment() ) 
1968         attr
->SetAlignment(m_hAlign
, m_vAlign
); 
1970     attr
->SetSize( m_sizeRows
, m_sizeCols 
); 
1974         attr
->SetRenderer(m_renderer
); 
1975         m_renderer
->IncRef(); 
1979         attr
->SetEditor(m_editor
); 
1984         attr
->SetReadOnly(); 
1986     attr
->SetKind( m_attrkind 
); 
1991 void wxGridCellAttr::MergeWith(wxGridCellAttr 
*mergefrom
) 
1993     if ( !HasTextColour() && mergefrom
->HasTextColour() ) 
1994         SetTextColour(mergefrom
->GetTextColour()); 
1995     if ( !HasBackgroundColour() && mergefrom
->HasBackgroundColour() ) 
1996         SetBackgroundColour(mergefrom
->GetBackgroundColour()); 
1997     if ( !HasFont() && mergefrom
->HasFont() ) 
1998         SetFont(mergefrom
->GetFont()); 
1999     if ( !HasAlignment() && mergefrom
->HasAlignment() ){ 
2001         mergefrom
->GetAlignment( &hAlign
, &vAlign
); 
2002         SetAlignment(hAlign
, vAlign
); 
2005     mergefrom
->GetSize( &m_sizeRows
, &m_sizeCols 
); 
2007     // Directly access member functions as GetRender/Editor don't just return 
2008     // m_renderer/m_editor 
2010     // Maybe add support for merge of Render and Editor? 
2011     if (!HasRenderer() && mergefrom
->HasRenderer() ) 
2013         m_renderer 
= mergefrom
->m_renderer
; 
2014         m_renderer
->IncRef(); 
2016     if ( !HasEditor() && mergefrom
->HasEditor() ) 
2018         m_editor 
=  mergefrom
->m_editor
; 
2021     if ( !HasReadWriteMode()  && mergefrom
->HasReadWriteMode() ) 
2022         SetReadOnly(mergefrom
->IsReadOnly()); 
2024     SetDefAttr(mergefrom
->m_defGridAttr
); 
2027 void wxGridCellAttr::SetSize(int num_rows
, int num_cols
) 
2029     // The size of a cell is normally 1,1 
2031     // If this cell is larger (2,2) then this is the top left cell 
2032     // the other cells that will be covered (lower right cells) must be 
2033     // set to negative or zero values such that 
2034     // row + num_rows of the covered cell points to the larger cell (this cell) 
2035     // same goes for the col + num_cols. 
2037     // Size of 0,0 is NOT valid, neither is <=0 and any positive value 
2039     wxASSERT_MSG( (!((num_rows
>0)&&(num_cols
<=0)) || 
2040                   !((num_rows
<=0)&&(num_cols
>0)) || 
2041                   !((num_rows
==0)&&(num_cols
==0))), 
2042                   wxT("wxGridCellAttr::SetSize only takes two postive values or negative/zero values")); 
2044     m_sizeRows 
= num_rows
; 
2045     m_sizeCols 
= num_cols
; 
2048 const wxColour
& wxGridCellAttr::GetTextColour() const 
2050     if (HasTextColour()) 
2054     else if (m_defGridAttr 
&& m_defGridAttr 
!= this) 
2056         return m_defGridAttr
->GetTextColour(); 
2060         wxFAIL_MSG(wxT("Missing default cell attribute")); 
2061         return wxNullColour
; 
2066 const wxColour
& wxGridCellAttr::GetBackgroundColour() const 
2068     if (HasBackgroundColour()) 
2070     else if (m_defGridAttr 
&& m_defGridAttr 
!= this) 
2071         return m_defGridAttr
->GetBackgroundColour(); 
2074         wxFAIL_MSG(wxT("Missing default cell attribute")); 
2075         return wxNullColour
; 
2080 const wxFont
& wxGridCellAttr::GetFont() const 
2084     else if (m_defGridAttr 
&& m_defGridAttr 
!= this) 
2085         return m_defGridAttr
->GetFont(); 
2088         wxFAIL_MSG(wxT("Missing default cell attribute")); 
2094 void wxGridCellAttr::GetAlignment(int *hAlign
, int *vAlign
) const 
2098         if ( hAlign 
) *hAlign 
= m_hAlign
; 
2099         if ( vAlign 
) *vAlign 
= m_vAlign
; 
2101     else if (m_defGridAttr 
&& m_defGridAttr 
!= this) 
2102         m_defGridAttr
->GetAlignment(hAlign
, vAlign
); 
2105         wxFAIL_MSG(wxT("Missing default cell attribute")); 
2109 void wxGridCellAttr::GetSize( int *num_rows
, int *num_cols 
) const 
2111     if ( num_rows 
) *num_rows 
= m_sizeRows
; 
2112     if ( num_cols 
) *num_cols 
= m_sizeCols
; 
2115 // GetRenderer and GetEditor use a slightly different decision path about 
2116 // which attribute to use.  If a non-default attr object has one then it is 
2117 // used, otherwise the default editor or renderer is fetched from the grid and 
2118 // used.  It should be the default for the data type of the cell.  If it is 
2119 // NULL (because the table has a type that the grid does not have in its 
2120 // registry,) then the grid's default editor or renderer is used. 
2122 wxGridCellRenderer
* wxGridCellAttr::GetRenderer(wxGrid
* grid
, int row
, int col
) const 
2124     wxGridCellRenderer 
*renderer
; 
2126     if ( m_renderer 
&& this != m_defGridAttr 
) 
2128         // use the cells renderer if it has one 
2129         renderer 
= m_renderer
; 
2132     else // no non default cell renderer 
2134         // get default renderer for the data type 
2137             // GetDefaultRendererForCell() will do IncRef() for us 
2138             renderer 
= grid
->GetDefaultRendererForCell(row
, col
); 
2147             if (m_defGridAttr 
&&  this != m_defGridAttr 
) 
2149                 // if we still don't have one then use the grid default 
2150                 // (no need for IncRef() here neither) 
2151                 renderer 
= m_defGridAttr
->GetRenderer(NULL
, 0, 0); 
2153             else // default grid attr 
2155                 // use m_renderer which we had decided not to use initially 
2156                 renderer 
= m_renderer
; 
2163     // we're supposed to always find something 
2164     wxASSERT_MSG(renderer
, wxT("Missing default cell renderer")); 
2169 // same as above, except for s/renderer/editor/g 
2170 wxGridCellEditor
* wxGridCellAttr::GetEditor(wxGrid
* grid
, int row
, int col
) const 
2172     wxGridCellEditor 
*editor
; 
2174     if ( m_editor 
&& this != m_defGridAttr 
) 
2176         // use the cells editor if it has one 
2180     else // no non default cell editor 
2182         // get default editor for the data type 
2185             // GetDefaultEditorForCell() will do IncRef() for us 
2186             editor 
= grid
->GetDefaultEditorForCell(row
, col
); 
2195             if ( m_defGridAttr 
&& this != m_defGridAttr 
) 
2197                 // if we still don't have one then use the grid default 
2198                 // (no need for IncRef() here neither) 
2199                 editor 
= m_defGridAttr
->GetEditor(NULL
, 0, 0); 
2201             else // default grid attr 
2203                 // use m_editor which we had decided not to use initially 
2211     // we're supposed to always find something 
2212     wxASSERT_MSG(editor
, wxT("Missing default cell editor")); 
2217 // ---------------------------------------------------------------------------- 
2218 // wxGridCellAttrData 
2219 // ---------------------------------------------------------------------------- 
2221 void wxGridCellAttrData::SetAttr(wxGridCellAttr 
*attr
, int row
, int col
) 
2223     int n 
= FindIndex(row
, col
); 
2224     if ( n 
== wxNOT_FOUND 
) 
2226         // add the attribute 
2227         m_attrs
.Add(new wxGridCellWithAttr(row
, col
, attr
)); 
2231         // free the old attribute 
2232         m_attrs
[(size_t)n
].attr
->DecRef(); 
2236             // change the attribute 
2237             m_attrs
[(size_t)n
].attr 
= attr
; 
2241             // remove this attribute 
2242             m_attrs
.RemoveAt((size_t)n
); 
2247 wxGridCellAttr 
*wxGridCellAttrData::GetAttr(int row
, int col
) const 
2249     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
2251     int n 
= FindIndex(row
, col
); 
2252     if ( n 
!= wxNOT_FOUND 
) 
2254         attr 
= m_attrs
[(size_t)n
].attr
; 
2261 void wxGridCellAttrData::UpdateAttrRows( size_t pos
, int numRows 
) 
2263     size_t count 
= m_attrs
.GetCount(); 
2264     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2266         wxGridCellCoords
& coords 
= m_attrs
[n
].coords
; 
2267         wxCoord row 
= coords
.GetRow(); 
2268         if ((size_t)row 
>= pos
) 
2272                 // If rows inserted, include row counter where necessary 
2273                 coords
.SetRow(row 
+ numRows
); 
2275             else if (numRows 
< 0) 
2277                 // If rows deleted ... 
2278                 if ((size_t)row 
>= pos 
- numRows
) 
2280                     // ...either decrement row counter (if row still exists)... 
2281                     coords
.SetRow(row 
+ numRows
); 
2285                     // ...or remove the attribute 
2286                     m_attrs
.RemoveAt((size_t)n
); 
2294 void wxGridCellAttrData::UpdateAttrCols( size_t pos
, int numCols 
) 
2296     size_t count 
= m_attrs
.GetCount(); 
2297     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2299         wxGridCellCoords
& coords 
= m_attrs
[n
].coords
; 
2300         wxCoord col 
= coords
.GetCol(); 
2301         if ( (size_t)col 
>= pos 
) 
2305                 // If rows inserted, include row counter where necessary 
2306                 coords
.SetCol(col 
+ numCols
); 
2308             else if (numCols 
< 0) 
2310                 // If rows deleted ... 
2311                 if ((size_t)col 
>= pos 
- numCols
) 
2313                     // ...either decrement row counter (if row still exists)... 
2314                     coords
.SetCol(col 
+ numCols
); 
2318                     // ...or remove the attribute 
2319                     m_attrs
.RemoveAt((size_t)n
); 
2327 int wxGridCellAttrData::FindIndex(int row
, int col
) const 
2329     size_t count 
= m_attrs
.GetCount(); 
2330     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2332         const wxGridCellCoords
& coords 
= m_attrs
[n
].coords
; 
2333         if ( (coords
.GetRow() == row
) && (coords
.GetCol() == col
) ) 
2342 // ---------------------------------------------------------------------------- 
2343 // wxGridRowOrColAttrData 
2344 // ---------------------------------------------------------------------------- 
2346 wxGridRowOrColAttrData::~wxGridRowOrColAttrData() 
2348     size_t count 
= m_attrs
.Count(); 
2349     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2351         m_attrs
[n
]->DecRef(); 
2355 wxGridCellAttr 
*wxGridRowOrColAttrData::GetAttr(int rowOrCol
) const 
2357     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
2359     int n 
= m_rowsOrCols
.Index(rowOrCol
); 
2360     if ( n 
!= wxNOT_FOUND 
) 
2362         attr 
= m_attrs
[(size_t)n
]; 
2369 void wxGridRowOrColAttrData::SetAttr(wxGridCellAttr 
*attr
, int rowOrCol
) 
2371     int i 
= m_rowsOrCols
.Index(rowOrCol
); 
2372     if ( i 
== wxNOT_FOUND 
) 
2374         // add the attribute 
2375         m_rowsOrCols
.Add(rowOrCol
); 
2380         size_t n 
= (size_t)i
; 
2383             // change the attribute 
2384             m_attrs
[n
]->DecRef(); 
2389             // remove this attribute 
2390             m_attrs
[n
]->DecRef(); 
2391             m_rowsOrCols
.RemoveAt(n
); 
2392             m_attrs
.RemoveAt(n
); 
2397 void wxGridRowOrColAttrData::UpdateAttrRowsOrCols( size_t pos
, int numRowsOrCols 
) 
2399     size_t count 
= m_attrs
.GetCount(); 
2400     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2402         int & rowOrCol 
= m_rowsOrCols
[n
]; 
2403         if ( (size_t)rowOrCol 
>= pos 
) 
2405             if ( numRowsOrCols 
> 0 ) 
2407                 // If rows inserted, include row counter where necessary 
2408                 rowOrCol 
+= numRowsOrCols
; 
2410             else if ( numRowsOrCols 
< 0) 
2412                 // If rows deleted, either decrement row counter (if row still exists) 
2413                 if ((size_t)rowOrCol 
>= pos 
- numRowsOrCols
) 
2414                     rowOrCol 
+= numRowsOrCols
; 
2417                     m_rowsOrCols
.RemoveAt((size_t)n
); 
2418                     m_attrs
.RemoveAt((size_t)n
); 
2426 // ---------------------------------------------------------------------------- 
2427 // wxGridCellAttrProvider 
2428 // ---------------------------------------------------------------------------- 
2430 wxGridCellAttrProvider::wxGridCellAttrProvider() 
2432     m_data 
= (wxGridCellAttrProviderData 
*)NULL
; 
2435 wxGridCellAttrProvider::~wxGridCellAttrProvider() 
2440 void wxGridCellAttrProvider::InitData() 
2442     m_data 
= new wxGridCellAttrProviderData
; 
2445 wxGridCellAttr 
*wxGridCellAttrProvider::GetAttr(int row
, int col
, 
2446                                                 wxGridCellAttr::wxAttrKind  kind 
) const 
2448     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
2453             case (wxGridCellAttr::Any
): 
2454                 //Get cached merge attributes. 
2455                 // Currenlty not used as no cache implemented as not mutiable 
2456                 // attr = m_data->m_mergeAttr.GetAttr(row, col); 
2459                     //Basicaly implement old version. 
2460                     //Also check merge cache, so we don't have to re-merge every time.. 
2461                     wxGridCellAttr 
*attrcell 
= (wxGridCellAttr 
*)NULL
, 
2462                                    *attrrow 
= (wxGridCellAttr 
*)NULL
, 
2463                                    *attrcol 
= (wxGridCellAttr 
*)NULL
; 
2465                     attrcell 
= m_data
->m_cellAttrs
.GetAttr(row
, col
); 
2466                     attrcol 
= m_data
->m_colAttrs
.GetAttr(col
); 
2467                     attrrow 
= m_data
->m_rowAttrs
.GetAttr(row
); 
2469                     if((attrcell 
!= attrrow
) && (attrrow 
!=attrcol
) && (attrcell 
!= attrcol
)){ 
2470                         // Two or move are non NULL 
2471                         attr 
= new wxGridCellAttr
; 
2472                         attr
->SetKind(wxGridCellAttr::Merged
); 
2476                             attr
->MergeWith(attrcell
); 
2480                             attr
->MergeWith(attrcol
); 
2484                             attr
->MergeWith(attrrow
); 
2487                         //store merge attr if cache implemented 
2489                         //m_data->m_mergeAttr.SetAttr(attr, row, col); 
2493                         // one or none is non null return it or null. 
2494                         if(attrrow
) attr 
= attrrow
; 
2495                         if(attrcol
) attr 
= attrcol
; 
2496                         if(attrcell
) attr 
= attrcell
; 
2500             case (wxGridCellAttr::Cell
): 
2501                 attr 
= m_data
->m_cellAttrs
.GetAttr(row
, col
); 
2503             case (wxGridCellAttr::Col
): 
2504                  attr 
= m_data
->m_colAttrs
.GetAttr(col
); 
2506             case (wxGridCellAttr::Row
): 
2507             attr 
= m_data
->m_rowAttrs
.GetAttr(row
); 
2511                 // (wxGridCellAttr::Default): 
2512                 // (wxGridCellAttr::Merged): 
2519 void wxGridCellAttrProvider::SetAttr(wxGridCellAttr 
*attr
, 
2525     m_data
->m_cellAttrs
.SetAttr(attr
, row
, col
); 
2528 void wxGridCellAttrProvider::SetRowAttr(wxGridCellAttr 
*attr
, int row
) 
2533     m_data
->m_rowAttrs
.SetAttr(attr
, row
); 
2536 void wxGridCellAttrProvider::SetColAttr(wxGridCellAttr 
*attr
, int col
) 
2541     m_data
->m_colAttrs
.SetAttr(attr
, col
); 
2544 void wxGridCellAttrProvider::UpdateAttrRows( size_t pos
, int numRows 
) 
2548         m_data
->m_cellAttrs
.UpdateAttrRows( pos
, numRows 
); 
2550         m_data
->m_rowAttrs
.UpdateAttrRowsOrCols( pos
, numRows 
); 
2554 void wxGridCellAttrProvider::UpdateAttrCols( size_t pos
, int numCols 
) 
2558         m_data
->m_cellAttrs
.UpdateAttrCols( pos
, numCols 
); 
2560         m_data
->m_colAttrs
.UpdateAttrRowsOrCols( pos
, numCols 
); 
2564 // ---------------------------------------------------------------------------- 
2565 // wxGridTypeRegistry 
2566 // ---------------------------------------------------------------------------- 
2568 wxGridTypeRegistry::~wxGridTypeRegistry() 
2570     size_t count 
= m_typeinfo
.Count(); 
2571     for ( size_t i 
= 0; i 
< count
; i
++ ) 
2572         delete m_typeinfo
[i
]; 
2576 void wxGridTypeRegistry::RegisterDataType(const wxString
& typeName
, 
2577                                           wxGridCellRenderer
* renderer
, 
2578                                           wxGridCellEditor
* editor
) 
2580     wxGridDataTypeInfo
* info 
= new wxGridDataTypeInfo(typeName
, renderer
, editor
); 
2582     // is it already registered? 
2583     int loc 
= FindRegisteredDataType(typeName
); 
2584     if ( loc 
!= wxNOT_FOUND 
) 
2586         delete m_typeinfo
[loc
]; 
2587         m_typeinfo
[loc
] = info
; 
2591         m_typeinfo
.Add(info
); 
2595 int wxGridTypeRegistry::FindRegisteredDataType(const wxString
& typeName
) 
2597     size_t count 
= m_typeinfo
.GetCount(); 
2598     for ( size_t i 
= 0; i 
< count
; i
++ ) 
2600         if ( typeName 
== m_typeinfo
[i
]->m_typeName 
) 
2609 int wxGridTypeRegistry::FindDataType(const wxString
& typeName
) 
2611     int index 
= FindRegisteredDataType(typeName
); 
2612     if ( index 
== wxNOT_FOUND 
) 
2614         // check whether this is one of the standard ones, in which case 
2615         // register it "on the fly" 
2617         if ( typeName 
== wxGRID_VALUE_STRING 
) 
2619             RegisterDataType(wxGRID_VALUE_STRING
, 
2620                              new wxGridCellStringRenderer
, 
2621                              new wxGridCellTextEditor
); 
2623 #endif // wxUSE_TEXTCTRL 
2625         if ( typeName 
== wxGRID_VALUE_BOOL 
) 
2627             RegisterDataType(wxGRID_VALUE_BOOL
, 
2628                              new wxGridCellBoolRenderer
, 
2629                              new wxGridCellBoolEditor
); 
2631 #endif // wxUSE_CHECKBOX 
2633         if ( typeName 
== wxGRID_VALUE_NUMBER 
) 
2635             RegisterDataType(wxGRID_VALUE_NUMBER
, 
2636                              new wxGridCellNumberRenderer
, 
2637                              new wxGridCellNumberEditor
); 
2639         else if ( typeName 
== wxGRID_VALUE_FLOAT 
) 
2641             RegisterDataType(wxGRID_VALUE_FLOAT
, 
2642                              new wxGridCellFloatRenderer
, 
2643                              new wxGridCellFloatEditor
); 
2645 #endif // wxUSE_TEXTCTRL 
2647         if ( typeName 
== wxGRID_VALUE_CHOICE 
) 
2649             RegisterDataType(wxGRID_VALUE_CHOICE
, 
2650                              new wxGridCellStringRenderer
, 
2651                              new wxGridCellChoiceEditor
); 
2653 #endif // wxUSE_COMBOBOX 
2658         // we get here only if just added the entry for this type, so return 
2660         index 
= m_typeinfo
.GetCount() - 1; 
2666 int wxGridTypeRegistry::FindOrCloneDataType(const wxString
& typeName
) 
2668     int index 
= FindDataType(typeName
); 
2669     if ( index 
== wxNOT_FOUND 
) 
2671         // the first part of the typename is the "real" type, anything after ':' 
2672         // are the parameters for the renderer 
2673         index 
= FindDataType(typeName
.BeforeFirst(_T(':'))); 
2674         if ( index 
== wxNOT_FOUND 
) 
2679         wxGridCellRenderer 
*renderer 
= GetRenderer(index
); 
2680         wxGridCellRenderer 
*rendererOld 
= renderer
; 
2681         renderer 
= renderer
->Clone(); 
2682         rendererOld
->DecRef(); 
2684         wxGridCellEditor 
*editor 
= GetEditor(index
); 
2685         wxGridCellEditor 
*editorOld 
= editor
; 
2686         editor 
= editor
->Clone(); 
2687         editorOld
->DecRef(); 
2689         // do it even if there are no parameters to reset them to defaults 
2690         wxString params 
= typeName
.AfterFirst(_T(':')); 
2691         renderer
->SetParameters(params
); 
2692         editor
->SetParameters(params
); 
2694         // register the new typename 
2695         RegisterDataType(typeName
, renderer
, editor
); 
2697         // we just registered it, it's the last one 
2698         index 
= m_typeinfo
.GetCount() - 1; 
2704 wxGridCellRenderer
* wxGridTypeRegistry::GetRenderer(int index
) 
2706     wxGridCellRenderer
* renderer 
= m_typeinfo
[index
]->m_renderer
; 
2712 wxGridCellEditor
* wxGridTypeRegistry::GetEditor(int index
) 
2714     wxGridCellEditor
* editor 
= m_typeinfo
[index
]->m_editor
; 
2720 // ---------------------------------------------------------------------------- 
2722 // ---------------------------------------------------------------------------- 
2724 IMPLEMENT_ABSTRACT_CLASS( wxGridTableBase
, wxObject 
) 
2727 wxGridTableBase::wxGridTableBase() 
2729     m_view 
= (wxGrid 
*) NULL
; 
2730     m_attrProvider 
= (wxGridCellAttrProvider 
*) NULL
; 
2733 wxGridTableBase::~wxGridTableBase() 
2735     delete m_attrProvider
; 
2738 void wxGridTableBase::SetAttrProvider(wxGridCellAttrProvider 
*attrProvider
) 
2740     delete m_attrProvider
; 
2741     m_attrProvider 
= attrProvider
; 
2744 bool wxGridTableBase::CanHaveAttributes() 
2746     if ( ! GetAttrProvider() ) 
2748         // use the default attr provider by default 
2749         SetAttrProvider(new wxGridCellAttrProvider
); 
2754 wxGridCellAttr 
*wxGridTableBase::GetAttr(int row
, int col
, wxGridCellAttr::wxAttrKind  kind
) 
2756     if ( m_attrProvider 
) 
2757         return m_attrProvider
->GetAttr(row
, col
, kind
); 
2759         return (wxGridCellAttr 
*)NULL
; 
2762 void wxGridTableBase::SetAttr(wxGridCellAttr
* attr
, int row
, int col
) 
2764     if ( m_attrProvider 
) 
2766         attr
->SetKind(wxGridCellAttr::Cell
); 
2767         m_attrProvider
->SetAttr(attr
, row
, col
); 
2771         // as we take ownership of the pointer and don't store it, we must 
2777 void wxGridTableBase::SetRowAttr(wxGridCellAttr 
*attr
, int row
) 
2779     if ( m_attrProvider 
) 
2781         attr
->SetKind(wxGridCellAttr::Row
); 
2782         m_attrProvider
->SetRowAttr(attr
, row
); 
2786         // as we take ownership of the pointer and don't store it, we must 
2792 void wxGridTableBase::SetColAttr(wxGridCellAttr 
*attr
, int col
) 
2794     if ( m_attrProvider 
) 
2796         attr
->SetKind(wxGridCellAttr::Col
); 
2797         m_attrProvider
->SetColAttr(attr
, col
); 
2801         // as we take ownership of the pointer and don't store it, we must 
2807 bool wxGridTableBase::InsertRows( size_t WXUNUSED(pos
), 
2808                                   size_t WXUNUSED(numRows
) ) 
2810     wxFAIL_MSG( wxT("Called grid table class function InsertRows\nbut your derived table class does not override this function") ); 
2815 bool wxGridTableBase::AppendRows( size_t WXUNUSED(numRows
) ) 
2817     wxFAIL_MSG( wxT("Called grid table class function AppendRows\nbut your derived table class does not override this function")); 
2822 bool wxGridTableBase::DeleteRows( size_t WXUNUSED(pos
), 
2823                                   size_t WXUNUSED(numRows
) ) 
2825     wxFAIL_MSG( wxT("Called grid table class function DeleteRows\nbut your derived table class does not override this function")); 
2830 bool wxGridTableBase::InsertCols( size_t WXUNUSED(pos
), 
2831                                   size_t WXUNUSED(numCols
) ) 
2833     wxFAIL_MSG( wxT("Called grid table class function InsertCols\nbut your derived table class does not override this function")); 
2838 bool wxGridTableBase::AppendCols( size_t WXUNUSED(numCols
) ) 
2840     wxFAIL_MSG(wxT("Called grid table class function AppendCols\nbut your derived table class does not override this function")); 
2845 bool wxGridTableBase::DeleteCols( size_t WXUNUSED(pos
), 
2846                                   size_t WXUNUSED(numCols
) ) 
2848     wxFAIL_MSG( wxT("Called grid table class function DeleteCols\nbut your derived table class does not override this function")); 
2854 wxString 
wxGridTableBase::GetRowLabelValue( int row 
) 
2857     s 
<< row 
+ 1; // RD: Starting the rows at zero confuses users, no matter 
2858                   //     how much it makes sense to us geeks. 
2862 wxString 
wxGridTableBase::GetColLabelValue( int col 
) 
2864     // default col labels are: 
2865     //   cols 0 to 25   : A-Z 
2866     //   cols 26 to 675 : AA-ZZ 
2871     for ( n 
= 1; ; n
++ ) 
2873         s 
+= (_T('A') + (wxChar
)( col%26 
)); 
2875         if ( col 
< 0 ) break; 
2878     // reverse the string... 
2880     for ( i 
= 0;  i 
< n
;  i
++ ) 
2889 wxString 
wxGridTableBase::GetTypeName( int WXUNUSED(row
), int WXUNUSED(col
) ) 
2891     return wxGRID_VALUE_STRING
; 
2894 bool wxGridTableBase::CanGetValueAs( int WXUNUSED(row
), int WXUNUSED(col
), 
2895                                      const wxString
& typeName 
) 
2897     return typeName 
== wxGRID_VALUE_STRING
; 
2900 bool wxGridTableBase::CanSetValueAs( int row
, int col
, const wxString
& typeName 
) 
2902     return CanGetValueAs(row
, col
, typeName
); 
2905 long wxGridTableBase::GetValueAsLong( int WXUNUSED(row
), int WXUNUSED(col
) ) 
2910 double wxGridTableBase::GetValueAsDouble( int WXUNUSED(row
), int WXUNUSED(col
) ) 
2915 bool wxGridTableBase::GetValueAsBool( int WXUNUSED(row
), int WXUNUSED(col
) ) 
2920 void wxGridTableBase::SetValueAsLong( int WXUNUSED(row
), int WXUNUSED(col
), 
2921                                       long WXUNUSED(value
) ) 
2925 void wxGridTableBase::SetValueAsDouble( int WXUNUSED(row
), int WXUNUSED(col
), 
2926                                         double WXUNUSED(value
) ) 
2930 void wxGridTableBase::SetValueAsBool( int WXUNUSED(row
), int WXUNUSED(col
), 
2931                                       bool WXUNUSED(value
) ) 
2936 void* wxGridTableBase::GetValueAsCustom( int WXUNUSED(row
), int WXUNUSED(col
), 
2937                                          const wxString
& WXUNUSED(typeName
) ) 
2942 void  wxGridTableBase::SetValueAsCustom( int WXUNUSED(row
), int WXUNUSED(col
), 
2943                                          const wxString
& WXUNUSED(typeName
), 
2944                                          void* WXUNUSED(value
) ) 
2948 ////////////////////////////////////////////////////////////////////// 
2950 // Message class for the grid table to send requests and notifications 
2954 wxGridTableMessage::wxGridTableMessage() 
2956     m_table 
= (wxGridTableBase 
*) NULL
; 
2962 wxGridTableMessage::wxGridTableMessage( wxGridTableBase 
*table
, int id
, 
2963                                         int commandInt1
, int commandInt2 
) 
2967     m_comInt1 
= commandInt1
; 
2968     m_comInt2 
= commandInt2
; 
2973 ////////////////////////////////////////////////////////////////////// 
2975 // A basic grid table for string data. An object of this class will 
2976 // created by wxGrid if you don't specify an alternative table class. 
2979 WX_DEFINE_OBJARRAY(wxGridStringArray
) 
2981 IMPLEMENT_DYNAMIC_CLASS( wxGridStringTable
, wxGridTableBase 
) 
2983 wxGridStringTable::wxGridStringTable() 
2988 wxGridStringTable::wxGridStringTable( int numRows
, int numCols 
) 
2991     m_data
.Alloc( numRows 
); 
2994     sa
.Alloc( numCols 
); 
2995     sa
.Add( wxEmptyString
, numCols 
); 
2997     m_data
.Add( sa
, numRows 
); 
3000 wxGridStringTable::~wxGridStringTable() 
3004 int wxGridStringTable::GetNumberRows() 
3006     return m_data
.GetCount(); 
3009 int wxGridStringTable::GetNumberCols() 
3011     if ( m_data
.GetCount() > 0 ) 
3012         return m_data
[0].GetCount(); 
3017 wxString 
wxGridStringTable::GetValue( int row
, int col 
) 
3019     wxASSERT_MSG( (row 
< GetNumberRows()) && (col 
< GetNumberCols()), 
3020                   _T("invalid row or column index in wxGridStringTable") ); 
3022     return m_data
[row
][col
]; 
3025 void wxGridStringTable::SetValue( int row
, int col
, const wxString
& value 
) 
3027     wxASSERT_MSG( (row 
< GetNumberRows()) && (col 
< GetNumberCols()), 
3028                   _T("invalid row or column index in wxGridStringTable") ); 
3030     m_data
[row
][col
] = value
; 
3033 bool wxGridStringTable::IsEmptyCell( int row
, int col 
) 
3035     wxASSERT_MSG( (row 
< GetNumberRows()) && (col 
< GetNumberCols()), 
3036                   _T("invalid row or column index in wxGridStringTable") ); 
3038     return (m_data
[row
][col
] == wxEmptyString
); 
3041 void wxGridStringTable::Clear() 
3044     int numRows
, numCols
; 
3046     numRows 
= m_data
.GetCount(); 
3049         numCols 
= m_data
[0].GetCount(); 
3051         for ( row 
= 0;  row 
< numRows
;  row
++ ) 
3053             for ( col 
= 0;  col 
< numCols
;  col
++ ) 
3055                 m_data
[row
][col
] = wxEmptyString
; 
3062 bool wxGridStringTable::InsertRows( size_t pos
, size_t numRows 
) 
3064     size_t curNumRows 
= m_data
.GetCount(); 
3065     size_t curNumCols 
= ( curNumRows 
> 0 ? m_data
[0].GetCount() : 
3066                           ( GetView() ? GetView()->GetNumberCols() : 0 ) ); 
3068     if ( pos 
>= curNumRows 
) 
3070         return AppendRows( numRows 
); 
3074     sa
.Alloc( curNumCols 
); 
3075     sa
.Add( wxEmptyString
, curNumCols 
); 
3076     m_data
.Insert( sa
, pos
, numRows 
); 
3079         wxGridTableMessage 
msg( this, 
3080                                 wxGRIDTABLE_NOTIFY_ROWS_INSERTED
, 
3084         GetView()->ProcessTableMessage( msg 
); 
3090 bool wxGridStringTable::AppendRows( size_t numRows 
) 
3092     size_t curNumRows 
= m_data
.GetCount(); 
3093     size_t curNumCols 
= ( curNumRows 
> 0 ? m_data
[0].GetCount() : 
3094                           ( GetView() ? GetView()->GetNumberCols() : 0 ) ); 
3097     if ( curNumCols 
> 0 ) 
3099         sa
.Alloc( curNumCols 
); 
3100         sa
.Add( wxEmptyString
, curNumCols 
); 
3103     m_data
.Add( sa
, numRows 
); 
3107         wxGridTableMessage 
msg( this, 
3108                                 wxGRIDTABLE_NOTIFY_ROWS_APPENDED
, 
3111         GetView()->ProcessTableMessage( msg 
); 
3117 bool wxGridStringTable::DeleteRows( size_t pos
, size_t numRows 
) 
3119     size_t curNumRows 
= m_data
.GetCount(); 
3121     if ( pos 
>= curNumRows 
) 
3123         wxFAIL_MSG( wxString::Format
 
3125                         wxT("Called wxGridStringTable::DeleteRows(pos=%lu, N=%lu)\nPos value is invalid for present table with %lu rows"), 
3127                         (unsigned long)numRows
, 
3128                         (unsigned long)curNumRows
 
3134     if ( numRows 
> curNumRows 
- pos 
) 
3136         numRows 
= curNumRows 
- pos
; 
3139     if ( numRows 
>= curNumRows 
) 
3145         m_data
.RemoveAt( pos
, numRows 
); 
3149         wxGridTableMessage 
msg( this, 
3150                                 wxGRIDTABLE_NOTIFY_ROWS_DELETED
, 
3154         GetView()->ProcessTableMessage( msg 
); 
3160 bool wxGridStringTable::InsertCols( size_t pos
, size_t numCols 
) 
3164     size_t curNumRows 
= m_data
.GetCount(); 
3165     size_t curNumCols 
= ( curNumRows 
> 0 ? m_data
[0].GetCount() : 
3166                           ( GetView() ? GetView()->GetNumberCols() : 0 ) ); 
3168     if ( pos 
>= curNumCols 
) 
3170         return AppendCols( numCols 
); 
3173     for ( row 
= 0;  row 
< curNumRows
;  row
++ ) 
3175         for ( col 
= pos
;  col 
< pos 
+ numCols
;  col
++ ) 
3177             m_data
[row
].Insert( wxEmptyString
, col 
); 
3182         wxGridTableMessage 
msg( this, 
3183                                 wxGRIDTABLE_NOTIFY_COLS_INSERTED
, 
3187         GetView()->ProcessTableMessage( msg 
); 
3193 bool wxGridStringTable::AppendCols( size_t numCols 
) 
3197     size_t curNumRows 
= m_data
.GetCount(); 
3201         // TODO: something better than this ? 
3203         wxFAIL_MSG( wxT("Unable to append cols to a grid table with no rows.\nCall AppendRows() first") ); 
3208     for ( row 
= 0;  row 
< curNumRows
;  row
++ ) 
3210         m_data
[row
].Add( wxEmptyString
, numCols 
); 
3215         wxGridTableMessage 
msg( this, 
3216                                 wxGRIDTABLE_NOTIFY_COLS_APPENDED
, 
3219         GetView()->ProcessTableMessage( msg 
); 
3225 bool wxGridStringTable::DeleteCols( size_t pos
, size_t numCols 
) 
3229     size_t curNumRows 
= m_data
.GetCount(); 
3230     size_t curNumCols 
= ( curNumRows 
> 0 ? m_data
[0].GetCount() : 
3231                           ( GetView() ? GetView()->GetNumberCols() : 0 ) ); 
3233     if ( pos 
>= curNumCols 
) 
3235         wxFAIL_MSG( wxString::Format
 
3237                         wxT("Called wxGridStringTable::DeleteCols(pos=%lu, N=%lu)\nPos value is invalid for present table with %lu cols"), 
3239                         (unsigned long)numCols
, 
3240                         (unsigned long)curNumCols
 
3245     if ( numCols 
> curNumCols 
- pos 
) 
3247         numCols 
= curNumCols 
- pos
; 
3250     for ( row 
= 0;  row 
< curNumRows
;  row
++ ) 
3252         if ( numCols 
>= curNumCols 
) 
3254             m_data
[row
].Clear(); 
3258             m_data
[row
].RemoveAt( pos
, numCols 
); 
3263         wxGridTableMessage 
msg( this, 
3264                                 wxGRIDTABLE_NOTIFY_COLS_DELETED
, 
3268         GetView()->ProcessTableMessage( msg 
); 
3274 wxString 
wxGridStringTable::GetRowLabelValue( int row 
) 
3276     if ( row 
> (int)(m_rowLabels
.GetCount()) - 1 ) 
3278         // using default label 
3280         return wxGridTableBase::GetRowLabelValue( row 
); 
3284         return m_rowLabels
[ row 
]; 
3288 wxString 
wxGridStringTable::GetColLabelValue( int col 
) 
3290     if ( col 
> (int)(m_colLabels
.GetCount()) - 1 ) 
3292         // using default label 
3294         return wxGridTableBase::GetColLabelValue( col 
); 
3298         return m_colLabels
[ col 
]; 
3302 void wxGridStringTable::SetRowLabelValue( int row
, const wxString
& value 
) 
3304     if ( row 
> (int)(m_rowLabels
.GetCount()) - 1 ) 
3306         int n 
= m_rowLabels
.GetCount(); 
3308         for ( i 
= n
;  i 
<= row
;  i
++ ) 
3310             m_rowLabels
.Add( wxGridTableBase::GetRowLabelValue(i
) ); 
3314     m_rowLabels
[row
] = value
; 
3317 void wxGridStringTable::SetColLabelValue( int col
, const wxString
& value 
) 
3319     if ( col 
> (int)(m_colLabels
.GetCount()) - 1 ) 
3321         int n 
= m_colLabels
.GetCount(); 
3323         for ( i 
= n
;  i 
<= col
;  i
++ ) 
3325             m_colLabels
.Add( wxGridTableBase::GetColLabelValue(i
) ); 
3329     m_colLabels
[col
] = value
; 
3334 ////////////////////////////////////////////////////////////////////// 
3335 ////////////////////////////////////////////////////////////////////// 
3337 IMPLEMENT_DYNAMIC_CLASS( wxGridRowLabelWindow
, wxWindow 
) 
3339 BEGIN_EVENT_TABLE( wxGridRowLabelWindow
, wxWindow 
) 
3340     EVT_PAINT( wxGridRowLabelWindow::OnPaint 
) 
3341     EVT_MOUSEWHEEL( wxGridRowLabelWindow::OnMouseWheel
) 
3342     EVT_MOUSE_EVENTS( wxGridRowLabelWindow::OnMouseEvent 
) 
3343     EVT_KEY_DOWN( wxGridRowLabelWindow::OnKeyDown 
) 
3344     EVT_KEY_UP( wxGridRowLabelWindow::OnKeyUp 
) 
3347 wxGridRowLabelWindow::wxGridRowLabelWindow( wxGrid 
*parent
, 
3349                                             const wxPoint 
&pos
, const wxSize 
&size 
) 
3350   : wxWindow( parent
, id
, pos
, size
, wxWANTS_CHARS 
) 
3355 void wxGridRowLabelWindow::OnPaint( wxPaintEvent
& WXUNUSED(event
) ) 
3359     // NO - don't do this because it will set both the x and y origin 
3360     // coords to match the parent scrolled window and we just want to 
3361     // set the y coord  - MB 
3363     // m_owner->PrepareDC( dc ); 
3366     m_owner
->CalcUnscrolledPosition( 0, 0, &x
, &y 
); 
3367     dc
.SetDeviceOrigin( 0, -y 
); 
3369     wxArrayInt rows 
= m_owner
->CalcRowLabelsExposed( GetUpdateRegion() ); 
3370     m_owner
->DrawRowLabels( dc 
, rows 
); 
3374 void wxGridRowLabelWindow::OnMouseEvent( wxMouseEvent
& event 
) 
3376     m_owner
->ProcessRowLabelMouseEvent( event 
); 
3380 void wxGridRowLabelWindow::OnMouseWheel( wxMouseEvent
& event 
) 
3382     m_owner
->GetEventHandler()->ProcessEvent(event
); 
3386 // This seems to be required for wxMotif otherwise the mouse 
3387 // cursor must be in the cell edit control to get key events 
3389 void wxGridRowLabelWindow::OnKeyDown( wxKeyEvent
& event 
) 
3391     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3394 void wxGridRowLabelWindow::OnKeyUp( wxKeyEvent
& event 
) 
3396     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3401 ////////////////////////////////////////////////////////////////////// 
3403 IMPLEMENT_DYNAMIC_CLASS( wxGridColLabelWindow
, wxWindow 
) 
3405 BEGIN_EVENT_TABLE( wxGridColLabelWindow
, wxWindow 
) 
3406     EVT_PAINT( wxGridColLabelWindow::OnPaint 
) 
3407     EVT_MOUSEWHEEL( wxGridColLabelWindow::OnMouseWheel
) 
3408     EVT_MOUSE_EVENTS( wxGridColLabelWindow::OnMouseEvent 
) 
3409     EVT_KEY_DOWN( wxGridColLabelWindow::OnKeyDown 
) 
3410     EVT_KEY_UP( wxGridColLabelWindow::OnKeyUp 
) 
3413 wxGridColLabelWindow::wxGridColLabelWindow( wxGrid 
*parent
, 
3415                                             const wxPoint 
&pos
, const wxSize 
&size 
) 
3416   : wxWindow( parent
, id
, pos
, size
, wxWANTS_CHARS 
) 
3421 void wxGridColLabelWindow::OnPaint( wxPaintEvent
& WXUNUSED(event
) ) 
3425     // NO - don't do this because it will set both the x and y origin 
3426     // coords to match the parent scrolled window and we just want to 
3427     // set the x coord  - MB 
3429     // m_owner->PrepareDC( dc ); 
3432     m_owner
->CalcUnscrolledPosition( 0, 0, &x
, &y 
); 
3433     dc
.SetDeviceOrigin( -x
, 0 ); 
3435     wxArrayInt cols 
= m_owner
->CalcColLabelsExposed( GetUpdateRegion() ); 
3436     m_owner
->DrawColLabels( dc 
, cols 
); 
3440 void wxGridColLabelWindow::OnMouseEvent( wxMouseEvent
& event 
) 
3442     m_owner
->ProcessColLabelMouseEvent( event 
); 
3445 void wxGridColLabelWindow::OnMouseWheel( wxMouseEvent
& event 
) 
3447     m_owner
->GetEventHandler()->ProcessEvent(event
); 
3451 // This seems to be required for wxMotif otherwise the mouse 
3452 // cursor must be in the cell edit control to get key events 
3454 void wxGridColLabelWindow::OnKeyDown( wxKeyEvent
& event 
) 
3456     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3459 void wxGridColLabelWindow::OnKeyUp( wxKeyEvent
& event 
) 
3461     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3466 ////////////////////////////////////////////////////////////////////// 
3468 IMPLEMENT_DYNAMIC_CLASS( wxGridCornerLabelWindow
, wxWindow 
) 
3470 BEGIN_EVENT_TABLE( wxGridCornerLabelWindow
, wxWindow 
) 
3471     EVT_MOUSEWHEEL( wxGridCornerLabelWindow::OnMouseWheel
) 
3472     EVT_MOUSE_EVENTS( wxGridCornerLabelWindow::OnMouseEvent 
) 
3473     EVT_PAINT( wxGridCornerLabelWindow::OnPaint
) 
3474     EVT_KEY_DOWN( wxGridCornerLabelWindow::OnKeyDown 
) 
3475     EVT_KEY_UP( wxGridCornerLabelWindow::OnKeyUp 
) 
3478 wxGridCornerLabelWindow::wxGridCornerLabelWindow( wxGrid 
*parent
, 
3480                                                   const wxPoint 
&pos
, const wxSize 
&size 
) 
3481   : wxWindow( parent
, id
, pos
, size
, wxWANTS_CHARS 
) 
3486 void wxGridCornerLabelWindow::OnPaint( wxPaintEvent
& WXUNUSED(event
) ) 
3490     int client_height 
= 0; 
3491     int client_width 
= 0; 
3492     GetClientSize( &client_width
, &client_height 
); 
3494     dc
.SetPen( *wxBLACK_PEN 
); 
3495     dc
.DrawLine( client_width
-1, client_height
-1, client_width
-1, 0 ); 
3496     dc
.DrawLine( client_width
-1, client_height
-1, 0, client_height
-1 ); 
3498     dc
.SetPen( *wxWHITE_PEN 
); 
3499     dc
.DrawLine( 0, 0, client_width
, 0 ); 
3500     dc
.DrawLine( 0, 0, 0, client_height 
); 
3504 void wxGridCornerLabelWindow::OnMouseEvent( wxMouseEvent
& event 
) 
3506     m_owner
->ProcessCornerLabelMouseEvent( event 
); 
3510 void wxGridCornerLabelWindow::OnMouseWheel( wxMouseEvent
& event 
) 
3512     m_owner
->GetEventHandler()->ProcessEvent(event
); 
3515 // This seems to be required for wxMotif otherwise the mouse 
3516 // cursor must be in the cell edit control to get key events 
3518 void wxGridCornerLabelWindow::OnKeyDown( wxKeyEvent
& event 
) 
3520     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3523 void wxGridCornerLabelWindow::OnKeyUp( wxKeyEvent
& event 
) 
3525     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3530 ////////////////////////////////////////////////////////////////////// 
3532 IMPLEMENT_DYNAMIC_CLASS( wxGridWindow
, wxWindow 
) 
3534 BEGIN_EVENT_TABLE( wxGridWindow
, wxWindow 
) 
3535     EVT_PAINT( wxGridWindow::OnPaint 
) 
3536     EVT_MOUSEWHEEL( wxGridWindow::OnMouseWheel
) 
3537     EVT_MOUSE_EVENTS( wxGridWindow::OnMouseEvent 
) 
3538     EVT_KEY_DOWN( wxGridWindow::OnKeyDown 
) 
3539     EVT_KEY_UP( wxGridWindow::OnKeyUp 
) 
3540     EVT_ERASE_BACKGROUND( wxGridWindow::OnEraseBackground 
) 
3543 wxGridWindow::wxGridWindow( wxGrid 
*parent
, 
3544                             wxGridRowLabelWindow 
*rowLblWin
, 
3545                             wxGridColLabelWindow 
*colLblWin
, 
3546                             wxWindowID id
, const wxPoint 
&pos
, const wxSize 
&size 
) 
3547         : wxWindow( parent
, id
, pos
, size
, wxWANTS_CHARS
, wxT("grid window") ) 
3550     m_rowLabelWin 
= rowLblWin
; 
3551     m_colLabelWin 
= colLblWin
; 
3552     SetBackgroundColour(_T("WHITE")); 
3556 wxGridWindow::~wxGridWindow() 
3561 void wxGridWindow::OnPaint( wxPaintEvent 
&WXUNUSED(event
) ) 
3563     wxPaintDC 
dc( this ); 
3564     m_owner
->PrepareDC( dc 
); 
3565     wxRegion reg 
= GetUpdateRegion(); 
3566     wxGridCellCoordsArray DirtyCells 
= m_owner
->CalcCellsExposed( reg 
); 
3567     m_owner
->DrawGridCellArea( dc 
, DirtyCells
); 
3568 #if WXGRID_DRAW_LINES 
3569     m_owner
->DrawAllGridLines( dc
, reg 
); 
3571     m_owner
->DrawGridSpace( dc 
); 
3572     m_owner
->DrawHighlight( dc 
, DirtyCells 
); 
3576 void wxGridWindow::ScrollWindow( int dx
, int dy
, const wxRect 
*rect 
) 
3578     wxWindow::ScrollWindow( dx
, dy
, rect 
); 
3579     m_rowLabelWin
->ScrollWindow( 0, dy
, rect 
); 
3580     m_colLabelWin
->ScrollWindow( dx
, 0, rect 
); 
3584 void wxGridWindow::OnMouseEvent( wxMouseEvent
& event 
) 
3586     m_owner
->ProcessGridCellMouseEvent( event 
); 
3589 void wxGridWindow::OnMouseWheel( wxMouseEvent
& event 
) 
3591     m_owner
->GetEventHandler()->ProcessEvent(event
); 
3594 // This seems to be required for wxMotif/wxGTK otherwise the mouse 
3595 // cursor must be in the cell edit control to get key events 
3597 void wxGridWindow::OnKeyDown( wxKeyEvent
& event 
) 
3599     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3602 void wxGridWindow::OnKeyUp( wxKeyEvent
& event 
) 
3604     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3607 void wxGridWindow::OnEraseBackground( wxEraseEvent
& WXUNUSED(event
) ) 
3612 ////////////////////////////////////////////////////////////////////// 
3614 // Internal Helper function for computing row or column from some 
3615 // (unscrolled) coordinate value, using either 
3616 // m_defaultRowHeight/m_defaultColWidth or binary search on array 
3617 // of m_rowBottoms/m_ColRights to speed up the search! 
3619 // Internal helper macros for simpler use of that function 
3621 static int CoordToRowOrCol(int coord
, int defaultDist
, int minDist
, 
3622                            const wxArrayInt
& BorderArray
, int nMax
, 
3623                            bool maxOnOverflow
); 
3625 #define internalXToCol(x) CoordToRowOrCol(x, m_defaultColWidth, \ 
3626                                           WXGRID_MIN_COL_WIDTH, \ 
3627                                           m_colRights, m_numCols, TRUE) 
3628 #define internalYToRow(y) CoordToRowOrCol(y, m_defaultRowHeight, \ 
3629                                           WXGRID_MIN_ROW_HEIGHT, \ 
3630                                           m_rowBottoms, m_numRows, TRUE) 
3631 ///////////////////////////////////////////////////////////////////// 
3633 IMPLEMENT_DYNAMIC_CLASS( wxGrid
, wxScrolledWindow 
) 
3635 BEGIN_EVENT_TABLE( wxGrid
, wxScrolledWindow 
) 
3636     EVT_PAINT( wxGrid::OnPaint 
) 
3637     EVT_SIZE( wxGrid::OnSize 
) 
3638     EVT_KEY_DOWN( wxGrid::OnKeyDown 
) 
3639     EVT_KEY_UP( wxGrid::OnKeyUp 
) 
3640     EVT_ERASE_BACKGROUND( wxGrid::OnEraseBackground 
) 
3643 wxGrid::wxGrid( wxWindow 
*parent
, 
3648                  const wxString
& name 
) 
3649   : wxScrolledWindow( parent
, id
, pos
, size
, (style 
| wxWANTS_CHARS
), name 
), 
3650     m_colMinWidths(GRID_HASH_SIZE
), 
3651     m_rowMinHeights(GRID_HASH_SIZE
) 
3659     // Must do this or ~wxScrollHelper will pop the wrong event handler 
3660     SetTargetWindow(this); 
3662     wxSafeDecRef(m_defaultCellAttr
); 
3664 #ifdef DEBUG_ATTR_CACHE 
3665     size_t total 
= gs_nAttrCacheHits 
+ gs_nAttrCacheMisses
; 
3666     wxPrintf(_T("wxGrid attribute cache statistics: " 
3667                 "total: %u, hits: %u (%u%%)\n"), 
3668              total
, gs_nAttrCacheHits
, 
3669              total 
? (gs_nAttrCacheHits
*100) / total 
: 0); 
3675     delete m_typeRegistry
; 
3681 // ----- internal init and update functions 
3684 void wxGrid::Create() 
3686     m_created 
= FALSE
;    // set to TRUE by CreateGrid 
3688     m_table        
= (wxGridTableBase 
*) NULL
; 
3691     m_cellEditCtrlEnabled 
= FALSE
; 
3693     m_defaultCellAttr 
= new wxGridCellAttr(); 
3695     // Set default cell attributes 
3696     m_defaultCellAttr
->SetDefAttr(m_defaultCellAttr
); 
3697     m_defaultCellAttr
->SetKind(wxGridCellAttr::Default
); 
3698     m_defaultCellAttr
->SetFont(GetFont()); 
3699     m_defaultCellAttr
->SetAlignment(wxALIGN_LEFT
, wxALIGN_TOP
); 
3700     m_defaultCellAttr
->SetTextColour( 
3701         wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT
)); 
3702     m_defaultCellAttr
->SetBackgroundColour( 
3703         wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW
)); 
3704     m_defaultCellAttr
->SetRenderer(new wxGridCellStringRenderer
); 
3705     m_defaultCellAttr
->SetEditor(new wxGridCellTextEditor
); 
3710     m_currentCellCoords 
= wxGridNoCellCoords
; 
3712     m_rowLabelWidth 
= WXGRID_DEFAULT_ROW_LABEL_WIDTH
; 
3713     m_colLabelHeight 
= WXGRID_DEFAULT_COL_LABEL_HEIGHT
; 
3715     // create the type registry 
3716     m_typeRegistry 
= new wxGridTypeRegistry
; 
3719     // subwindow components that make up the wxGrid 
3720     m_cornerLabelWin 
= new wxGridCornerLabelWindow( this, 
3725     m_rowLabelWin 
= new wxGridRowLabelWindow( this, 
3730     m_colLabelWin 
= new wxGridColLabelWindow( this, 
3735     m_gridWin 
= new wxGridWindow( this, 
3742     SetTargetWindow( m_gridWin 
); 
3748 bool wxGrid::CreateGrid( int numRows
, int numCols
, 
3749                          wxGrid::wxGridSelectionModes selmode 
) 
3751     wxCHECK_MSG( !m_created
, 
3753                  wxT("wxGrid::CreateGrid or wxGrid::SetTable called more than once") ); 
3755     m_numRows 
= numRows
; 
3756     m_numCols 
= numCols
; 
3758     m_table 
= new wxGridStringTable( m_numRows
, m_numCols 
); 
3759     m_table
->SetView( this ); 
3761     m_selection 
= new wxGridSelection( this, selmode 
); 
3770 void wxGrid::SetSelectionMode(wxGrid::wxGridSelectionModes selmode
) 
3772     wxCHECK_RET( m_created
, 
3773                  wxT("Called wxGrid::SetSelectionMode() before calling CreateGrid()") ); 
3775     m_selection
->SetSelectionMode( selmode 
); 
3778 wxGrid::wxGridSelectionModes 
wxGrid::GetSelectionMode() const 
3780     wxCHECK_MSG( m_created
, wxGrid::wxGridSelectCells
, 
3781                  wxT("Called wxGrid::GetSelectionMode() before calling CreateGrid()") ); 
3783     return m_selection
->GetSelectionMode(); 
3786 bool wxGrid::SetTable( wxGridTableBase 
*table
, bool takeOwnership
, 
3787                        wxGrid::wxGridSelectionModes selmode 
) 
3791         // RD: Actually, this should probably be allowed.  I think it would be 
3792         //     nice to be able to switch multiple Tables in and out of a single 
3793         //     View at runtime.  Is there anything in the implementation that 
3794         //     would prevent this? 
3796         // At least, you now have to cope with m_selection 
3797         wxFAIL_MSG( wxT("wxGrid::CreateGrid or wxGrid::SetTable called more than once") ); 
3802         m_numRows 
= table
->GetNumberRows(); 
3803         m_numCols 
= table
->GetNumberCols(); 
3806         m_table
->SetView( this ); 
3809         m_selection 
= new wxGridSelection( this, selmode 
); 
3822     m_rowLabelWidth  
= WXGRID_DEFAULT_ROW_LABEL_WIDTH
; 
3823     m_colLabelHeight 
= WXGRID_DEFAULT_COL_LABEL_HEIGHT
; 
3825     if ( m_rowLabelWin 
) 
3827         m_labelBackgroundColour 
= m_rowLabelWin
->GetBackgroundColour(); 
3831         m_labelBackgroundColour 
= wxColour( _T("WHITE") ); 
3834     m_labelTextColour 
= wxColour( _T("BLACK") ); 
3837     m_attrCache
.row 
= -1; 
3838     m_attrCache
.col 
= -1; 
3839     m_attrCache
.attr 
= NULL
; 
3841     // TODO: something better than this ? 
3843     m_labelFont 
= this->GetFont(); 
3844     m_labelFont
.SetWeight( m_labelFont
.GetWeight() + 2 ); 
3846     m_rowLabelHorizAlign 
= wxALIGN_LEFT
; 
3847     m_rowLabelVertAlign  
= wxALIGN_CENTRE
; 
3849     m_colLabelHorizAlign 
= wxALIGN_CENTRE
; 
3850     m_colLabelVertAlign  
= wxALIGN_TOP
; 
3852     m_defaultColWidth  
= WXGRID_DEFAULT_COL_WIDTH
; 
3853     m_defaultRowHeight 
= m_gridWin
->GetCharHeight(); 
3855 #if defined(__WXMOTIF__) || defined(__WXGTK__)  // see also text ctrl sizing in ShowCellEditControl() 
3856     m_defaultRowHeight 
+= 8; 
3858     m_defaultRowHeight 
+= 4; 
3861     m_gridLineColour 
= wxColour( 128, 128, 255 ); 
3862     m_gridLinesEnabled 
= TRUE
; 
3863     m_cellHighlightColour 
= m_gridLineColour
; 
3864     m_cellHighlightPenWidth 
= 2; 
3865     m_cellHighlightROPenWidth 
= 1; 
3867     m_cursorMode  
= WXGRID_CURSOR_SELECT_CELL
; 
3868     m_winCapture 
= (wxWindow 
*)NULL
; 
3869     m_canDragRowSize 
= TRUE
; 
3870     m_canDragColSize 
= TRUE
; 
3871     m_canDragGridSize 
= TRUE
; 
3873     m_dragRowOrCol 
= -1; 
3874     m_isDragging 
= FALSE
; 
3875     m_startDragPos 
= wxDefaultPosition
; 
3877     m_waitForSlowClick 
= FALSE
; 
3879     m_rowResizeCursor 
= wxCursor( wxCURSOR_SIZENS 
); 
3880     m_colResizeCursor 
= wxCursor( wxCURSOR_SIZEWE 
); 
3882     m_currentCellCoords 
= wxGridNoCellCoords
; 
3884     m_selectingTopLeft 
= wxGridNoCellCoords
; 
3885     m_selectingBottomRight 
= wxGridNoCellCoords
; 
3886     m_selectionBackground 
= wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHT
); 
3887     m_selectionForeground 
= wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHTTEXT
); 
3889     m_editable 
= TRUE
;  // default for whole grid 
3891     m_inOnKeyDown 
= FALSE
; 
3898 // ---------------------------------------------------------------------------- 
3899 // the idea is to call these functions only when necessary because they create 
3900 // quite big arrays which eat memory mostly unnecessary - in particular, if 
3901 // default widths/heights are used for all rows/columns, we may not use these 
3904 // with some extra code, it should be possible to only store the 
3905 // widths/heights different from default ones but this will be done later... 
3906 // ---------------------------------------------------------------------------- 
3908 void wxGrid::InitRowHeights() 
3910     m_rowHeights
.Empty(); 
3911     m_rowBottoms
.Empty(); 
3913     m_rowHeights
.Alloc( m_numRows 
); 
3914     m_rowBottoms
.Alloc( m_numRows 
); 
3918     m_rowHeights
.Add( m_defaultRowHeight
, m_numRows 
); 
3920     for ( int i 
= 0;  i 
< m_numRows
;  i
++ ) 
3922         rowBottom 
+= m_defaultRowHeight
; 
3923         m_rowBottoms
.Add( rowBottom 
); 
3927 void wxGrid::InitColWidths() 
3929     m_colWidths
.Empty(); 
3930     m_colRights
.Empty(); 
3932     m_colWidths
.Alloc( m_numCols 
); 
3933     m_colRights
.Alloc( m_numCols 
); 
3936     m_colWidths
.Add( m_defaultColWidth
, m_numCols 
); 
3938     for ( int i 
= 0;  i 
< m_numCols
;  i
++ ) 
3940         colRight 
+= m_defaultColWidth
; 
3941         m_colRights
.Add( colRight 
); 
3945 int wxGrid::GetColWidth(int col
) const 
3947     return m_colWidths
.IsEmpty() ? m_defaultColWidth 
: m_colWidths
[col
]; 
3950 int wxGrid::GetColLeft(int col
) const 
3952     return m_colRights
.IsEmpty() ? col 
* m_defaultColWidth
 
3953                                  : m_colRights
[col
] - m_colWidths
[col
]; 
3956 int wxGrid::GetColRight(int col
) const 
3958     return m_colRights
.IsEmpty() ? (col 
+ 1) * m_defaultColWidth
 
3962 int wxGrid::GetRowHeight(int row
) const 
3964     return m_rowHeights
.IsEmpty() ? m_defaultRowHeight 
: m_rowHeights
[row
]; 
3967 int wxGrid::GetRowTop(int row
) const 
3969     return m_rowBottoms
.IsEmpty() ? row 
* m_defaultRowHeight
 
3970                                   : m_rowBottoms
[row
] - m_rowHeights
[row
]; 
3973 int wxGrid::GetRowBottom(int row
) const 
3975     return m_rowBottoms
.IsEmpty() ? (row 
+ 1) * m_defaultRowHeight
 
3976                                   : m_rowBottoms
[row
]; 
3979 void wxGrid::CalcDimensions() 
3982     GetClientSize( &cw
, &ch 
); 
3984     if ( m_rowLabelWin
->IsShown() ) 
3985         cw 
-= m_rowLabelWidth
; 
3986     if ( m_colLabelWin
->IsShown() ) 
3987         ch 
-= m_colLabelHeight
; 
3990     int w 
= m_numCols 
> 0 ? GetColRight(m_numCols 
- 1) + m_extraWidth 
+ 1 : 0; 
3991     int h 
= m_numRows 
> 0 ? GetRowBottom(m_numRows 
- 1) + m_extraHeight 
+ 1 : 0; 
3993     // preserve (more or less) the previous position 
3995     GetViewStart( &x
, &y 
); 
3997     // maybe we don't need scrollbars at all? 
3999     // also adjust the position to be valid for the new scroll rangs 
4020     // do set scrollbar parameters 
4021     SetScrollbars( GRID_SCROLL_LINE_X
, GRID_SCROLL_LINE_Y
, 
4022                    GetScrollX(w
), GetScrollY(h
), x
, y
, 
4023                    GetBatchCount() != 0); 
4025     // if our OnSize() hadn't been called (it would if we have scrollbars), we 
4026     // still must reposition the children 
4031 void wxGrid::CalcWindowSizes() 
4034     GetClientSize( &cw
, &ch 
); 
4036     if ( m_cornerLabelWin
->IsShown() ) 
4037         m_cornerLabelWin
->SetSize( 0, 0, m_rowLabelWidth
, m_colLabelHeight 
); 
4039     if ( m_colLabelWin
->IsShown() ) 
4040         m_colLabelWin
->SetSize( m_rowLabelWidth
, 0, cw
-m_rowLabelWidth
, m_colLabelHeight
); 
4042     if ( m_rowLabelWin
->IsShown() ) 
4043         m_rowLabelWin
->SetSize( 0, m_colLabelHeight
, m_rowLabelWidth
, ch
-m_colLabelHeight
); 
4045     if ( m_gridWin
->IsShown() ) 
4046         m_gridWin
->SetSize( m_rowLabelWidth
, m_colLabelHeight
, cw
-m_rowLabelWidth
, ch
-m_colLabelHeight
); 
4050 // this is called when the grid table sends a message to say that it 
4051 // has been redimensioned 
4053 bool wxGrid::Redimension( wxGridTableMessage
& msg 
) 
4056     bool result 
= FALSE
; 
4058     // Clear the attribute cache as the attribute might refer to a different 
4059     // cell than stored in the cache after adding/removing rows/columns. 
4061     // By the same reasoning, the editor should be dismissed if columns are 
4062     // added or removed. And for consistency, it should IMHO always be 
4063     // removed, not only if the cell "underneath" it actually changes. 
4064     // For now, I intentionally do not save the editor's content as the 
4065     // cell it might want to save that stuff to might no longer exist. 
4066     HideCellEditControl(); 
4068     // if we were using the default widths/heights so far, we must change them 
4070     if ( m_colWidths
.IsEmpty() ) 
4075     if ( m_rowHeights
.IsEmpty() ) 
4081     switch ( msg
.GetId() ) 
4083         case wxGRIDTABLE_NOTIFY_ROWS_INSERTED
: 
4085             size_t pos 
= msg
.GetCommandInt(); 
4086             int numRows 
= msg
.GetCommandInt2(); 
4088             m_numRows 
+= numRows
; 
4090             if ( !m_rowHeights
.IsEmpty() ) 
4092                 m_rowHeights
.Insert( m_defaultRowHeight
, pos
, numRows 
); 
4093                 m_rowBottoms
.Insert( 0, pos
, numRows 
); 
4096                 if ( pos 
> 0 ) bottom 
= m_rowBottoms
[pos
-1]; 
4098                 for ( i 
= pos
;  i 
< m_numRows
;  i
++ ) 
4100                     bottom 
+= m_rowHeights
[i
]; 
4101                     m_rowBottoms
[i
] = bottom
; 
4104             if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
4106                 // if we have just inserted cols into an empty grid the current 
4107                 // cell will be undefined... 
4109                 SetCurrentCell( 0, 0 ); 
4113                 m_selection
->UpdateRows( pos
, numRows 
); 
4114             wxGridCellAttrProvider 
* attrProvider 
= m_table
->GetAttrProvider(); 
4116                 attrProvider
->UpdateAttrRows( pos
, numRows 
); 
4118             if ( !GetBatchCount() ) 
4121                 m_rowLabelWin
->Refresh(); 
4127         case wxGRIDTABLE_NOTIFY_ROWS_APPENDED
: 
4129             int numRows 
= msg
.GetCommandInt(); 
4130             int oldNumRows 
= m_numRows
; 
4131             m_numRows 
+= numRows
; 
4133             if ( !m_rowHeights
.IsEmpty() ) 
4135                 m_rowHeights
.Add( m_defaultRowHeight
, numRows 
); 
4136                 m_rowBottoms
.Add( 0, numRows 
); 
4139                 if ( oldNumRows 
> 0 ) bottom 
= m_rowBottoms
[oldNumRows
-1]; 
4141                 for ( i 
= oldNumRows
;  i 
< m_numRows
;  i
++ ) 
4143                     bottom 
+= m_rowHeights
[i
]; 
4144                     m_rowBottoms
[i
] = bottom
; 
4147             if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
4149                 // if we have just inserted cols into an empty grid the current 
4150                 // cell will be undefined... 
4152                 SetCurrentCell( 0, 0 ); 
4154             if ( !GetBatchCount() ) 
4157                 m_rowLabelWin
->Refresh(); 
4163         case wxGRIDTABLE_NOTIFY_ROWS_DELETED
: 
4165             size_t pos 
= msg
.GetCommandInt(); 
4166             int numRows 
= msg
.GetCommandInt2(); 
4167             m_numRows 
-= numRows
; 
4169             if ( !m_rowHeights
.IsEmpty() ) 
4171                 m_rowHeights
.RemoveAt( pos
, numRows 
); 
4172                 m_rowBottoms
.RemoveAt( pos
, numRows 
); 
4175                 for ( i 
= 0;  i 
< m_numRows
;  i
++ ) 
4177                     h 
+= m_rowHeights
[i
]; 
4178                     m_rowBottoms
[i
] = h
; 
4183                 m_currentCellCoords 
= wxGridNoCellCoords
; 
4187                 if ( m_currentCellCoords
.GetRow() >= m_numRows 
) 
4188                     m_currentCellCoords
.Set( 0, 0 ); 
4192                 m_selection
->UpdateRows( pos
, -((int)numRows
) ); 
4193             wxGridCellAttrProvider 
* attrProvider 
= m_table
->GetAttrProvider(); 
4195                 attrProvider
->UpdateAttrRows( pos
, -((int)numRows
) ); 
4196 // ifdef'd out following patch from Paul Gammans 
4198                 // No need to touch column attributes, unless we 
4199                 // removed _all_ rows, in this case, we remove 
4200                 // all column attributes. 
4201                 // I hate to do this here, but the 
4202                 // needed data is not available inside UpdateAttrRows. 
4203                 if ( !GetNumberRows() ) 
4204                     attrProvider
->UpdateAttrCols( 0, -GetNumberCols() ); 
4207             if ( !GetBatchCount() ) 
4210                 m_rowLabelWin
->Refresh(); 
4216         case wxGRIDTABLE_NOTIFY_COLS_INSERTED
: 
4218             size_t pos 
= msg
.GetCommandInt(); 
4219             int numCols 
= msg
.GetCommandInt2(); 
4220             m_numCols 
+= numCols
; 
4222             if ( !m_colWidths
.IsEmpty() ) 
4224                 m_colWidths
.Insert( m_defaultColWidth
, pos
, numCols 
); 
4225                 m_colRights
.Insert( 0, pos
, numCols 
); 
4228                 if ( pos 
> 0 ) right 
= m_colRights
[pos
-1]; 
4230                 for ( i 
= pos
;  i 
< m_numCols
;  i
++ ) 
4232                     right 
+= m_colWidths
[i
]; 
4233                     m_colRights
[i
] = right
; 
4236             if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
4238                 // if we have just inserted cols into an empty grid the current 
4239                 // cell will be undefined... 
4241                 SetCurrentCell( 0, 0 ); 
4245                 m_selection
->UpdateCols( pos
, numCols 
); 
4246             wxGridCellAttrProvider 
* attrProvider 
= m_table
->GetAttrProvider(); 
4248                 attrProvider
->UpdateAttrCols( pos
, numCols 
); 
4249             if ( !GetBatchCount() ) 
4252                 m_colLabelWin
->Refresh(); 
4259         case wxGRIDTABLE_NOTIFY_COLS_APPENDED
: 
4261             int numCols 
= msg
.GetCommandInt(); 
4262             int oldNumCols 
= m_numCols
; 
4263             m_numCols 
+= numCols
; 
4264             if ( !m_colWidths
.IsEmpty() ) 
4266                 m_colWidths
.Add( m_defaultColWidth
, numCols 
); 
4267                 m_colRights
.Add( 0, numCols 
); 
4270                 if ( oldNumCols 
> 0 ) right 
= m_colRights
[oldNumCols
-1]; 
4272                 for ( i 
= oldNumCols
;  i 
< m_numCols
;  i
++ ) 
4274                     right 
+= m_colWidths
[i
]; 
4275                     m_colRights
[i
] = right
; 
4278             if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
4280                 // if we have just inserted cols into an empty grid the current 
4281                 // cell will be undefined... 
4283                 SetCurrentCell( 0, 0 ); 
4285             if ( !GetBatchCount() ) 
4288                 m_colLabelWin
->Refresh(); 
4294         case wxGRIDTABLE_NOTIFY_COLS_DELETED
: 
4296             size_t pos 
= msg
.GetCommandInt(); 
4297             int numCols 
= msg
.GetCommandInt2(); 
4298             m_numCols 
-= numCols
; 
4300             if ( !m_colWidths
.IsEmpty() ) 
4302                 m_colWidths
.RemoveAt( pos
, numCols 
); 
4303                 m_colRights
.RemoveAt( pos
, numCols 
); 
4306                 for ( i 
= 0;  i 
< m_numCols
;  i
++ ) 
4308                     w 
+= m_colWidths
[i
]; 
4314                 m_currentCellCoords 
= wxGridNoCellCoords
; 
4318                 if ( m_currentCellCoords
.GetCol() >= m_numCols 
) 
4319                   m_currentCellCoords
.Set( 0, 0 ); 
4323                 m_selection
->UpdateCols( pos
, -((int)numCols
) ); 
4324             wxGridCellAttrProvider 
* attrProvider 
= m_table
->GetAttrProvider(); 
4326                 attrProvider
->UpdateAttrCols( pos
, -((int)numCols
) ); 
4327 // ifdef'd out following patch from Paul Gammans 
4329                 // No need to touch row attributes, unless we 
4330                 // removed _all_ columns, in this case, we remove 
4331                 // all row attributes. 
4332                 // I hate to do this here, but the 
4333                 // needed data is not available inside UpdateAttrCols. 
4334                 if ( !GetNumberCols() ) 
4335                     attrProvider
->UpdateAttrRows( 0, -GetNumberRows() ); 
4338             if ( !GetBatchCount() ) 
4341                 m_colLabelWin
->Refresh(); 
4348     if (result 
&& !GetBatchCount() ) 
4349         m_gridWin
->Refresh(); 
4354 wxArrayInt 
wxGrid::CalcRowLabelsExposed( const wxRegion
& reg 
) 
4356     wxRegionIterator 
iter( reg 
); 
4359     wxArrayInt  rowlabels
; 
4366         // TODO: remove this when we can... 
4367         // There is a bug in wxMotif that gives garbage update 
4368         // rectangles if you jump-scroll a long way by clicking the 
4369         // scrollbar with middle button.  This is a work-around 
4371 #if defined(__WXMOTIF__) 
4373         m_gridWin
->GetClientSize( &cw
, &ch 
); 
4374         if ( r
.GetTop() > ch 
) r
.SetTop( 0 ); 
4375         r
.SetBottom( wxMin( r
.GetBottom(), ch 
) ); 
4378         // logical bounds of update region 
4381         CalcUnscrolledPosition( 0, r
.GetTop(), &dummy
, &top 
); 
4382         CalcUnscrolledPosition( 0, r
.GetBottom(), &dummy
, &bottom 
); 
4384         // find the row labels within these bounds 
4387         for ( row 
= internalYToRow(top
);  row 
< m_numRows
;  row
++ ) 
4389             if ( GetRowBottom(row
) < top 
) 
4392             if ( GetRowTop(row
) > bottom 
) 
4395             rowlabels
.Add( row 
); 
4405 wxArrayInt 
wxGrid::CalcColLabelsExposed( const wxRegion
& reg 
) 
4407     wxRegionIterator 
iter( reg 
); 
4410     wxArrayInt colLabels
; 
4417         // TODO: remove this when we can... 
4418         // There is a bug in wxMotif that gives garbage update 
4419         // rectangles if you jump-scroll a long way by clicking the 
4420         // scrollbar with middle button.  This is a work-around 
4422 #if defined(__WXMOTIF__) 
4424         m_gridWin
->GetClientSize( &cw
, &ch 
); 
4425         if ( r
.GetLeft() > cw 
) r
.SetLeft( 0 ); 
4426         r
.SetRight( wxMin( r
.GetRight(), cw 
) ); 
4429         // logical bounds of update region 
4432         CalcUnscrolledPosition( r
.GetLeft(), 0, &left
, &dummy 
); 
4433         CalcUnscrolledPosition( r
.GetRight(), 0, &right
, &dummy 
); 
4435         // find the cells within these bounds 
4438         for ( col 
= internalXToCol(left
);  col 
< m_numCols
;  col
++ ) 
4440             if ( GetColRight(col
) < left 
) 
4443             if ( GetColLeft(col
) > right 
) 
4446             colLabels
.Add( col 
); 
4455 wxGridCellCoordsArray 
wxGrid::CalcCellsExposed( const wxRegion
& reg 
) 
4457     wxRegionIterator 
iter( reg 
); 
4460     wxGridCellCoordsArray  cellsExposed
; 
4462     int left
, top
, right
, bottom
; 
4467         // TODO: remove this when we can... 
4468         // There is a bug in wxMotif that gives garbage update 
4469         // rectangles if you jump-scroll a long way by clicking the 
4470         // scrollbar with middle button.  This is a work-around 
4472 #if defined(__WXMOTIF__) 
4474         m_gridWin
->GetClientSize( &cw
, &ch 
); 
4475         if ( r
.GetTop() > ch 
) r
.SetTop( 0 ); 
4476         if ( r
.GetLeft() > cw 
) r
.SetLeft( 0 ); 
4477         r
.SetRight( wxMin( r
.GetRight(), cw 
) ); 
4478         r
.SetBottom( wxMin( r
.GetBottom(), ch 
) ); 
4481         // logical bounds of update region 
4483         CalcUnscrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top 
); 
4484         CalcUnscrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom 
); 
4486         // find the cells within these bounds 
4489         for ( row 
= internalYToRow(top
);  row 
< m_numRows
;  row
++ ) 
4491             if ( GetRowBottom(row
) <= top 
) 
4494             if ( GetRowTop(row
) > bottom 
) 
4497             for ( col 
= internalXToCol(left
);  col 
< m_numCols
;  col
++ ) 
4499                 if ( GetColRight(col
) <= left 
) 
4502                 if ( GetColLeft(col
) > right 
) 
4505                 cellsExposed
.Add( wxGridCellCoords( row
, col 
) ); 
4512     return cellsExposed
; 
4516 void wxGrid::ProcessRowLabelMouseEvent( wxMouseEvent
& event 
) 
4519     wxPoint 
pos( event
.GetPosition() ); 
4520     CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y 
); 
4522     if ( event
.Dragging() ) 
4526             m_isDragging 
= TRUE
; 
4527             m_rowLabelWin
->CaptureMouse(); 
4530         if ( event
.LeftIsDown() ) 
4532             switch( m_cursorMode 
) 
4534                 case WXGRID_CURSOR_RESIZE_ROW
: 
4536                     int cw
, ch
, left
, dummy
; 
4537                     m_gridWin
->GetClientSize( &cw
, &ch 
); 
4538                     CalcUnscrolledPosition( 0, 0, &left
, &dummy 
); 
4540                     wxClientDC 
dc( m_gridWin 
); 
4543                                GetRowTop(m_dragRowOrCol
) + 
4544                                GetRowMinimalHeight(m_dragRowOrCol
) ); 
4545                     dc
.SetLogicalFunction(wxINVERT
); 
4546                     if ( m_dragLastPos 
>= 0 ) 
4548                         dc
.DrawLine( left
, m_dragLastPos
, left
+cw
, m_dragLastPos 
); 
4550                     dc
.DrawLine( left
, y
, left
+cw
, y 
); 
4555                 case WXGRID_CURSOR_SELECT_ROW
: 
4556                     if ( (row 
= YToRow( y 
)) >= 0 ) 
4560                             m_selection
->SelectRow( row
, 
4561                                                     event
.ControlDown(), 
4568                 // default label to suppress warnings about "enumeration value 
4569                 // 'xxx' not handled in switch 
4577     if ( m_isDragging 
&& (event
.Entering() || event
.Leaving()) ) 
4582         if (m_rowLabelWin
->HasCapture()) m_rowLabelWin
->ReleaseMouse(); 
4583         m_isDragging 
= FALSE
; 
4586     // ------------ Entering or leaving the window 
4588     if ( event
.Entering() || event
.Leaving() ) 
4590         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_rowLabelWin
); 
4594     // ------------ Left button pressed 
4596     else if ( event
.LeftDown() ) 
4598         // don't send a label click event for a hit on the 
4599         // edge of the row label - this is probably the user 
4600         // wanting to resize the row 
4602         if ( YToEdgeOfRow(y
) < 0 ) 
4606                  !SendEvent( wxEVT_GRID_LABEL_LEFT_CLICK
, row
, -1, event 
) ) 
4608                 if ( !event
.ShiftDown() && !event
.ControlDown() ) 
4612                     if ( event
.ShiftDown() ) 
4614                         m_selection
->SelectBlock( m_currentCellCoords
.GetRow(), 
4617                                                   GetNumberCols() - 1, 
4618                                                   event
.ControlDown(), 
4625                         m_selection
->SelectRow( row
, 
4626                                                 event
.ControlDown(), 
4633                 ChangeCursorMode(WXGRID_CURSOR_SELECT_ROW
, m_rowLabelWin
); 
4638             // starting to drag-resize a row 
4640             if ( CanDragRowSize() ) 
4641                 ChangeCursorMode(WXGRID_CURSOR_RESIZE_ROW
, m_rowLabelWin
); 
4646     // ------------ Left double click 
4648     else if (event
.LeftDClick() ) 
4650         if ( YToEdgeOfRow(y
) < 0 ) 
4653             SendEvent( wxEVT_GRID_LABEL_LEFT_DCLICK
, row
, -1, event 
); 
4658     // ------------ Left button released 
4660     else if ( event
.LeftUp() ) 
4662         if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_ROW 
) 
4664             DoEndDragResizeRow(); 
4666             // Note: we are ending the event *after* doing 
4667             // default processing in this case 
4669             SendEvent( wxEVT_GRID_ROW_SIZE
, m_dragRowOrCol
, -1, event 
); 
4672         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_rowLabelWin
); 
4677     // ------------ Right button down 
4679     else if ( event
.RightDown() ) 
4682         if ( !SendEvent( wxEVT_GRID_LABEL_RIGHT_CLICK
, row
, -1, event 
) ) 
4684             // no default action at the moment 
4689     // ------------ Right double click 
4691     else if ( event
.RightDClick() ) 
4694         if ( !SendEvent( wxEVT_GRID_LABEL_RIGHT_DCLICK
, row
, -1, event 
) ) 
4696             // no default action at the moment 
4701     // ------------ No buttons down and mouse moving 
4703     else if ( event
.Moving() ) 
4705         m_dragRowOrCol 
= YToEdgeOfRow( y 
); 
4706         if ( m_dragRowOrCol 
>= 0 ) 
4708             if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
4710                 // don't capture the mouse yet 
4711                 if ( CanDragRowSize() ) 
4712                     ChangeCursorMode(WXGRID_CURSOR_RESIZE_ROW
, m_rowLabelWin
, FALSE
); 
4715         else if ( m_cursorMode 
!= WXGRID_CURSOR_SELECT_CELL 
) 
4717             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_rowLabelWin
, FALSE
); 
4723 void wxGrid::ProcessColLabelMouseEvent( wxMouseEvent
& event 
) 
4726     wxPoint 
pos( event
.GetPosition() ); 
4727     CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y 
); 
4729     if ( event
.Dragging() ) 
4733             m_isDragging 
= TRUE
; 
4734             m_colLabelWin
->CaptureMouse(); 
4737         if ( event
.LeftIsDown() ) 
4739             switch( m_cursorMode 
) 
4741                 case WXGRID_CURSOR_RESIZE_COL
: 
4743                     int cw
, ch
, dummy
, top
; 
4744                     m_gridWin
->GetClientSize( &cw
, &ch 
); 
4745                     CalcUnscrolledPosition( 0, 0, &dummy
, &top 
); 
4747                     wxClientDC 
dc( m_gridWin 
); 
4750                     x 
= wxMax( x
, GetColLeft(m_dragRowOrCol
) + 
4751                                   GetColMinimalWidth(m_dragRowOrCol
)); 
4752                     dc
.SetLogicalFunction(wxINVERT
); 
4753                     if ( m_dragLastPos 
>= 0 ) 
4755                         dc
.DrawLine( m_dragLastPos
, top
, m_dragLastPos
, top
+ch 
); 
4757                     dc
.DrawLine( x
, top
, x
, top
+ch 
); 
4762                 case WXGRID_CURSOR_SELECT_COL
: 
4763                     if ( (col 
= XToCol( x 
)) >= 0 ) 
4767                             m_selection
->SelectCol( col
, 
4768                                                     event
.ControlDown(), 
4775                 // default label to suppress warnings about "enumeration value 
4776                 // 'xxx' not handled in switch 
4784     if ( m_isDragging 
&& (event
.Entering() || event
.Leaving()) ) 
4789         if (m_colLabelWin
->HasCapture()) m_colLabelWin
->ReleaseMouse(); 
4790         m_isDragging 
= FALSE
; 
4793     // ------------ Entering or leaving the window 
4795     if ( event
.Entering() || event
.Leaving() ) 
4797         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
); 
4801     // ------------ Left button pressed 
4803     else if ( event
.LeftDown() ) 
4805         // don't send a label click event for a hit on the 
4806         // edge of the col label - this is probably the user 
4807         // wanting to resize the col 
4809         if ( XToEdgeOfCol(x
) < 0 ) 
4813                  !SendEvent( wxEVT_GRID_LABEL_LEFT_CLICK
, -1, col
, event 
) ) 
4815                 if ( !event
.ShiftDown() && !event
.ControlDown() ) 
4819                     if ( event
.ShiftDown() ) 
4821                         m_selection
->SelectBlock( 0, 
4822                                                   m_currentCellCoords
.GetCol(), 
4823                                                   GetNumberRows() - 1, col
, 
4824                                                   event
.ControlDown(), 
4831                         m_selection
->SelectCol( col
, 
4832                                                 event
.ControlDown(), 
4839                 ChangeCursorMode(WXGRID_CURSOR_SELECT_COL
, m_colLabelWin
); 
4844             // starting to drag-resize a col 
4846             if ( CanDragColSize() ) 
4847                 ChangeCursorMode(WXGRID_CURSOR_RESIZE_COL
, m_colLabelWin
); 
4852     // ------------ Left double click 
4854     if ( event
.LeftDClick() ) 
4856         if ( XToEdgeOfCol(x
) < 0 ) 
4859             SendEvent( wxEVT_GRID_LABEL_LEFT_DCLICK
, -1, col
, event 
); 
4864     // ------------ Left button released 
4866     else if ( event
.LeftUp() ) 
4868         if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_COL 
) 
4870             DoEndDragResizeCol(); 
4872             // Note: we are ending the event *after* doing 
4873             // default processing in this case 
4875             SendEvent( wxEVT_GRID_COL_SIZE
, -1, m_dragRowOrCol
, event 
); 
4878         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
); 
4883     // ------------ Right button down 
4885     else if ( event
.RightDown() ) 
4888         if ( !SendEvent( wxEVT_GRID_LABEL_RIGHT_CLICK
, -1, col
, event 
) ) 
4890             // no default action at the moment 
4895     // ------------ Right double click 
4897     else if ( event
.RightDClick() ) 
4900         if ( !SendEvent( wxEVT_GRID_LABEL_RIGHT_DCLICK
, -1, col
, event 
) ) 
4902             // no default action at the moment 
4907     // ------------ No buttons down and mouse moving 
4909     else if ( event
.Moving() ) 
4911         m_dragRowOrCol 
= XToEdgeOfCol( x 
); 
4912         if ( m_dragRowOrCol 
>= 0 ) 
4914             if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
4916                 // don't capture the cursor yet 
4917                 if ( CanDragColSize() ) 
4918                     ChangeCursorMode(WXGRID_CURSOR_RESIZE_COL
, m_colLabelWin
, FALSE
); 
4921         else if ( m_cursorMode 
!= WXGRID_CURSOR_SELECT_CELL 
) 
4923             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
, FALSE
); 
4929 void wxGrid::ProcessCornerLabelMouseEvent( wxMouseEvent
& event 
) 
4931     if ( event
.LeftDown() ) 
4933         // indicate corner label by having both row and 
4936         if ( !SendEvent( wxEVT_GRID_LABEL_LEFT_CLICK
, -1, -1, event 
) ) 
4942     else if ( event
.LeftDClick() ) 
4944         SendEvent( wxEVT_GRID_LABEL_LEFT_DCLICK
, -1, -1, event 
); 
4947     else if ( event
.RightDown() ) 
4949         if ( !SendEvent( wxEVT_GRID_LABEL_RIGHT_CLICK
, -1, -1, event 
) ) 
4951             // no default action at the moment 
4955     else if ( event
.RightDClick() ) 
4957         if ( !SendEvent( wxEVT_GRID_LABEL_RIGHT_DCLICK
, -1, -1, event 
) ) 
4959             // no default action at the moment 
4964 void wxGrid::ChangeCursorMode(CursorMode mode
, 
4969     static const wxChar 
*cursorModes
[] = 
4978     wxLogTrace(_T("grid"), 
4979                _T("wxGrid cursor mode (mouse capture for %s): %s -> %s"), 
4980                win 
== m_colLabelWin 
? _T("colLabelWin") 
4981                                     : win 
? _T("rowLabelWin") 
4983                cursorModes
[m_cursorMode
], cursorModes
[mode
]); 
4984 #endif // __WXDEBUG__ 
4986     if ( mode 
== m_cursorMode 
&& 
4987          win 
== m_winCapture 
&& 
4988          captureMouse 
== (m_winCapture 
!= NULL
)) 
4993         // by default use the grid itself 
4999         if (m_winCapture
->HasCapture()) m_winCapture
->ReleaseMouse(); 
5000         m_winCapture 
= (wxWindow 
*)NULL
; 
5003     m_cursorMode 
= mode
; 
5005     switch ( m_cursorMode 
) 
5007         case WXGRID_CURSOR_RESIZE_ROW
: 
5008             win
->SetCursor( m_rowResizeCursor 
); 
5011         case WXGRID_CURSOR_RESIZE_COL
: 
5012             win
->SetCursor( m_colResizeCursor 
); 
5016             win
->SetCursor( *wxSTANDARD_CURSOR 
); 
5019     // we need to capture mouse when resizing 
5020     bool resize 
= m_cursorMode 
== WXGRID_CURSOR_RESIZE_ROW 
|| 
5021                   m_cursorMode 
== WXGRID_CURSOR_RESIZE_COL
; 
5023     if ( captureMouse 
&& resize 
) 
5025         win
->CaptureMouse(); 
5030 void wxGrid::ProcessGridCellMouseEvent( wxMouseEvent
& event 
) 
5033     wxPoint 
pos( event
.GetPosition() ); 
5034     CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y 
); 
5036     wxGridCellCoords coords
; 
5037     XYToCell( x
, y
, coords 
); 
5039     int cell_rows
, cell_cols
; 
5040     GetCellSize( coords
.GetRow(), coords
.GetCol(), &cell_rows
, &cell_cols 
); 
5041     if ((cell_rows 
< 0) || (cell_cols 
< 0)) 
5043         coords
.SetRow(coords
.GetRow() + cell_rows
); 
5044         coords
.SetCol(coords
.GetCol() + cell_cols
); 
5047     if ( event
.Dragging() ) 
5049         //wxLogDebug("pos(%d, %d) coords(%d, %d)", pos.x, pos.y, coords.GetRow(), coords.GetCol()); 
5051         // Don't start doing anything until the mouse has been drug at 
5052         // least 3 pixels in any direction... 
5055             if (m_startDragPos 
== wxDefaultPosition
) 
5057                 m_startDragPos 
= pos
; 
5060             if (abs(m_startDragPos
.x 
- pos
.x
) < 4 && abs(m_startDragPos
.y 
- pos
.y
) < 4) 
5064         m_isDragging 
= TRUE
; 
5065         if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
5067             // Hide the edit control, so it 
5068             // won't interfer with drag-shrinking. 
5069             if ( IsCellEditControlShown() ) 
5071                 HideCellEditControl(); 
5072                 SaveEditControlValue(); 
5075             // Have we captured the mouse yet? 
5078                 m_winCapture 
= m_gridWin
; 
5079                 m_winCapture
->CaptureMouse(); 
5082             if ( coords 
!= wxGridNoCellCoords 
) 
5084                 if ( event
.ControlDown() ) 
5086                     if ( m_selectingKeyboard 
== wxGridNoCellCoords
) 
5087                         m_selectingKeyboard 
= coords
; 
5088                     HighlightBlock ( m_selectingKeyboard
, coords 
); 
5092                     if ( !IsSelection() ) 
5094                         HighlightBlock( coords
, coords 
); 
5098                         HighlightBlock( m_currentCellCoords
, coords 
); 
5102                 if (! IsVisible(coords
)) 
5104                     MakeCellVisible(coords
); 
5105                     // TODO: need to introduce a delay or something here.  The 
5106                     // scrolling is way to fast, at least on MSW - also on GTK. 
5110         else if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_ROW 
) 
5112             int cw
, ch
, left
, dummy
; 
5113             m_gridWin
->GetClientSize( &cw
, &ch 
); 
5114             CalcUnscrolledPosition( 0, 0, &left
, &dummy 
); 
5116             wxClientDC 
dc( m_gridWin 
); 
5118             y 
= wxMax( y
, GetRowTop(m_dragRowOrCol
) + 
5119                           GetRowMinimalHeight(m_dragRowOrCol
) ); 
5120             dc
.SetLogicalFunction(wxINVERT
); 
5121             if ( m_dragLastPos 
>= 0 ) 
5123                 dc
.DrawLine( left
, m_dragLastPos
, left
+cw
, m_dragLastPos 
); 
5125             dc
.DrawLine( left
, y
, left
+cw
, y 
); 
5128         else if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_COL 
) 
5130             int cw
, ch
, dummy
, top
; 
5131             m_gridWin
->GetClientSize( &cw
, &ch 
); 
5132             CalcUnscrolledPosition( 0, 0, &dummy
, &top 
); 
5134             wxClientDC 
dc( m_gridWin 
); 
5136             x 
= wxMax( x
, GetColLeft(m_dragRowOrCol
) + 
5137                           GetColMinimalWidth(m_dragRowOrCol
) ); 
5138             dc
.SetLogicalFunction(wxINVERT
); 
5139             if ( m_dragLastPos 
>= 0 ) 
5141                 dc
.DrawLine( m_dragLastPos
, top
, m_dragLastPos
, top
+ch 
); 
5143             dc
.DrawLine( x
, top
, x
, top
+ch 
); 
5150     m_isDragging 
= FALSE
; 
5151     m_startDragPos 
= wxDefaultPosition
; 
5153     // VZ: if we do this, the mode is reset to WXGRID_CURSOR_SELECT_CELL 
5154     //     immediately after it becomes WXGRID_CURSOR_RESIZE_ROW/COL under 
5157     if ( event
.Entering() || event
.Leaving() ) 
5159         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5160         m_gridWin
->SetCursor( *wxSTANDARD_CURSOR 
); 
5165     // ------------ Left button pressed 
5167     if ( event
.LeftDown() && coords 
!= wxGridNoCellCoords 
) 
5169         if ( !SendEvent( wxEVT_GRID_CELL_LEFT_CLICK
, 
5174             if ( !event
.ControlDown() ) 
5176             if ( event
.ShiftDown() ) 
5180                     m_selection
->SelectBlock( m_currentCellCoords
.GetRow(), 
5181                                               m_currentCellCoords
.GetCol(), 
5184                                               event
.ControlDown(), 
5190             else if ( XToEdgeOfCol(x
) < 0  && 
5191                       YToEdgeOfRow(y
) < 0 ) 
5193                 DisableCellEditControl(); 
5194                 MakeCellVisible( coords 
); 
5196                 // if this is the second click on this cell then start 
5198                 if ( m_waitForSlowClick 
&& 
5199                      (coords 
== m_currentCellCoords
) && 
5200                      CanEnableCellControl()) 
5202                     EnableCellEditControl(); 
5204                     wxGridCellAttr
* attr 
= GetCellAttr(m_currentCellCoords
); 
5205                     wxGridCellEditor 
*editor 
= attr
->GetEditor(this, 
5208                     editor
->StartingClick(); 
5212                     m_waitForSlowClick 
= FALSE
; 
5216                     if ( event
.ControlDown() ) 
5220                             m_selection
->ToggleCellSelection( coords
.GetRow(), 
5222                                                               event
.ControlDown(), 
5227                         m_selectingTopLeft 
= wxGridNoCellCoords
; 
5228                         m_selectingBottomRight 
= wxGridNoCellCoords
; 
5229                         m_selectingKeyboard 
= coords
; 
5233                         SetCurrentCell( coords 
); 
5236                             if ( m_selection
->GetSelectionMode() != 
5237                                     wxGrid::wxGridSelectCells 
) 
5239                                 HighlightBlock( coords
, coords 
); 
5243                     m_waitForSlowClick 
= TRUE
; 
5250     // ------------ Left double click 
5252     else if ( event
.LeftDClick() && coords 
!= wxGridNoCellCoords 
) 
5254         DisableCellEditControl(); 
5256         if ( XToEdgeOfCol(x
) < 0  &&  YToEdgeOfRow(y
) < 0 ) 
5258             SendEvent( wxEVT_GRID_CELL_LEFT_DCLICK
, 
5266     // ------------ Left button released 
5268     else if ( event
.LeftUp() ) 
5270         if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
5272             if ( m_selectingTopLeft 
!= wxGridNoCellCoords 
&& 
5273                  m_selectingBottomRight 
!= wxGridNoCellCoords 
) 
5277                     if (m_winCapture
->HasCapture()) m_winCapture
->ReleaseMouse(); 
5278                     m_winCapture 
= NULL
; 
5283                     m_selection
->SelectBlock( m_selectingTopLeft
.GetRow(), 
5284                                               m_selectingTopLeft
.GetCol(), 
5285                                               m_selectingBottomRight
.GetRow(), 
5286                                               m_selectingBottomRight
.GetCol(), 
5287                                               event
.ControlDown(), 
5293                 m_selectingTopLeft 
= wxGridNoCellCoords
; 
5294                 m_selectingBottomRight 
= wxGridNoCellCoords
; 
5297             // Show the edit control, if it has been hidden for 
5299             ShowCellEditControl(); 
5301         else if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_ROW 
) 
5303             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5304             DoEndDragResizeRow(); 
5306             // Note: we are ending the event *after* doing 
5307             // default processing in this case 
5309             SendEvent( wxEVT_GRID_ROW_SIZE
, m_dragRowOrCol
, -1, event 
); 
5311         else if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_COL 
) 
5313             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5314             DoEndDragResizeCol(); 
5316             // Note: we are ending the event *after* doing 
5317             // default processing in this case 
5319             SendEvent( wxEVT_GRID_COL_SIZE
, -1, m_dragRowOrCol
, event 
); 
5326     // ------------ Right button down 
5328     else if ( event
.RightDown() && coords 
!= wxGridNoCellCoords 
) 
5330         DisableCellEditControl(); 
5331         if ( !SendEvent( wxEVT_GRID_CELL_RIGHT_CLICK
, 
5336             // no default action at the moment 
5341     // ------------ Right double click 
5343     else if ( event
.RightDClick() && coords 
!= wxGridNoCellCoords 
) 
5345         DisableCellEditControl(); 
5346         if ( !SendEvent( wxEVT_GRID_CELL_RIGHT_DCLICK
, 
5351             // no default action at the moment 
5355     // ------------ Moving and no button action 
5357     else if ( event
.Moving() && !event
.IsButton() ) 
5359         if( coords
.GetRow() < 0 || coords
.GetCol() < 0 ) 
5361             // out of grid cell area 
5362             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5366         int dragRow 
= YToEdgeOfRow( y 
); 
5367         int dragCol 
= XToEdgeOfCol( x 
); 
5369         // Dragging on the corner of a cell to resize in both 
5370         // directions is not implemented yet... 
5372         if ( dragRow 
>= 0  &&  dragCol 
>= 0 ) 
5374             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5380             m_dragRowOrCol 
= dragRow
; 
5382             if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
5384                 if ( CanDragRowSize() && CanDragGridSize() ) 
5385                     ChangeCursorMode(WXGRID_CURSOR_RESIZE_ROW
); 
5390                 m_dragRowOrCol 
= dragCol
; 
5398             m_dragRowOrCol 
= dragCol
; 
5400             if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
5402                 if ( CanDragColSize() && CanDragGridSize() ) 
5403                     ChangeCursorMode(WXGRID_CURSOR_RESIZE_COL
); 
5409         // Neither on a row or col edge 
5411         if ( m_cursorMode 
!= WXGRID_CURSOR_SELECT_CELL 
) 
5413             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5419 void wxGrid::DoEndDragResizeRow() 
5421     if ( m_dragLastPos 
>= 0 ) 
5423         // erase the last line and resize the row 
5425         int cw
, ch
, left
, dummy
; 
5426         m_gridWin
->GetClientSize( &cw
, &ch 
); 
5427         CalcUnscrolledPosition( 0, 0, &left
, &dummy 
); 
5429         wxClientDC 
dc( m_gridWin 
); 
5431         dc
.SetLogicalFunction( wxINVERT 
); 
5432         dc
.DrawLine( left
, m_dragLastPos
, left
+cw
, m_dragLastPos 
); 
5433         HideCellEditControl(); 
5434         SaveEditControlValue(); 
5436         int rowTop 
= GetRowTop(m_dragRowOrCol
); 
5437         SetRowSize( m_dragRowOrCol
, 
5438                     wxMax( m_dragLastPos 
- rowTop
, WXGRID_MIN_ROW_HEIGHT 
) ); 
5440         if ( !GetBatchCount() ) 
5442             // Only needed to get the correct rect.y: 
5443             wxRect 
rect ( CellToRect( m_dragRowOrCol
, 0 ) ); 
5445             CalcScrolledPosition(0, rect
.y
, &dummy
, &rect
.y
); 
5446             rect
.width 
= m_rowLabelWidth
; 
5447             rect
.height 
= ch 
- rect
.y
; 
5448             m_rowLabelWin
->Refresh( TRUE
, &rect 
); 
5450             // if there is a multicell block, paint all of it 
5453                 int i
, cell_rows
, cell_cols
, subtract_rows 
= 0; 
5454                 int leftCol 
= XToCol(left
); 
5455                 int rightCol 
= XToCol(left
+cw
); 
5458                     if (rightCol 
< 0) rightCol 
= m_numCols
; 
5459                     for (i
=leftCol
; i
<rightCol
; i
++) 
5461                         GetCellSize(m_dragRowOrCol
, i
, &cell_rows
, &cell_cols
); 
5462                         if (cell_rows 
< subtract_rows
) 
5463                             subtract_rows 
= cell_rows
; 
5465                     rect
.y 
= GetRowTop(m_dragRowOrCol 
+ subtract_rows
); 
5466                     CalcScrolledPosition(0, rect
.y
, &dummy
, &rect
.y
); 
5467                     rect
.height 
= ch 
- rect
.y
; 
5470             m_gridWin
->Refresh( FALSE
, &rect 
); 
5473         ShowCellEditControl(); 
5478 void wxGrid::DoEndDragResizeCol() 
5480     if ( m_dragLastPos 
>= 0 ) 
5482         // erase the last line and resize the col 
5484         int cw
, ch
, dummy
, top
; 
5485         m_gridWin
->GetClientSize( &cw
, &ch 
); 
5486         CalcUnscrolledPosition( 0, 0, &dummy
, &top 
); 
5488         wxClientDC 
dc( m_gridWin 
); 
5490         dc
.SetLogicalFunction( wxINVERT 
); 
5491         dc
.DrawLine( m_dragLastPos
, top
, m_dragLastPos
, top
+ch 
); 
5492         HideCellEditControl(); 
5493         SaveEditControlValue(); 
5495         int colLeft 
= GetColLeft(m_dragRowOrCol
); 
5496         SetColSize( m_dragRowOrCol
, 
5497                     wxMax( m_dragLastPos 
- colLeft
, 
5498                            GetColMinimalWidth(m_dragRowOrCol
) ) ); 
5500         if ( !GetBatchCount() ) 
5502             // Only needed to get the correct rect.x: 
5503             wxRect 
rect ( CellToRect( 0, m_dragRowOrCol 
) ); 
5505             CalcScrolledPosition(rect
.x
, 0, &rect
.x
, &dummy
); 
5506             rect
.width 
= cw 
- rect
.x
; 
5507             rect
.height 
= m_colLabelHeight
; 
5508             m_colLabelWin
->Refresh( TRUE
, &rect 
); 
5510             // if there is a multicell block, paint all of it 
5513                 int i
, cell_rows
, cell_cols
, subtract_cols 
= 0; 
5514                 int topRow 
= YToRow(top
); 
5515                 int bottomRow 
= YToRow(top
+cw
); 
5518                     if (bottomRow 
< 0) bottomRow 
= m_numRows
; 
5519                     for (i
=topRow
; i
<bottomRow
; i
++) 
5521                         GetCellSize(i
, m_dragRowOrCol
, &cell_rows
, &cell_cols
); 
5522                         if (cell_cols 
< subtract_cols
) 
5523                             subtract_cols 
= cell_cols
; 
5525                     rect
.x 
= GetColLeft(m_dragRowOrCol 
+ subtract_cols
); 
5526                     CalcScrolledPosition(rect
.x
, 0, &rect
.x
, &dummy
); 
5527                     rect
.width 
= cw 
- rect
.x
; 
5530             m_gridWin
->Refresh( FALSE
, &rect 
); 
5533         ShowCellEditControl(); 
5540 // ------ interaction with data model 
5542 bool wxGrid::ProcessTableMessage( wxGridTableMessage
& msg 
) 
5544     switch ( msg
.GetId() ) 
5546         case wxGRIDTABLE_REQUEST_VIEW_GET_VALUES
: 
5547             return GetModelValues(); 
5549         case wxGRIDTABLE_REQUEST_VIEW_SEND_VALUES
: 
5550             return SetModelValues(); 
5552         case wxGRIDTABLE_NOTIFY_ROWS_INSERTED
: 
5553         case wxGRIDTABLE_NOTIFY_ROWS_APPENDED
: 
5554         case wxGRIDTABLE_NOTIFY_ROWS_DELETED
: 
5555         case wxGRIDTABLE_NOTIFY_COLS_INSERTED
: 
5556         case wxGRIDTABLE_NOTIFY_COLS_APPENDED
: 
5557         case wxGRIDTABLE_NOTIFY_COLS_DELETED
: 
5558             return Redimension( msg 
); 
5567 // The behaviour of this function depends on the grid table class 
5568 // Clear() function.  For the default wxGridStringTable class the 
5569 // behavious is to replace all cell contents with wxEmptyString but 
5570 // not to change the number of rows or cols. 
5572 void wxGrid::ClearGrid() 
5576         if (IsCellEditControlEnabled()) 
5577             DisableCellEditControl(); 
5580         if ( !GetBatchCount() ) m_gridWin
->Refresh(); 
5585 bool wxGrid::InsertRows( int pos
, int numRows
, bool WXUNUSED(updateLabels
) ) 
5587     // TODO: something with updateLabels flag 
5591         wxFAIL_MSG( wxT("Called wxGrid::InsertRows() before calling CreateGrid()") ); 
5597         if (IsCellEditControlEnabled()) 
5598             DisableCellEditControl(); 
5600         return m_table
->InsertRows( pos
, numRows 
); 
5602         // the table will have sent the results of the insert row 
5603         // operation to this view object as a grid table message 
5609 bool wxGrid::AppendRows( int numRows
, bool WXUNUSED(updateLabels
) ) 
5611     // TODO: something with updateLabels flag 
5615         wxFAIL_MSG( wxT("Called wxGrid::AppendRows() before calling CreateGrid()") ); 
5619     return ( m_table 
&& m_table
->AppendRows( numRows 
) ); 
5620     // the table will have sent the results of the append row 
5621     // operation to this view object as a grid table message 
5625 bool wxGrid::DeleteRows( int pos
, int numRows
, bool WXUNUSED(updateLabels
) ) 
5627     // TODO: something with updateLabels flag 
5631         wxFAIL_MSG( wxT("Called wxGrid::DeleteRows() before calling CreateGrid()") ); 
5637         if (IsCellEditControlEnabled()) 
5638             DisableCellEditControl(); 
5640         return (m_table
->DeleteRows( pos
, numRows 
)); 
5641         // the table will have sent the results of the delete row 
5642         // operation to this view object as a grid table message 
5648 bool wxGrid::InsertCols( int pos
, int numCols
, bool WXUNUSED(updateLabels
) ) 
5650     // TODO: something with updateLabels flag 
5654         wxFAIL_MSG( wxT("Called wxGrid::InsertCols() before calling CreateGrid()") ); 
5660         if (IsCellEditControlEnabled()) 
5661             DisableCellEditControl(); 
5663         return m_table
->InsertCols( pos
, numCols 
); 
5664         // the table will have sent the results of the insert col 
5665         // operation to this view object as a grid table message 
5671 bool wxGrid::AppendCols( int numCols
, bool WXUNUSED(updateLabels
) ) 
5673     // TODO: something with updateLabels flag 
5677         wxFAIL_MSG( wxT("Called wxGrid::AppendCols() before calling CreateGrid()") ); 
5681     return ( m_table 
&& m_table
->AppendCols( numCols 
) ); 
5682     // the table will have sent the results of the append col 
5683     // operation to this view object as a grid table message 
5687 bool wxGrid::DeleteCols( int pos
, int numCols
, bool WXUNUSED(updateLabels
) ) 
5689     // TODO: something with updateLabels flag 
5693         wxFAIL_MSG( wxT("Called wxGrid::DeleteCols() before calling CreateGrid()") ); 
5699         if (IsCellEditControlEnabled()) 
5700             DisableCellEditControl(); 
5702         return ( m_table
->DeleteCols( pos
, numCols 
) ); 
5703         // the table will have sent the results of the delete col 
5704         // operation to this view object as a grid table message 
5712 // ----- event handlers 
5715 // Generate a grid event based on a mouse event and 
5716 // return the result of ProcessEvent() 
5718 int wxGrid::SendEvent( const wxEventType type
, 
5720                         wxMouseEvent
& mouseEv 
) 
5725    if ( type 
== wxEVT_GRID_ROW_SIZE 
|| type 
== wxEVT_GRID_COL_SIZE 
) 
5727        int rowOrCol 
= (row 
== -1 ? col 
: row
); 
5729        wxGridSizeEvent 
gridEvt( GetId(), 
5733                mouseEv
.GetX() + GetRowLabelSize(), 
5734                mouseEv
.GetY() + GetColLabelSize(), 
5735                mouseEv
.ControlDown(), 
5736                mouseEv
.ShiftDown(), 
5738                mouseEv
.MetaDown() ); 
5740        claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
5741        vetoed 
= !gridEvt
.IsAllowed(); 
5743    else if ( type 
== wxEVT_GRID_RANGE_SELECT 
) 
5745        // Right now, it should _never_ end up here! 
5746        wxGridRangeSelectEvent 
gridEvt( GetId(), 
5750                m_selectingBottomRight
, 
5752                mouseEv
.ControlDown(), 
5753                mouseEv
.ShiftDown(), 
5755                mouseEv
.MetaDown() ); 
5757        claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
5758        vetoed 
= !gridEvt
.IsAllowed(); 
5762        wxGridEvent 
gridEvt( GetId(), 
5766                mouseEv
.GetX() + GetRowLabelSize(), 
5767                mouseEv
.GetY() + GetColLabelSize(), 
5769                mouseEv
.ControlDown(), 
5770                mouseEv
.ShiftDown(), 
5772                mouseEv
.MetaDown() ); 
5773        claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
5774        vetoed 
= !gridEvt
.IsAllowed(); 
5777    // A Veto'd event may not be `claimed' so test this first 
5778    if (vetoed
) return -1; 
5779    return claimed 
? 1 : 0; 
5783 // Generate a grid event of specified type and return the result 
5784 // of ProcessEvent(). 
5786 int wxGrid::SendEvent( const wxEventType type
, 
5792     if ( type 
== wxEVT_GRID_ROW_SIZE 
|| type 
== wxEVT_GRID_COL_SIZE 
) 
5794         int rowOrCol 
= (row 
== -1 ? col 
: row
); 
5796         wxGridSizeEvent 
gridEvt( GetId(), 
5801         claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
5802         vetoed  
= !gridEvt
.IsAllowed(); 
5806         wxGridEvent 
gridEvt( GetId(), 
5811         claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
5812         vetoed  
= !gridEvt
.IsAllowed(); 
5815     // A Veto'd event may not be `claimed' so test this first 
5816     if (vetoed
) return -1; 
5817     return claimed 
? 1 : 0; 
5821 void wxGrid::OnPaint( wxPaintEvent
& WXUNUSED(event
) ) 
5823     wxPaintDC 
dc(this);  // needed to prevent zillions of paint events on MSW 
5826 void wxGrid::Refresh(bool eraseb
, const wxRect
* rect
) 
5828     // Don't do anything if between Begin/EndBatch... 
5829     // EndBatch() will do all this on the last nested one anyway. 
5830     if (! GetBatchCount()) 
5832         // Refresh to get correct scrolled position: 
5833         wxScrolledWindow::Refresh(eraseb
,rect
); 
5837             int rect_x
, rect_y
, rectWidth
, rectHeight
; 
5838             int width_label
, width_cell
, height_label
, height_cell
; 
5841             //Copy rectangle can get scroll offsets.. 
5842             rect_x 
= rect
->GetX(); 
5843             rect_y 
= rect
->GetY(); 
5844             rectWidth 
= rect
->GetWidth(); 
5845             rectHeight 
= rect
->GetHeight(); 
5847             width_label 
= m_rowLabelWidth 
- rect_x
; 
5848             if (width_label 
> rectWidth
) width_label 
= rectWidth
; 
5850             height_label 
= m_colLabelHeight 
- rect_y
; 
5851             if (height_label 
> rectHeight
) height_label 
= rectHeight
; 
5853             if (rect_x 
> m_rowLabelWidth
) 
5855                 x 
= rect_x 
- m_rowLabelWidth
; 
5856                 width_cell 
= rectWidth
; 
5861                 width_cell 
= rectWidth 
- (m_rowLabelWidth 
- rect_x
); 
5864             if (rect_y 
> m_colLabelHeight
) 
5866                 y 
= rect_y 
- m_colLabelHeight
; 
5867                 height_cell 
= rectHeight
; 
5872                 height_cell 
= rectHeight 
- (m_colLabelHeight 
- rect_y
); 
5875             // Paint corner label part intersecting rect. 
5876             if ( width_label 
> 0 && height_label 
> 0 ) 
5878                 wxRect 
anotherrect(rect_x
, rect_y
, width_label
, height_label
); 
5879                 m_cornerLabelWin
->Refresh(eraseb
, &anotherrect
); 
5882             // Paint col labels part intersecting rect. 
5883             if ( width_cell 
> 0 && height_label 
> 0 ) 
5885                 wxRect 
anotherrect(x
, rect_y
, width_cell
, height_label
); 
5886                 m_colLabelWin
->Refresh(eraseb
, &anotherrect
); 
5889             // Paint row labels part intersecting rect. 
5890             if ( width_label 
> 0 && height_cell 
> 0 ) 
5892                 wxRect 
anotherrect(rect_x
, y
, width_label
, height_cell
); 
5893                 m_rowLabelWin
->Refresh(eraseb
, &anotherrect
); 
5896             // Paint cell area part intersecting rect. 
5897             if ( width_cell 
> 0 && height_cell 
> 0 ) 
5899                 wxRect 
anotherrect(x
, y
, width_cell
, height_cell
); 
5900                 m_gridWin
->Refresh(eraseb
, &anotherrect
); 
5905             m_cornerLabelWin
->Refresh(eraseb
, NULL
); 
5906             m_colLabelWin
->Refresh(eraseb
, NULL
); 
5907             m_rowLabelWin
->Refresh(eraseb
, NULL
); 
5908             m_gridWin
->Refresh(eraseb
, NULL
); 
5913 void wxGrid::OnSize( wxSizeEvent
& event 
) 
5915     // position the child windows 
5918     // don't call CalcDimensions() from here, the base class handles the size 
5924 void wxGrid::OnKeyDown( wxKeyEvent
& event 
) 
5926     if ( m_inOnKeyDown 
) 
5928         // shouldn't be here - we are going round in circles... 
5930         wxFAIL_MSG( wxT("wxGrid::OnKeyDown called while already active") ); 
5933     m_inOnKeyDown 
= TRUE
; 
5935     // propagate the event up and see if it gets processed 
5937     wxWindow 
*parent 
= GetParent(); 
5938     wxKeyEvent 
keyEvt( event 
); 
5939     keyEvt
.SetEventObject( parent 
); 
5941     if ( !parent
->GetEventHandler()->ProcessEvent( keyEvt 
) ) 
5944         // try local handlers 
5946         switch ( event
.KeyCode() ) 
5949                 if ( event
.ControlDown() ) 
5951                     MoveCursorUpBlock( event
.ShiftDown() ); 
5955                     MoveCursorUp( event
.ShiftDown() ); 
5960                 if ( event
.ControlDown() ) 
5962                     MoveCursorDownBlock( event
.ShiftDown() ); 
5966                     MoveCursorDown( event
.ShiftDown() ); 
5971                 if ( event
.ControlDown() ) 
5973                     MoveCursorLeftBlock( event
.ShiftDown() ); 
5977                     MoveCursorLeft( event
.ShiftDown() ); 
5982                 if ( event
.ControlDown() ) 
5984                     MoveCursorRightBlock( event
.ShiftDown() ); 
5988                     MoveCursorRight( event
.ShiftDown() ); 
5993             case WXK_NUMPAD_ENTER
: 
5994                 if ( event
.ControlDown() ) 
5996                     event
.Skip();  // to let the edit control have the return 
6000                     if ( GetGridCursorRow() < GetNumberRows()-1 ) 
6002                         MoveCursorDown( event
.ShiftDown() ); 
6006                         // at the bottom of a column 
6007                         HideCellEditControl(); 
6008                         SaveEditControlValue(); 
6018                 if (event
.ShiftDown()) 
6020                     if ( GetGridCursorCol() > 0 ) 
6022                         MoveCursorLeft( FALSE 
); 
6027                         HideCellEditControl(); 
6028                         SaveEditControlValue(); 
6033                     if ( GetGridCursorCol() < GetNumberCols()-1 ) 
6035                         MoveCursorRight( FALSE 
); 
6040                         HideCellEditControl(); 
6041                         SaveEditControlValue(); 
6047                 if ( event
.ControlDown() ) 
6049                     MakeCellVisible( 0, 0 ); 
6050                     SetCurrentCell( 0, 0 ); 
6059                 if ( event
.ControlDown() ) 
6061                     MakeCellVisible( m_numRows
-1, m_numCols
-1 ); 
6062                     SetCurrentCell( m_numRows
-1, m_numCols
-1 ); 
6079                 if ( event
.ControlDown() ) 
6083                         m_selection
->ToggleCellSelection( m_currentCellCoords
.GetRow(), 
6084                                                           m_currentCellCoords
.GetCol(), 
6085                                                           event
.ControlDown(), 
6092                 if ( !IsEditable() ) 
6094                     MoveCursorRight( FALSE 
); 
6097                 // Otherwise fall through to default 
6100                 // is it possible to edit the current cell at all? 
6101                 if ( !IsCellEditControlEnabled() && CanEnableCellControl() ) 
6103                     // yes, now check whether the cells editor accepts the key 
6104                     int row 
= m_currentCellCoords
.GetRow(); 
6105                     int col 
= m_currentCellCoords
.GetCol(); 
6106                     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
6107                     wxGridCellEditor 
*editor 
= attr
->GetEditor(this, row
, col
); 
6109                     // <F2> is special and will always start editing, for 
6110                     // other keys - ask the editor itself 
6111                     if ( (event
.KeyCode() == WXK_F2 
&& !event
.HasModifiers()) 
6112                          || editor
->IsAcceptedKey(event
) ) 
6114                         EnableCellEditControl(); 
6116                         // the editor could be not shown for a variety of 
6117                         // reasons (i.e. blocked by the app or whatever), so 
6118                         // check if it really was created 
6119                         if ( m_cellEditCtrlEnabled 
) 
6121                             editor
->StartingKey(event
); 
6134                     // let others process char events with modifiers or all 
6135                     // char events for readonly cells 
6142     m_inOnKeyDown 
= FALSE
; 
6145 void wxGrid::OnKeyUp( wxKeyEvent
& event 
) 
6147     // try local handlers 
6149     if ( event
.KeyCode() == WXK_SHIFT 
) 
6151         if ( m_selectingTopLeft 
!= wxGridNoCellCoords 
&& 
6152              m_selectingBottomRight 
!= wxGridNoCellCoords 
) 
6156                 m_selection
->SelectBlock( m_selectingTopLeft
.GetRow(), 
6157                                           m_selectingTopLeft
.GetCol(), 
6158                                           m_selectingBottomRight
.GetRow(), 
6159                                           m_selectingBottomRight
.GetCol(), 
6160                                           event
.ControlDown(), 
6167         m_selectingTopLeft 
= wxGridNoCellCoords
; 
6168         m_selectingBottomRight 
= wxGridNoCellCoords
; 
6169         m_selectingKeyboard 
= wxGridNoCellCoords
; 
6173 void wxGrid::OnEraseBackground(wxEraseEvent
&) 
6177 void wxGrid::SetCurrentCell( const wxGridCellCoords
& coords 
) 
6179     if ( SendEvent( wxEVT_GRID_SELECT_CELL
, coords
.GetRow(), coords
.GetCol() ) ) 
6181         // the event has been intercepted - do nothing 
6185     wxClientDC 
dc(m_gridWin
); 
6188     if ( m_currentCellCoords 
!= wxGridNoCellCoords 
) 
6190         HideCellEditControl(); 
6191         DisableCellEditControl(); 
6193         if ( IsVisible( m_currentCellCoords
, FALSE 
) ) 
6196             r 
= BlockToDeviceRect(m_currentCellCoords
, m_currentCellCoords
); 
6197             if ( !m_gridLinesEnabled 
) 
6205             wxGridCellCoordsArray cells 
= CalcCellsExposed( r 
); 
6207             // Otherwise refresh redraws the highlight! 
6208             m_currentCellCoords 
= coords
; 
6210             DrawGridCellArea(dc
,cells
); 
6211             DrawAllGridLines( dc
, r 
); 
6215     m_currentCellCoords 
= coords
; 
6217     wxGridCellAttr
* attr 
= GetCellAttr(coords
); 
6218     DrawCellHighlight(dc
, attr
); 
6223 void wxGrid::HighlightBlock( int topRow
, int leftCol
, int bottomRow
, int rightCol 
) 
6226     wxGridCellCoords updateTopLeft
, updateBottomRight
; 
6230         if ( m_selection
->GetSelectionMode() == wxGrid::wxGridSelectRows 
) 
6233             rightCol 
= GetNumberCols() - 1; 
6235         else if ( m_selection
->GetSelectionMode() == wxGrid::wxGridSelectColumns 
) 
6238             bottomRow 
= GetNumberRows() - 1; 
6242     if ( topRow 
> bottomRow 
) 
6249     if ( leftCol 
> rightCol 
) 
6256     updateTopLeft 
= wxGridCellCoords( topRow
, leftCol 
); 
6257     updateBottomRight 
= wxGridCellCoords( bottomRow
, rightCol 
); 
6259     // First the case that we selected a completely new area 
6260     if ( m_selectingTopLeft 
== wxGridNoCellCoords 
|| 
6261          m_selectingBottomRight 
== wxGridNoCellCoords 
) 
6264         rect 
= BlockToDeviceRect( wxGridCellCoords ( topRow
, leftCol 
), 
6265                                   wxGridCellCoords ( bottomRow
, rightCol 
) ); 
6266         m_gridWin
->Refresh( FALSE
, &rect 
); 
6268     // Now handle changing an existing selection area. 
6269     else if ( m_selectingTopLeft 
!= updateTopLeft 
|| 
6270               m_selectingBottomRight 
!= updateBottomRight 
) 
6272         // Compute two optimal update rectangles: 
6273         // Either one rectangle is a real subset of the 
6274         // other, or they are (almost) disjoint! 
6276         bool    need_refresh
[4]; 
6280         need_refresh
[3] = FALSE
; 
6283         // Store intermediate values 
6284         wxCoord oldLeft   
= m_selectingTopLeft
.GetCol(); 
6285         wxCoord oldTop    
= m_selectingTopLeft
.GetRow(); 
6286         wxCoord oldRight  
= m_selectingBottomRight
.GetCol(); 
6287         wxCoord oldBottom 
= m_selectingBottomRight
.GetRow(); 
6289         // Determine the outer/inner coordinates. 
6290         if (oldLeft 
> leftCol
) 
6296         if (oldTop 
> topRow 
) 
6302         if (oldRight 
< rightCol 
) 
6305             oldRight 
= rightCol
; 
6308         if (oldBottom 
< bottomRow
) 
6311             oldBottom 
= bottomRow
; 
6315         // Now, either the stuff marked old is the outer 
6316         // rectangle or we don't have a situation where one 
6317         // is contained in the other. 
6319         if ( oldLeft 
< leftCol 
) 
6321             // Refresh the newly selected or deselected 
6322             // area to the left of the old or new selection. 
6323             need_refresh
[0] = TRUE
; 
6324             rect
[0] = BlockToDeviceRect( wxGridCellCoords ( oldTop
, 
6326                                          wxGridCellCoords ( oldBottom
, 
6330         if ( oldTop  
< topRow 
) 
6332             // Refresh the newly selected or deselected 
6333             // area above the old or new selection. 
6334             need_refresh
[1] = TRUE
; 
6335             rect
[1] = BlockToDeviceRect( wxGridCellCoords ( oldTop
, 
6337                                          wxGridCellCoords ( topRow 
- 1, 
6341         if ( oldRight 
> rightCol 
) 
6343             // Refresh the newly selected or deselected 
6344             // area to the right of the old or new selection. 
6345             need_refresh
[2] = TRUE
; 
6346             rect
[2] = BlockToDeviceRect( wxGridCellCoords ( oldTop
, 
6348                                          wxGridCellCoords ( oldBottom
, 
6352         if ( oldBottom 
> bottomRow 
) 
6354             // Refresh the newly selected or deselected 
6355             // area below the old or new selection. 
6356             need_refresh
[3] = TRUE
; 
6357             rect
[3] = BlockToDeviceRect( wxGridCellCoords ( bottomRow 
+ 1, 
6359                                          wxGridCellCoords ( oldBottom
, 
6363         // various Refresh() calls 
6364         for (i 
= 0; i 
< 4; i
++ ) 
6365             if ( need_refresh
[i
] && rect
[i
] != wxGridNoCellRect 
) 
6366                 m_gridWin
->Refresh( FALSE
, &(rect
[i
]) ); 
6369     m_selectingTopLeft 
= updateTopLeft
; 
6370     m_selectingBottomRight 
= updateBottomRight
; 
6374 // ------ functions to get/send data (see also public functions) 
6377 bool wxGrid::GetModelValues() 
6379     // Hide the editor, so it won't hide a changed value. 
6380     HideCellEditControl(); 
6384         // all we need to do is repaint the grid 
6386         m_gridWin
->Refresh(); 
6394 bool wxGrid::SetModelValues() 
6398     // Disable the editor, so it won't hide a changed value. 
6399     // Do we also want to save the current value of the editor first? 
6401     DisableCellEditControl(); 
6405         for ( row 
= 0;  row 
< m_numRows
;  row
++ ) 
6407             for ( col 
= 0;  col 
< m_numCols
;  col
++ ) 
6409                 m_table
->SetValue( row
, col
, GetCellValue(row
, col
) ); 
6421 // Note - this function only draws cells that are in the list of 
6422 // exposed cells (usually set from the update region by 
6423 // CalcExposedCells) 
6425 void wxGrid::DrawGridCellArea( wxDC
& dc
, const wxGridCellCoordsArray
& cells 
) 
6427     if ( !m_numRows 
|| !m_numCols 
) return; 
6429     int i
, numCells 
= cells
.GetCount(); 
6430     int row
, col
, cell_rows
, cell_cols
; 
6431     wxGridCellCoordsArray redrawCells
; 
6433     for ( i 
= numCells
-1; i 
>= 0;  i
-- ) 
6435         row 
= cells
[i
].GetRow(); 
6436         col 
= cells
[i
].GetCol(); 
6437         GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
6439         // If this cell is part of a multicell block, find owner for repaint 
6440         if ( cell_rows 
<= 0 || cell_cols 
<= 0 ) 
6442             wxGridCellCoords 
cell(row
+cell_rows
, col
+cell_cols
); 
6443             bool marked 
= FALSE
; 
6444             for ( int j 
= 0;  j 
< numCells
;  j
++ ) 
6446                 if ( cell 
== cells
[j
] ) 
6454                 int count 
= redrawCells
.GetCount(); 
6455                 for (int j 
= 0; j 
< count
; j
++) 
6457                     if ( cell 
== redrawCells
[j
] ) 
6463                 if (!marked
) redrawCells
.Add( cell 
); 
6465             continue; // don't bother drawing this cell 
6468         // If this cell is empty, find cell to left that might want to overflow 
6469         if (m_table 
&& m_table
->IsEmptyCell(row
, col
)) 
6471             for ( int l 
= 0; l 
< cell_rows
; l
++ ) 
6473                 // find a cell in this row to left alreay marked for repaint 
6475                 for (int k 
= 0; k 
< int(redrawCells
.GetCount()); k
++) 
6476                     if ((redrawCells
[k
].GetCol() < left
) && 
6477                         (redrawCells
[k
].GetRow() == row
)) 
6478                         left
=redrawCells
[k
].GetCol(); 
6480                 if (left 
== col
) left 
= 0; // oh well 
6482                 for (int j 
= col
-1; j 
>= left
; j
--) 
6484                     if (!m_table
->IsEmptyCell(row
+l
, j
)) 
6486                         if (GetCellOverflow(row
+l
, j
)) 
6488                             wxGridCellCoords 
cell(row
+l
, j
); 
6489                             bool marked 
= FALSE
; 
6491                             for (int k 
= 0; k 
< numCells
; k
++) 
6493                                 if ( cell 
== cells
[k
] ) 
6501                                 int count 
= redrawCells
.GetCount(); 
6502                                 for (int k 
= 0; k 
< count
; k
++) 
6504                                     if ( cell 
== redrawCells
[k
] ) 
6510                                 if (!marked
) redrawCells
.Add( cell 
); 
6518         DrawCell( dc
, cells
[i
] ); 
6521     numCells 
= redrawCells
.GetCount(); 
6523     for ( i 
= numCells 
- 1; i 
>= 0;  i
-- ) 
6525         DrawCell( dc
, redrawCells
[i
] ); 
6530 void wxGrid::DrawGridSpace( wxDC
& dc 
) 
6533   m_gridWin
->GetClientSize( &cw
, &ch 
); 
6536   CalcUnscrolledPosition( cw
, ch
, &right
, &bottom 
); 
6538   int rightCol 
= m_numCols 
> 0 ? GetColRight(m_numCols 
- 1) : 0; 
6539   int bottomRow 
= m_numRows 
> 0 ? GetRowBottom(m_numRows 
- 1) : 0 ; 
6541   if ( right 
> rightCol 
|| bottom 
> bottomRow 
) 
6544       CalcUnscrolledPosition( 0, 0, &left
, &top 
); 
6546       dc
.SetBrush( wxBrush(GetDefaultCellBackgroundColour(), wxSOLID
) ); 
6547       dc
.SetPen( *wxTRANSPARENT_PEN 
); 
6549       if ( right 
> rightCol 
) 
6551           dc
.DrawRectangle( rightCol
, top
, right 
- rightCol
, ch
); 
6554       if ( bottom 
> bottomRow 
) 
6556           dc
.DrawRectangle( left
, bottomRow
, cw
, bottom 
- bottomRow
); 
6562 void wxGrid::DrawCell( wxDC
& dc
, const wxGridCellCoords
& coords 
) 
6564     int row 
= coords
.GetRow(); 
6565     int col 
= coords
.GetCol(); 
6567     if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
6570     // we draw the cell border ourselves 
6571 #if !WXGRID_DRAW_LINES 
6572     if ( m_gridLinesEnabled 
) 
6573         DrawCellBorder( dc
, coords 
); 
6576     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
6578     bool isCurrent 
= coords 
== m_currentCellCoords
; 
6580     wxRect rect 
= CellToRect( row
, col 
); 
6582     // if the editor is shown, we should use it and not the renderer 
6583     // Note: However, only if it is really _shown_, i.e. not hidden! 
6584     if ( isCurrent 
&& IsCellEditControlShown() ) 
6586         wxGridCellEditor 
*editor 
= attr
->GetEditor(this, row
, col
); 
6587         editor
->PaintBackground(rect
, attr
); 
6592         // but all the rest is drawn by the cell renderer and hence may be 
6594         wxGridCellRenderer 
*renderer 
= attr
->GetRenderer(this, row
, col
); 
6595         renderer
->Draw(*this, *attr
, dc
, rect
, row
, col
, IsInSelection(coords
)); 
6602 void wxGrid::DrawCellHighlight( wxDC
& dc
, const wxGridCellAttr 
*attr 
) 
6604     int row 
= m_currentCellCoords
.GetRow(); 
6605     int col 
= m_currentCellCoords
.GetCol(); 
6607     if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
6610     wxRect rect 
= CellToRect(row
, col
); 
6612     // hmmm... what could we do here to show that the cell is disabled? 
6613     // for now, I just draw a thinner border than for the other ones, but 
6614     // it doesn't look really good 
6616     int penWidth 
= attr
->IsReadOnly() ? m_cellHighlightROPenWidth 
: m_cellHighlightPenWidth
; 
6620         // The center of th drawn line is where the position/width/height of 
6621         // the rectangle is actually at, (on wxMSW atr least,) so we will 
6622         // reduce the size of the rectangle to compensate for the thickness of 
6623         // the line.  If this is too strange on non wxMSW platforms then 
6624         // please #ifdef this appropriately. 
6625         rect
.x 
+= penWidth
/2; 
6626         rect
.y 
+= penWidth
/2; 
6627         rect
.width 
-= penWidth
-1; 
6628         rect
.height 
-= penWidth
-1; 
6631         // Now draw the rectangle 
6632         dc
.SetPen(wxPen(m_cellHighlightColour
, penWidth
, wxSOLID
)); 
6633         dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
6634         dc
.DrawRectangle(rect
); 
6638         // VZ: my experiments with 3d borders... 
6640         // how to properly set colours for arbitrary bg? 
6641         wxCoord x1 
= rect
.x
, 
6643                 x2 
= rect
.x 
+ rect
.width 
-1, 
6644                 y2 
= rect
.y 
+ rect
.height 
-1; 
6646         dc
.SetPen(*wxWHITE_PEN
); 
6647         dc
.DrawLine(x1
, y1
, x2
, y1
); 
6648         dc
.DrawLine(x1
, y1
, x1
, y2
); 
6650         dc
.DrawLine(x1 
+ 1, y2 
- 1, x2 
- 1, y2 
- 1); 
6651         dc
.DrawLine(x2 
- 1, y1 
+ 1, x2 
- 1, y2 
); 
6653         dc
.SetPen(*wxBLACK_PEN
); 
6654         dc
.DrawLine(x1
, y2
, x2
, y2
); 
6655         dc
.DrawLine(x2
, y1
, x2
, y2
+1); 
6660 void wxGrid::DrawCellBorder( wxDC
& dc
, const wxGridCellCoords
& coords 
) 
6662     int row 
= coords
.GetRow(); 
6663     int col 
= coords
.GetCol(); 
6664     if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
6667     dc
.SetPen( wxPen(GetGridLineColour(), 1, wxSOLID
) ); 
6669     wxRect rect 
= CellToRect( row
, col 
); 
6671     // right hand border 
6673     dc
.DrawLine( rect
.x 
+ rect
.width
, rect
.y
, 
6674                  rect
.x 
+ rect
.width
, rect
.y 
+ rect
.height 
+ 1 ); 
6678     dc
.DrawLine( rect
.x
,              rect
.y 
+ rect
.height
, 
6679                  rect
.x 
+ rect
.width
, rect
.y 
+ rect
.height
); 
6682 void wxGrid::DrawHighlight(wxDC
& dc
,const  wxGridCellCoordsArray
& cells
) 
6684     // This if block was previously in wxGrid::OnPaint but that doesn't 
6685     // seem to get called under wxGTK - MB 
6687     if ( m_currentCellCoords 
== wxGridNoCellCoords  
&& 
6688          m_numRows 
&& m_numCols 
) 
6690         m_currentCellCoords
.Set(0, 0); 
6693     if ( IsCellEditControlShown() ) 
6695         // don't show highlight when the edit control is shown 
6699     // if the active cell was repainted, repaint its highlight too because it 
6700     // might have been damaged by the grid lines 
6701     size_t count 
= cells
.GetCount(); 
6702     for ( size_t n 
= 0; n 
< count
; n
++ ) 
6704         if ( cells
[n
] == m_currentCellCoords 
) 
6706             wxGridCellAttr
* attr 
= GetCellAttr(m_currentCellCoords
); 
6707             DrawCellHighlight(dc
, attr
); 
6715 // TODO: remove this ??? 
6716 // This is used to redraw all grid lines e.g. when the grid line colour 
6719 void wxGrid::DrawAllGridLines( wxDC
& dc
, const wxRegion 
& WXUNUSED(reg
) ) 
6721 #if !WXGRID_DRAW_LINES 
6725     if ( !m_gridLinesEnabled 
|| 
6727          !m_numCols 
) return; 
6729     int top
, bottom
, left
, right
; 
6731 #if 0  //#ifndef __WXGTK__ 
6735       m_gridWin
->GetClientSize(&cw
, &ch
); 
6737       // virtual coords of visible area 
6739       CalcUnscrolledPosition( 0, 0, &left
, &top 
); 
6740       CalcUnscrolledPosition( cw
, ch
, &right
, &bottom 
); 
6745       reg
.GetBox(x
, y
, w
, h
); 
6746       CalcUnscrolledPosition( x
, y
, &left
, &top 
); 
6747       CalcUnscrolledPosition( x 
+ w
, y 
+ h
, &right
, &bottom 
); 
6751       m_gridWin
->GetClientSize(&cw
, &ch
); 
6752       CalcUnscrolledPosition( 0, 0, &left
, &top 
); 
6753       CalcUnscrolledPosition( cw
, ch
, &right
, &bottom 
); 
6756     // avoid drawing grid lines past the last row and col 
6758     right 
= wxMin( right
, GetColRight(m_numCols 
- 1) ); 
6759     bottom 
= wxMin( bottom
, GetRowBottom(m_numRows 
- 1) ); 
6761     // no gridlines inside multicells, clip them out 
6762     int leftCol   
= XToCol(left
); 
6763     int topRow    
= YToRow(top
); 
6764     int rightCol  
= XToCol(right
); 
6765     int bottomRow 
= YToRow(bottom
); 
6766     wxRegion 
clippedcells(0, 0, cw
, ch
); 
6768     if ((leftCol 
>= 0) && (topRow 
>= 0)) 
6770         if (rightCol  
< 0) rightCol  
= m_numCols
; 
6771         if (bottomRow 
< 0) bottomRow 
= m_numRows
; 
6773         int i
, j
, cell_rows
, cell_cols
; 
6776         for (j
=topRow
; j
<bottomRow
; j
++) 
6778             for (i
=leftCol
; i
<rightCol
; i
++) 
6780                 GetCellSize( j
, i
, &cell_rows
, &cell_cols 
); 
6781                 if ((cell_rows 
> 1) || (cell_cols 
> 1)) 
6783                     rect 
= CellToRect(j
,i
); 
6784                     CalcScrolledPosition( rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
6785                     clippedcells
.Subtract(rect
); 
6787                 else if ((cell_rows 
< 0) || (cell_cols 
< 0)) 
6789                     rect 
= CellToRect(j
+cell_rows
, i
+cell_cols
); 
6790                     CalcScrolledPosition( rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
6791                     clippedcells
.Subtract(rect
); 
6796     dc
.SetClippingRegion( clippedcells 
); 
6798     dc
.SetPen( wxPen(GetGridLineColour(), 1, wxSOLID
) ); 
6800     // horizontal grid lines 
6803     for ( i 
= internalYToRow(top
); i 
< m_numRows
; i
++ ) 
6805         int bot 
= GetRowBottom(i
) - 1; 
6814             dc
.DrawLine( left
, bot
, right
, bot 
); 
6819     // vertical grid lines 
6821     for ( i 
= internalXToCol(left
); i 
< m_numCols
; i
++ ) 
6823         int colRight 
= GetColRight(i
) - 1; 
6824         if ( colRight 
> right 
) 
6829         if ( colRight 
>= left 
) 
6831             dc
.DrawLine( colRight
, top
, colRight
, bottom 
); 
6834     dc
.DestroyClippingRegion(); 
6838 void wxGrid::DrawRowLabels( wxDC
& dc 
,const wxArrayInt
& rows
) 
6840     if ( !m_numRows 
) return; 
6843     size_t numLabels 
= rows
.GetCount(); 
6845     for ( i 
= 0;  i 
< numLabels
;  i
++ ) 
6847         DrawRowLabel( dc
, rows
[i
] ); 
6852 void wxGrid::DrawRowLabel( wxDC
& dc
, int row 
) 
6854     if ( GetRowHeight(row
) <= 0 ) 
6857     int rowTop 
= GetRowTop(row
), 
6858         rowBottom 
= GetRowBottom(row
) - 1; 
6860     dc
.SetPen( *wxBLACK_PEN 
); 
6861     dc
.DrawLine( m_rowLabelWidth
-1, rowTop
, 
6862                  m_rowLabelWidth
-1, rowBottom 
); 
6864     dc
.DrawLine( 0, rowBottom
, m_rowLabelWidth
-1, rowBottom 
); 
6866     dc
.SetPen( *wxWHITE_PEN 
); 
6867     dc
.DrawLine( 0, rowTop
, 0, rowBottom 
); 
6868     dc
.DrawLine( 0, rowTop
, m_rowLabelWidth
-1, rowTop 
); 
6870     dc
.SetBackgroundMode( wxTRANSPARENT 
); 
6871     dc
.SetTextForeground( GetLabelTextColour() ); 
6872     dc
.SetFont( GetLabelFont() ); 
6875     GetRowLabelAlignment( &hAlign
, &vAlign 
); 
6879     rect
.SetY( GetRowTop(row
) + 2 ); 
6880     rect
.SetWidth( m_rowLabelWidth 
- 4 ); 
6881     rect
.SetHeight( GetRowHeight(row
) - 4 ); 
6882     DrawTextRectangle( dc
, GetRowLabelValue( row 
), rect
, hAlign
, vAlign 
); 
6886 void wxGrid::DrawColLabels( wxDC
& dc
,const wxArrayInt
& cols 
) 
6888     if ( !m_numCols 
) return; 
6891     size_t numLabels 
= cols
.GetCount(); 
6893     for ( i 
= 0;  i 
< numLabels
;  i
++ ) 
6895         DrawColLabel( dc
, cols
[i
] ); 
6900 void wxGrid::DrawColLabel( wxDC
& dc
, int col 
) 
6902     if ( GetColWidth(col
) <= 0 ) 
6905     int colLeft 
= GetColLeft(col
), 
6906         colRight 
= GetColRight(col
) - 1; 
6908     dc
.SetPen( *wxBLACK_PEN 
); 
6909     dc
.DrawLine( colRight
, 0, 
6910                  colRight
, m_colLabelHeight
-1 ); 
6912     dc
.DrawLine( colLeft
, m_colLabelHeight
-1, 
6913                  colRight
, m_colLabelHeight
-1 ); 
6915     dc
.SetPen( *wxWHITE_PEN 
); 
6916     dc
.DrawLine( colLeft
, 0, colLeft
, m_colLabelHeight
-1 ); 
6917     dc
.DrawLine( colLeft
, 0, colRight
, 0 ); 
6919     dc
.SetBackgroundMode( wxTRANSPARENT 
); 
6920     dc
.SetTextForeground( GetLabelTextColour() ); 
6921     dc
.SetFont( GetLabelFont() ); 
6923     dc
.SetBackgroundMode( wxTRANSPARENT 
); 
6924     dc
.SetTextForeground( GetLabelTextColour() ); 
6925     dc
.SetFont( GetLabelFont() ); 
6928     GetColLabelAlignment( &hAlign
, &vAlign 
); 
6931     rect
.SetX( colLeft 
+ 2 ); 
6933     rect
.SetWidth( GetColWidth(col
) - 4 ); 
6934     rect
.SetHeight( m_colLabelHeight 
- 4 ); 
6935     DrawTextRectangle( dc
, GetColLabelValue( col 
), rect
, hAlign
, vAlign 
); 
6938 void wxGrid::DrawTextRectangle( wxDC
& dc
, 
6939                                 const wxString
& value
, 
6944     wxArrayString lines
; 
6946     StringToLines( value
, lines 
); 
6949     //Forward to new API. 
6950     DrawTextRectangle(  dc
, 
6958 void wxGrid::DrawTextRectangle( wxDC
& dc
, 
6959                                 const wxArrayString
& lines
, 
6964     long textWidth
, textHeight
; 
6965     long lineWidth
, lineHeight
; 
6967     dc
.SetClippingRegion( rect 
); 
6968     if ( lines
.GetCount() ) 
6970         GetTextBoxSize( dc
, lines
, &textWidth
, &textHeight 
); 
6971         dc
.GetTextExtent( lines
[0], &lineWidth
, &lineHeight 
); 
6974         switch ( horizAlign 
) 
6977                 x 
= rect
.x 
+ (rect
.width 
- textWidth 
- 1); 
6980             case wxALIGN_CENTRE
: 
6981                 x 
= rect
.x 
+ ((rect
.width 
- textWidth
)/2); 
6990         switch ( vertAlign 
) 
6992             case wxALIGN_BOTTOM
: 
6993                 y 
= rect
.y 
+ (rect
.height 
- textHeight 
- 1); 
6996             case wxALIGN_CENTRE
: 
6997                 y 
= rect
.y 
+ ((rect
.height 
- textHeight
)/2); 
7006         for ( size_t i 
= 0;  i 
< lines
.GetCount();  i
++ ) 
7008             dc
.DrawText( lines
[i
], (int)x
, (int)y 
); 
7013     dc
.DestroyClippingRegion(); 
7017 // Split multi line text up into an array of strings.  Any existing 
7018 // contents of the string array are preserved. 
7020 void wxGrid::StringToLines( const wxString
& value
, wxArrayString
& lines 
) 
7024     wxString eol 
= wxTextFile::GetEOL( wxTextFileType_Unix 
); 
7025     wxString tVal 
= wxTextFile::Translate( value
, wxTextFileType_Unix 
); 
7027     while ( startPos 
< (int)tVal
.Length() ) 
7029         pos 
= tVal
.Mid(startPos
).Find( eol 
); 
7034         else if ( pos 
== 0 ) 
7036             lines
.Add( wxEmptyString 
); 
7040             lines
.Add( value
.Mid(startPos
, pos
) ); 
7044     if ( startPos 
< (int)value
.Length() ) 
7046         lines
.Add( value
.Mid( startPos 
) ); 
7051 void wxGrid::GetTextBoxSize( wxDC
& dc
, 
7052                              const wxArrayString
& lines
, 
7053                              long *width
, long *height 
) 
7060     for ( i 
= 0;  i 
< lines
.GetCount();  i
++ ) 
7062         dc
.GetTextExtent( lines
[i
], &lineW
, &lineH 
); 
7063         w 
= wxMax( w
, lineW 
); 
7072 // ------ Batch processing. 
7074 void wxGrid::EndBatch() 
7076     if ( m_batchCount 
> 0 ) 
7079         if ( !m_batchCount 
) 
7082             m_rowLabelWin
->Refresh(); 
7083             m_colLabelWin
->Refresh(); 
7084             m_cornerLabelWin
->Refresh(); 
7085             m_gridWin
->Refresh(); 
7090 // Use this, rather than wxWindow::Refresh(), to force an immediate 
7091 // repainting of the grid. Has no effect if you are already inside a 
7092 // BeginBatch / EndBatch block. 
7094 void wxGrid::ForceRefresh() 
7102 // ------ Edit control functions 
7106 void wxGrid::EnableEditing( bool edit 
) 
7108     // TODO: improve this ? 
7110     if ( edit 
!= m_editable 
) 
7112         if(!edit
) EnableCellEditControl(edit
); 
7118 void wxGrid::EnableCellEditControl( bool enable 
) 
7123     if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
7124         SetCurrentCell( 0, 0 ); 
7126     if ( enable 
!= m_cellEditCtrlEnabled 
) 
7130             if (SendEvent( wxEVT_GRID_EDITOR_SHOWN
) <0) 
7133             // this should be checked by the caller! 
7134             wxASSERT_MSG( CanEnableCellControl(), 
7135                           _T("can't enable editing for this cell!") ); 
7137             // do it before ShowCellEditControl() 
7138             m_cellEditCtrlEnabled 
= enable
; 
7140             ShowCellEditControl(); 
7144             //FIXME:add veto support 
7145             SendEvent( wxEVT_GRID_EDITOR_HIDDEN
); 
7147             HideCellEditControl(); 
7148             SaveEditControlValue(); 
7150             // do it after HideCellEditControl() 
7151             m_cellEditCtrlEnabled 
= enable
; 
7156 bool wxGrid::IsCurrentCellReadOnly() const 
7159     wxGridCellAttr
* attr 
= ((wxGrid 
*)this)->GetCellAttr(m_currentCellCoords
); 
7160     bool readonly 
= attr
->IsReadOnly(); 
7166 bool wxGrid::CanEnableCellControl() const 
7168     return m_editable 
&& !IsCurrentCellReadOnly(); 
7171 bool wxGrid::IsCellEditControlEnabled() const 
7173     // the cell edit control might be disable for all cells or just for the 
7174     // current one if it's read only 
7175     return m_cellEditCtrlEnabled 
? !IsCurrentCellReadOnly() : FALSE
; 
7178 bool wxGrid::IsCellEditControlShown() const 
7180     bool isShown 
= FALSE
; 
7182     if ( m_cellEditCtrlEnabled 
) 
7184         int row 
= m_currentCellCoords
.GetRow(); 
7185         int col 
= m_currentCellCoords
.GetCol(); 
7186         wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
7187         wxGridCellEditor
* editor 
= attr
->GetEditor((wxGrid
*) this, row
, col
); 
7192             if ( editor
->IsCreated() ) 
7194                 isShown 
= editor
->GetControl()->IsShown(); 
7204 void wxGrid::ShowCellEditControl() 
7206     if ( IsCellEditControlEnabled() ) 
7208         if ( !IsVisible( m_currentCellCoords 
) ) 
7210             m_cellEditCtrlEnabled 
= FALSE
; 
7215             wxRect rect 
= CellToRect( m_currentCellCoords 
); 
7216             int row 
= m_currentCellCoords
.GetRow(); 
7217             int col 
= m_currentCellCoords
.GetCol(); 
7219             // if this is part of a multicell, find owner (topleft) 
7220             int cell_rows
, cell_cols
; 
7221             GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
7222             if ( cell_rows 
<= 0 || cell_cols 
<= 0 ) 
7226                 m_currentCellCoords
.SetRow( row 
); 
7227                 m_currentCellCoords
.SetCol( col 
); 
7230             // convert to scrolled coords 
7232             CalcScrolledPosition( rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
7234             // done in PaintBackground() 
7236             // erase the highlight and the cell contents because the editor 
7237             // might not cover the entire cell 
7238             wxClientDC 
dc( m_gridWin 
); 
7240             dc
.SetBrush(*wxLIGHT_GREY_BRUSH
); //wxBrush(attr->GetBackgroundColour(), wxSOLID)); 
7241             dc
.SetPen(*wxTRANSPARENT_PEN
); 
7242             dc
.DrawRectangle(rect
); 
7245             // cell is shifted by one pixel 
7246             // However, don't allow x or y to become negative 
7247             // since the SetSize() method interprets that as 
7254             wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
7255             wxGridCellEditor
* editor 
= attr
->GetEditor(this, row
, col
); 
7256             if ( !editor
->IsCreated() ) 
7258                 editor
->Create(m_gridWin
, -1, 
7259                                new wxGridCellEditorEvtHandler(this, editor
)); 
7261                 wxGridEditorCreatedEvent 
evt(GetId(), 
7262                                              wxEVT_GRID_EDITOR_CREATED
, 
7266                                              editor
->GetControl()); 
7267                 GetEventHandler()->ProcessEvent(evt
); 
7270             editor
->Show( TRUE
, attr 
); 
7272             // resize editor to overflow into righthand cells if allowed 
7273             int maxWidth 
= rect
.width
; 
7274             wxString value 
= GetCellValue(row
, col
); 
7275             if ( (value 
!= wxEmptyString
) && (attr
->GetOverflow()) ) 
7278                 GetTextExtent(value
, &maxWidth
, &y
, 
7279                         NULL
, NULL
, &attr
->GetFont()); 
7280                 if (maxWidth 
< rect
.width
) maxWidth 
= rect
.width
; 
7282             int client_right 
= m_gridWin
->GetClientSize().GetWidth(); 
7283             if (rect
.x
+maxWidth 
> client_right
) 
7284                 maxWidth 
= client_right 
- rect
.x
; 
7286             if ((maxWidth 
> rect
.width
) && (col 
< m_numCols
) && m_table
) 
7288                 GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
7289                 // may have changed earlier 
7290                 for (int i 
= col
+cell_cols
; i 
< m_numCols
; i
++) 
7293                     GetCellSize( row
, i
, &c_rows
, &c_cols 
); 
7294                     // looks weird going over a multicell 
7295                     if (m_table
->IsEmptyCell(row
,i
) && 
7296                             (rect
.width 
< maxWidth
) && (c_rows 
== 1)) 
7297                         rect
.width 
+= GetColWidth(i
); 
7301                 if (rect
.GetRight() > client_right
) 
7302                     rect
.SetRight(client_right
-1); 
7304             editor
->SetSize( rect 
); 
7306             editor
->BeginEdit(row
, col
, this); 
7315 void wxGrid::HideCellEditControl() 
7317     if ( IsCellEditControlEnabled() ) 
7319         int row 
= m_currentCellCoords
.GetRow(); 
7320         int col 
= m_currentCellCoords
.GetCol(); 
7322         wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
7323         wxGridCellEditor 
*editor 
= attr
->GetEditor(this, row
, col
); 
7324         editor
->Show( FALSE 
); 
7327         m_gridWin
->SetFocus(); 
7328         // refresh whole row to the right 
7329         wxRect 
rect( CellToRect(row
, col
) ); 
7330         CalcScrolledPosition(rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
7331         rect
.width 
= m_gridWin
->GetClientSize().GetWidth() - rect
.x
; 
7332         m_gridWin
->Refresh( FALSE
, &rect 
); 
7337 void wxGrid::SaveEditControlValue() 
7339     if ( IsCellEditControlEnabled() ) 
7341         int row 
= m_currentCellCoords
.GetRow(); 
7342         int col 
= m_currentCellCoords
.GetCol(); 
7344         wxString oldval 
= GetCellValue(row
,col
); 
7346         wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
7347         wxGridCellEditor
* editor 
= attr
->GetEditor(this, row
, col
); 
7348         bool changed 
= editor
->EndEdit(row
, col
, this); 
7355             if ( SendEvent( wxEVT_GRID_CELL_CHANGE
, 
7356                        m_currentCellCoords
.GetRow(), 
7357                        m_currentCellCoords
.GetCol() ) < 0 ) { 
7359                 // Event has been vetoed, set the data back. 
7360                 SetCellValue(row
,col
,oldval
); 
7368 // ------ Grid location functions 
7369 //  Note that all of these functions work with the logical coordinates of 
7370 //  grid cells and labels so you will need to convert from device 
7371 //  coordinates for mouse events etc. 
7374 void wxGrid::XYToCell( int x
, int y
, wxGridCellCoords
& coords 
) 
7376     int row 
= YToRow(y
); 
7377     int col 
= XToCol(x
); 
7379     if ( row 
== -1  ||  col 
== -1 ) 
7381         coords 
= wxGridNoCellCoords
; 
7385         coords
.Set( row
, col 
); 
7390 // Internal Helper function for computing row or column from some 
7391 // (unscrolled) coordinate value, using either 
7392 // m_defaultRowHeight/m_defaultColWidth or binary search on array 
7393 // of m_rowBottoms/m_ColRights to speed up the search! 
7395 static int CoordToRowOrCol(int coord
, int defaultDist
, int minDist
, 
7396                            const wxArrayInt
& BorderArray
, int nMax
, 
7401     size_t i_max 
= coord 
/ defaultDist
, 
7404     if (BorderArray
.IsEmpty()) 
7406         if((int) i_max 
< nMax
) 
7408         return maxOnOverflow 
? nMax 
- 1 : -1; 
7411     if ( i_max 
>= BorderArray
.GetCount()) 
7412         i_max 
= BorderArray
.GetCount() - 1; 
7415         if ( coord 
>= BorderArray
[i_max
]) 
7418             i_max 
= coord 
/ minDist
; 
7420         if ( i_max 
>= BorderArray
.GetCount()) 
7421             i_max 
= BorderArray
.GetCount() - 1; 
7423     if ( coord 
>= BorderArray
[i_max
]) 
7424         return maxOnOverflow 
? (int)i_max 
: -1; 
7425     if ( coord 
< BorderArray
[0] ) 
7428     while ( i_max 
- i_min 
> 0 ) 
7430         wxCHECK_MSG(BorderArray
[i_min
] <= coord 
&& coord 
< BorderArray
[i_max
], 
7431                     0, _T("wxGrid: internal error in CoordToRowOrCol")); 
7432         if (coord 
>=  BorderArray
[ i_max 
- 1]) 
7436         int median 
= i_min 
+ (i_max 
- i_min 
+ 1) / 2; 
7437         if (coord 
< BorderArray
[median
]) 
7445 int wxGrid::YToRow( int y 
) 
7447     return CoordToRowOrCol(y
, m_defaultRowHeight
, 
7448                            WXGRID_MIN_ROW_HEIGHT
, m_rowBottoms
, m_numRows
, FALSE
); 
7452 int wxGrid::XToCol( int x 
) 
7454     return CoordToRowOrCol(x
, m_defaultColWidth
, 
7455                            WXGRID_MIN_COL_WIDTH
, m_colRights
, m_numCols
, FALSE
); 
7459 // return the row number that that the y coord is near the edge of, or 
7460 // -1 if not near an edge 
7462 int wxGrid::YToEdgeOfRow( int y 
) 
7465     i 
= internalYToRow(y
); 
7467     if ( GetRowHeight(i
) > WXGRID_LABEL_EDGE_ZONE 
) 
7469         // We know that we are in row i, test whether we are 
7470         // close enough to lower or upper border, respectively. 
7471         if ( abs(GetRowBottom(i
) - y
) < WXGRID_LABEL_EDGE_ZONE 
) 
7473         else if( i 
> 0 && y 
- GetRowTop(i
) < WXGRID_LABEL_EDGE_ZONE 
) 
7481 // return the col number that that the x coord is near the edge of, or 
7482 // -1 if not near an edge 
7484 int wxGrid::XToEdgeOfCol( int x 
) 
7487     i 
= internalXToCol(x
); 
7489     if ( GetColWidth(i
) > WXGRID_LABEL_EDGE_ZONE 
) 
7491         // We know that we are in column i,  test whether we are 
7492         // close enough to right or left border, respectively. 
7493         if ( abs(GetColRight(i
) - x
) < WXGRID_LABEL_EDGE_ZONE 
) 
7495         else if( i 
> 0 && x 
- GetColLeft(i
) < WXGRID_LABEL_EDGE_ZONE 
) 
7503 wxRect 
wxGrid::CellToRect( int row
, int col 
) 
7505     wxRect 
rect( -1, -1, -1, -1 ); 
7507     if ( row 
>= 0  &&  row 
< m_numRows  
&& 
7508          col 
>= 0  &&  col 
< m_numCols 
) 
7510         int i
, cell_rows
, cell_cols
; 
7511         rect
.width 
= rect
.height 
= 0; 
7512         GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
7513         // if negative then find multicell owner 
7514         if (cell_rows 
< 0) row 
+= cell_rows
; 
7515         if (cell_cols 
< 0) col 
+= cell_cols
; 
7516         GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
7518         rect
.x 
= GetColLeft(col
); 
7519         rect
.y 
= GetRowTop(row
); 
7520         for (i
=col
; i
<col
+cell_cols
; i
++) 
7521             rect
.width  
+= GetColWidth(i
); 
7522         for (i
=row
; i
<row
+cell_rows
; i
++) 
7523             rect
.height 
+= GetRowHeight(i
); 
7526     // if grid lines are enabled, then the area of the cell is a bit smaller 
7527     if (m_gridLinesEnabled
) { 
7535 bool wxGrid::IsVisible( int row
, int col
, bool wholeCellVisible 
) 
7537     // get the cell rectangle in logical coords 
7539     wxRect 
r( CellToRect( row
, col 
) ); 
7541     // convert to device coords 
7543     int left
, top
, right
, bottom
; 
7544     CalcScrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top 
); 
7545     CalcScrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom 
); 
7547     // check against the client area of the grid window 
7550     m_gridWin
->GetClientSize( &cw
, &ch 
); 
7552     if ( wholeCellVisible 
) 
7554         // is the cell wholly visible ? 
7556         return ( left 
>= 0  &&  right 
<= cw  
&& 
7557                  top 
>= 0  &&  bottom 
<= ch 
); 
7561         // is the cell partly visible ? 
7563         return ( ((left 
>=0 && left 
< cw
) || (right 
> 0 && right 
<= cw
))  && 
7564                  ((top 
>=0 && top 
< ch
) || (bottom 
> 0 && bottom 
<= ch
)) ); 
7569 // make the specified cell location visible by doing a minimal amount 
7572 void wxGrid::MakeCellVisible( int row
, int col 
) 
7576     int xpos 
= -1, ypos 
= -1; 
7578     if ( row 
>= 0  &&  row 
< m_numRows  
&& 
7579          col 
>= 0  &&  col 
< m_numCols 
) 
7581         // get the cell rectangle in logical coords 
7583         wxRect 
r( CellToRect( row
, col 
) ); 
7585         // convert to device coords 
7587         int left
, top
, right
, bottom
; 
7588         CalcScrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top 
); 
7589         CalcScrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom 
); 
7592         m_gridWin
->GetClientSize( &cw
, &ch 
); 
7598         else if ( bottom 
> ch 
) 
7600             int h 
= r
.GetHeight(); 
7602             for ( i 
= row
-1;  i 
>= 0;  i
-- ) 
7604                 int rowHeight 
= GetRowHeight(i
); 
7605                 if ( h 
+ rowHeight 
> ch 
) 
7612             // we divide it later by GRID_SCROLL_LINE, make sure that we don't 
7613             // have rounding errors (this is important, because if we do, we 
7614             // might not scroll at all and some cells won't be redrawn) 
7616             // Sometimes GRID_SCROLL_LINE/2 is not enough, so just add a full 
7618             ypos 
+= GRID_SCROLL_LINE_Y
; 
7625         else if ( right 
> cw 
) 
7627             int w 
= r
.GetWidth(); 
7629             for ( i 
= col
-1;  i 
>= 0;  i
-- ) 
7631                 int colWidth 
= GetColWidth(i
); 
7632                 if ( w 
+ colWidth 
> cw 
) 
7639             // see comment for ypos above 
7640             xpos 
+= GRID_SCROLL_LINE_X
; 
7643         if ( xpos 
!= -1  ||  ypos 
!= -1 ) 
7646                 xpos 
/= GRID_SCROLL_LINE_X
; 
7648                 ypos 
/= GRID_SCROLL_LINE_Y
; 
7649             Scroll( xpos
, ypos 
); 
7657 // ------ Grid cursor movement functions 
7660 bool wxGrid::MoveCursorUp( bool expandSelection 
) 
7662     if ( m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
7663          m_currentCellCoords
.GetRow() >= 0 ) 
7665         if ( expandSelection
) 
7667             if ( m_selectingKeyboard 
== wxGridNoCellCoords 
) 
7668                 m_selectingKeyboard 
= m_currentCellCoords
; 
7669             if ( m_selectingKeyboard
.GetRow() > 0 ) 
7671                 m_selectingKeyboard
.SetRow( m_selectingKeyboard
.GetRow() - 1 ); 
7672                 MakeCellVisible( m_selectingKeyboard
.GetRow(), 
7673                                  m_selectingKeyboard
.GetCol() ); 
7674                 HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
7677         else if ( m_currentCellCoords
.GetRow() > 0 ) 
7680             MakeCellVisible( m_currentCellCoords
.GetRow() - 1, 
7681                              m_currentCellCoords
.GetCol() ); 
7682             SetCurrentCell( m_currentCellCoords
.GetRow() - 1, 
7683                             m_currentCellCoords
.GetCol() ); 
7694 bool wxGrid::MoveCursorDown( bool expandSelection 
) 
7696     if ( m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
7697          m_currentCellCoords
.GetRow() < m_numRows 
) 
7699         if ( expandSelection 
) 
7701             if ( m_selectingKeyboard 
== wxGridNoCellCoords 
) 
7702                 m_selectingKeyboard 
= m_currentCellCoords
; 
7703             if ( m_selectingKeyboard
.GetRow() < m_numRows
-1 ) 
7705                 m_selectingKeyboard
.SetRow( m_selectingKeyboard
.GetRow() + 1 ); 
7706                 MakeCellVisible( m_selectingKeyboard
.GetRow(), 
7707                         m_selectingKeyboard
.GetCol() ); 
7708                 HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
7711         else if ( m_currentCellCoords
.GetRow() < m_numRows 
- 1 ) 
7714             MakeCellVisible( m_currentCellCoords
.GetRow() + 1, 
7715                              m_currentCellCoords
.GetCol() ); 
7716             SetCurrentCell( m_currentCellCoords
.GetRow() + 1, 
7717                             m_currentCellCoords
.GetCol() ); 
7728 bool wxGrid::MoveCursorLeft( bool expandSelection 
) 
7730     if ( m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
7731          m_currentCellCoords
.GetCol() >= 0 ) 
7733         if ( expandSelection 
) 
7735             if ( m_selectingKeyboard 
== wxGridNoCellCoords 
) 
7736                 m_selectingKeyboard 
= m_currentCellCoords
; 
7737             if ( m_selectingKeyboard
.GetCol() > 0 ) 
7739                 m_selectingKeyboard
.SetCol( m_selectingKeyboard
.GetCol() - 1 ); 
7740                 MakeCellVisible( m_selectingKeyboard
.GetRow(), 
7741                         m_selectingKeyboard
.GetCol() ); 
7742                 HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
7745         else if ( m_currentCellCoords
.GetCol() > 0 ) 
7748             MakeCellVisible( m_currentCellCoords
.GetRow(), 
7749                              m_currentCellCoords
.GetCol() - 1 ); 
7750             SetCurrentCell( m_currentCellCoords
.GetRow(), 
7751                             m_currentCellCoords
.GetCol() - 1 ); 
7762 bool wxGrid::MoveCursorRight( bool expandSelection 
) 
7764     if ( m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
7765          m_currentCellCoords
.GetCol() < m_numCols 
) 
7767         if ( expandSelection 
) 
7769             if ( m_selectingKeyboard 
== wxGridNoCellCoords 
) 
7770                 m_selectingKeyboard 
= m_currentCellCoords
; 
7771             if ( m_selectingKeyboard
.GetCol() < m_numCols 
- 1 ) 
7773                 m_selectingKeyboard
.SetCol( m_selectingKeyboard
.GetCol() + 1 ); 
7774                 MakeCellVisible( m_selectingKeyboard
.GetRow(), 
7775                         m_selectingKeyboard
.GetCol() ); 
7776                 HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
7779         else if ( m_currentCellCoords
.GetCol() < m_numCols 
- 1 ) 
7782             MakeCellVisible( m_currentCellCoords
.GetRow(), 
7783                              m_currentCellCoords
.GetCol() + 1 ); 
7784             SetCurrentCell( m_currentCellCoords
.GetRow(), 
7785                             m_currentCellCoords
.GetCol() + 1 ); 
7796 bool wxGrid::MovePageUp() 
7798     if ( m_currentCellCoords 
== wxGridNoCellCoords 
) return FALSE
; 
7800     int row 
= m_currentCellCoords
.GetRow(); 
7804         m_gridWin
->GetClientSize( &cw
, &ch 
); 
7806         int y 
= GetRowTop(row
); 
7807         int newRow 
= YToRow( y 
- ch 
+ 1 ); 
7812         else if ( newRow 
== row 
) 
7817         MakeCellVisible( newRow
, m_currentCellCoords
.GetCol() ); 
7818         SetCurrentCell( newRow
, m_currentCellCoords
.GetCol() ); 
7826 bool wxGrid::MovePageDown() 
7828     if ( m_currentCellCoords 
== wxGridNoCellCoords 
) return FALSE
; 
7830     int row 
= m_currentCellCoords
.GetRow(); 
7831     if ( row 
< m_numRows 
) 
7834         m_gridWin
->GetClientSize( &cw
, &ch 
); 
7836         int y 
= GetRowTop(row
); 
7837         int newRow 
= YToRow( y 
+ ch 
); 
7840             newRow 
= m_numRows 
- 1; 
7842         else if ( newRow 
== row 
) 
7847         MakeCellVisible( newRow
, m_currentCellCoords
.GetCol() ); 
7848         SetCurrentCell( newRow
, m_currentCellCoords
.GetCol() ); 
7856 bool wxGrid::MoveCursorUpBlock( bool expandSelection 
) 
7859          m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
7860          m_currentCellCoords
.GetRow() > 0 ) 
7862         int row 
= m_currentCellCoords
.GetRow(); 
7863         int col 
= m_currentCellCoords
.GetCol(); 
7865         if ( m_table
->IsEmptyCell(row
, col
) ) 
7867             // starting in an empty cell: find the next block of 
7873                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
7876         else if ( m_table
->IsEmptyCell(row
-1, col
) ) 
7878             // starting at the top of a block: find the next block 
7884                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
7889             // starting within a block: find the top of the block 
7894                 if ( m_table
->IsEmptyCell(row
, col
) ) 
7902         MakeCellVisible( row
, col 
); 
7903         if ( expandSelection 
) 
7905             m_selectingKeyboard 
= wxGridCellCoords( row
, col 
); 
7906             HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
7911             SetCurrentCell( row
, col 
); 
7919 bool wxGrid::MoveCursorDownBlock( bool expandSelection 
) 
7922          m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
7923          m_currentCellCoords
.GetRow() < m_numRows
-1 ) 
7925         int row 
= m_currentCellCoords
.GetRow(); 
7926         int col 
= m_currentCellCoords
.GetCol(); 
7928         if ( m_table
->IsEmptyCell(row
, col
) ) 
7930             // starting in an empty cell: find the next block of 
7933             while ( row 
< m_numRows
-1 ) 
7936                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
7939         else if ( m_table
->IsEmptyCell(row
+1, col
) ) 
7941             // starting at the bottom of a block: find the next block 
7944             while ( row 
< m_numRows
-1 ) 
7947                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
7952             // starting within a block: find the bottom of the block 
7954             while ( row 
< m_numRows
-1 ) 
7957                 if ( m_table
->IsEmptyCell(row
, col
) ) 
7965         MakeCellVisible( row
, col 
); 
7966         if ( expandSelection 
) 
7968             m_selectingKeyboard 
= wxGridCellCoords( row
, col 
); 
7969             HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
7974             SetCurrentCell( row
, col 
); 
7983 bool wxGrid::MoveCursorLeftBlock( bool expandSelection 
) 
7986          m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
7987          m_currentCellCoords
.GetCol() > 0 ) 
7989         int row 
= m_currentCellCoords
.GetRow(); 
7990         int col 
= m_currentCellCoords
.GetCol(); 
7992         if ( m_table
->IsEmptyCell(row
, col
) ) 
7994             // starting in an empty cell: find the next block of 
8000                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8003         else if ( m_table
->IsEmptyCell(row
, col
-1) ) 
8005             // starting at the left of a block: find the next block 
8011                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8016             // starting within a block: find the left of the block 
8021                 if ( m_table
->IsEmptyCell(row
, col
) ) 
8029         MakeCellVisible( row
, col 
); 
8030         if ( expandSelection 
) 
8032             m_selectingKeyboard 
= wxGridCellCoords( row
, col 
); 
8033             HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8038             SetCurrentCell( row
, col 
); 
8047 bool wxGrid::MoveCursorRightBlock( bool expandSelection 
) 
8050          m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
8051          m_currentCellCoords
.GetCol() < m_numCols
-1 ) 
8053         int row 
= m_currentCellCoords
.GetRow(); 
8054         int col 
= m_currentCellCoords
.GetCol(); 
8056         if ( m_table
->IsEmptyCell(row
, col
) ) 
8058             // starting in an empty cell: find the next block of 
8061             while ( col 
< m_numCols
-1 ) 
8064                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8067         else if ( m_table
->IsEmptyCell(row
, col
+1) ) 
8069             // starting at the right of a block: find the next block 
8072             while ( col 
< m_numCols
-1 ) 
8075                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8080             // starting within a block: find the right of the block 
8082             while ( col 
< m_numCols
-1 ) 
8085                 if ( m_table
->IsEmptyCell(row
, col
) ) 
8093         MakeCellVisible( row
, col 
); 
8094         if ( expandSelection 
) 
8096             m_selectingKeyboard 
= wxGridCellCoords( row
, col 
); 
8097             HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8102             SetCurrentCell( row
, col 
); 
8114 // ------ Label values and formatting 
8117 void wxGrid::GetRowLabelAlignment( int *horiz
, int *vert 
) 
8119     *horiz 
= m_rowLabelHorizAlign
; 
8120     *vert  
= m_rowLabelVertAlign
; 
8123 void wxGrid::GetColLabelAlignment( int *horiz
, int *vert 
) 
8125     *horiz 
= m_colLabelHorizAlign
; 
8126     *vert  
= m_colLabelVertAlign
; 
8129 wxString 
wxGrid::GetRowLabelValue( int row 
) 
8133         return m_table
->GetRowLabelValue( row 
); 
8143 wxString 
wxGrid::GetColLabelValue( int col 
) 
8147         return m_table
->GetColLabelValue( col 
); 
8158 void wxGrid::SetRowLabelSize( int width 
) 
8160     width 
= wxMax( width
, 0 ); 
8161     if ( width 
!= m_rowLabelWidth 
) 
8165             m_rowLabelWin
->Show( FALSE 
); 
8166             m_cornerLabelWin
->Show( FALSE 
); 
8168         else if ( m_rowLabelWidth 
== 0 ) 
8170             m_rowLabelWin
->Show( TRUE 
); 
8171             if ( m_colLabelHeight 
> 0 ) m_cornerLabelWin
->Show( TRUE 
); 
8174         m_rowLabelWidth 
= width
; 
8176         wxScrolledWindow::Refresh( TRUE 
); 
8181 void wxGrid::SetColLabelSize( int height 
) 
8183     height 
= wxMax( height
, 0 ); 
8184     if ( height 
!= m_colLabelHeight 
) 
8188             m_colLabelWin
->Show( FALSE 
); 
8189             m_cornerLabelWin
->Show( FALSE 
); 
8191         else if ( m_colLabelHeight 
== 0 ) 
8193             m_colLabelWin
->Show( TRUE 
); 
8194             if ( m_rowLabelWidth 
> 0 ) m_cornerLabelWin
->Show( TRUE 
); 
8197         m_colLabelHeight 
= height
; 
8199         wxScrolledWindow::Refresh( TRUE 
); 
8204 void wxGrid::SetLabelBackgroundColour( const wxColour
& colour 
) 
8206     if ( m_labelBackgroundColour 
!= colour 
) 
8208         m_labelBackgroundColour 
= colour
; 
8209         m_rowLabelWin
->SetBackgroundColour( colour 
); 
8210         m_colLabelWin
->SetBackgroundColour( colour 
); 
8211         m_cornerLabelWin
->SetBackgroundColour( colour 
); 
8213         if ( !GetBatchCount() ) 
8215             m_rowLabelWin
->Refresh(); 
8216             m_colLabelWin
->Refresh(); 
8217             m_cornerLabelWin
->Refresh(); 
8222 void wxGrid::SetLabelTextColour( const wxColour
& colour 
) 
8224     if ( m_labelTextColour 
!= colour 
) 
8226         m_labelTextColour 
= colour
; 
8227         if ( !GetBatchCount() ) 
8229             m_rowLabelWin
->Refresh(); 
8230             m_colLabelWin
->Refresh(); 
8235 void wxGrid::SetLabelFont( const wxFont
& font 
) 
8238     if ( !GetBatchCount() ) 
8240         m_rowLabelWin
->Refresh(); 
8241         m_colLabelWin
->Refresh(); 
8245 void wxGrid::SetRowLabelAlignment( int horiz
, int vert 
) 
8247     // allow old (incorrect) defs to be used 
8250         case wxLEFT
:   horiz 
= wxALIGN_LEFT
; break; 
8251         case wxRIGHT
:  horiz 
= wxALIGN_RIGHT
; break; 
8252         case wxCENTRE
: horiz 
= wxALIGN_CENTRE
; break; 
8257         case wxTOP
:    vert 
= wxALIGN_TOP
;    break; 
8258         case wxBOTTOM
: vert 
= wxALIGN_BOTTOM
; break; 
8259         case wxCENTRE
: vert 
= wxALIGN_CENTRE
; break; 
8262     if ( horiz 
== wxALIGN_LEFT 
|| horiz 
== wxALIGN_CENTRE 
|| horiz 
== wxALIGN_RIGHT 
) 
8264         m_rowLabelHorizAlign 
= horiz
; 
8267     if ( vert 
== wxALIGN_TOP 
|| vert 
== wxALIGN_CENTRE 
|| vert 
== wxALIGN_BOTTOM 
) 
8269         m_rowLabelVertAlign 
= vert
; 
8272     if ( !GetBatchCount() ) 
8274         m_rowLabelWin
->Refresh(); 
8278 void wxGrid::SetColLabelAlignment( int horiz
, int vert 
) 
8280     // allow old (incorrect) defs to be used 
8283         case wxLEFT
:   horiz 
= wxALIGN_LEFT
; break; 
8284         case wxRIGHT
:  horiz 
= wxALIGN_RIGHT
; break; 
8285         case wxCENTRE
: horiz 
= wxALIGN_CENTRE
; break; 
8290         case wxTOP
:    vert 
= wxALIGN_TOP
;    break; 
8291         case wxBOTTOM
: vert 
= wxALIGN_BOTTOM
; break; 
8292         case wxCENTRE
: vert 
= wxALIGN_CENTRE
; break; 
8295     if ( horiz 
== wxALIGN_LEFT 
|| horiz 
== wxALIGN_CENTRE 
|| horiz 
== wxALIGN_RIGHT 
) 
8297         m_colLabelHorizAlign 
= horiz
; 
8300     if ( vert 
== wxALIGN_TOP 
|| vert 
== wxALIGN_CENTRE 
|| vert 
== wxALIGN_BOTTOM 
) 
8302         m_colLabelVertAlign 
= vert
; 
8305     if ( !GetBatchCount() ) 
8307         m_colLabelWin
->Refresh(); 
8311 void wxGrid::SetRowLabelValue( int row
, const wxString
& s 
) 
8315         m_table
->SetRowLabelValue( row
, s 
); 
8316         if ( !GetBatchCount() ) 
8318             wxRect rect 
= CellToRect( row
, 0); 
8319             if ( rect
.height 
> 0 ) 
8321                 CalcScrolledPosition(0, rect
.y
, &rect
.x
, &rect
.y
); 
8323                 rect
.width 
= m_rowLabelWidth
; 
8324                 m_rowLabelWin
->Refresh( TRUE
, &rect 
); 
8330 void wxGrid::SetColLabelValue( int col
, const wxString
& s 
) 
8334         m_table
->SetColLabelValue( col
, s 
); 
8335         if ( !GetBatchCount() ) 
8337             wxRect rect 
= CellToRect( 0, col 
); 
8338             if ( rect
.width 
> 0 ) 
8340                 CalcScrolledPosition(rect
.x
, 0, &rect
.x
, &rect
.y
); 
8342                 rect
.height 
= m_colLabelHeight
; 
8343                 m_colLabelWin
->Refresh( TRUE
, &rect 
); 
8349 void wxGrid::SetGridLineColour( const wxColour
& colour 
) 
8351     if ( m_gridLineColour 
!= colour 
) 
8353         m_gridLineColour 
= colour
; 
8355         wxClientDC 
dc( m_gridWin 
); 
8357         DrawAllGridLines( dc
, wxRegion() ); 
8362 void wxGrid::SetCellHighlightColour( const wxColour
& colour 
) 
8364     if ( m_cellHighlightColour 
!= colour 
) 
8366         m_cellHighlightColour 
= colour
; 
8368         wxClientDC 
dc( m_gridWin 
); 
8370         wxGridCellAttr
* attr 
= GetCellAttr(m_currentCellCoords
); 
8371         DrawCellHighlight(dc
, attr
); 
8376 void wxGrid::SetCellHighlightPenWidth(int width
) 
8378     if (m_cellHighlightPenWidth 
!= width
) { 
8379         m_cellHighlightPenWidth 
= width
; 
8381         // Just redrawing the cell highlight is not enough since that won't 
8382         // make any visible change if the the thickness is getting smaller. 
8383         int row 
= m_currentCellCoords
.GetRow(); 
8384         int col 
= m_currentCellCoords
.GetCol(); 
8385         if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
8387         wxRect rect 
= CellToRect(row
, col
); 
8388         m_gridWin
->Refresh(TRUE
, &rect
); 
8392 void wxGrid::SetCellHighlightROPenWidth(int width
) 
8394     if (m_cellHighlightROPenWidth 
!= width
) { 
8395         m_cellHighlightROPenWidth 
= width
; 
8397         // Just redrawing the cell highlight is not enough since that won't 
8398         // make any visible change if the the thickness is getting smaller. 
8399         int row 
= m_currentCellCoords
.GetRow(); 
8400         int col 
= m_currentCellCoords
.GetCol(); 
8401         if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
8403         wxRect rect 
= CellToRect(row
, col
); 
8404         m_gridWin
->Refresh(TRUE
, &rect
); 
8408 void wxGrid::EnableGridLines( bool enable 
) 
8410     if ( enable 
!= m_gridLinesEnabled 
) 
8412         m_gridLinesEnabled 
= enable
; 
8414         if ( !GetBatchCount() ) 
8418                 wxClientDC 
dc( m_gridWin 
); 
8420                 DrawAllGridLines( dc
, wxRegion() ); 
8424                 m_gridWin
->Refresh(); 
8431 int wxGrid::GetDefaultRowSize() 
8433     return m_defaultRowHeight
; 
8436 int wxGrid::GetRowSize( int row 
) 
8438     wxCHECK_MSG( row 
>= 0 && row 
< m_numRows
, 0, _T("invalid row index") ); 
8440     return GetRowHeight(row
); 
8443 int wxGrid::GetDefaultColSize() 
8445     return m_defaultColWidth
; 
8448 int wxGrid::GetColSize( int col 
) 
8450     wxCHECK_MSG( col 
>= 0 && col 
< m_numCols
, 0, _T("invalid column index") ); 
8452     return GetColWidth(col
); 
8455 // ============================================================================ 
8456 // access to the grid attributes: each of them has a default value in the grid 
8457 // itself and may be overidden on a per-cell basis 
8458 // ============================================================================ 
8460 // ---------------------------------------------------------------------------- 
8461 // setting default attributes 
8462 // ---------------------------------------------------------------------------- 
8464 void wxGrid::SetDefaultCellBackgroundColour( const wxColour
& col 
) 
8466     m_defaultCellAttr
->SetBackgroundColour(col
); 
8468     m_gridWin
->SetBackgroundColour(col
); 
8472 void wxGrid::SetDefaultCellTextColour( const wxColour
& col 
) 
8474     m_defaultCellAttr
->SetTextColour(col
); 
8477 void wxGrid::SetDefaultCellAlignment( int horiz
, int vert 
) 
8479     m_defaultCellAttr
->SetAlignment(horiz
, vert
); 
8482 void wxGrid::SetDefaultCellOverflow( bool allow 
) 
8484     m_defaultCellAttr
->SetOverflow(allow
); 
8487 void wxGrid::SetDefaultCellFont( const wxFont
& font 
) 
8489     m_defaultCellAttr
->SetFont(font
); 
8492 void wxGrid::SetDefaultRenderer(wxGridCellRenderer 
*renderer
) 
8494     m_defaultCellAttr
->SetRenderer(renderer
); 
8497 void wxGrid::SetDefaultEditor(wxGridCellEditor 
*editor
) 
8499     m_defaultCellAttr
->SetEditor(editor
); 
8502 // ---------------------------------------------------------------------------- 
8503 // access to the default attrbiutes 
8504 // ---------------------------------------------------------------------------- 
8506 wxColour 
wxGrid::GetDefaultCellBackgroundColour() 
8508     return m_defaultCellAttr
->GetBackgroundColour(); 
8511 wxColour 
wxGrid::GetDefaultCellTextColour() 
8513     return m_defaultCellAttr
->GetTextColour(); 
8516 wxFont 
wxGrid::GetDefaultCellFont() 
8518     return m_defaultCellAttr
->GetFont(); 
8521 void wxGrid::GetDefaultCellAlignment( int *horiz
, int *vert 
) 
8523     m_defaultCellAttr
->GetAlignment(horiz
, vert
); 
8526 bool wxGrid::GetDefaultCellOverflow() 
8528     return m_defaultCellAttr
->GetOverflow(); 
8531 wxGridCellRenderer 
*wxGrid::GetDefaultRenderer() const 
8533     return m_defaultCellAttr
->GetRenderer(NULL
, 0, 0); 
8536 wxGridCellEditor 
*wxGrid::GetDefaultEditor() const 
8538     return m_defaultCellAttr
->GetEditor(NULL
,0,0); 
8541 // ---------------------------------------------------------------------------- 
8542 // access to cell attributes 
8543 // ---------------------------------------------------------------------------- 
8545 wxColour 
wxGrid::GetCellBackgroundColour(int row
, int col
) 
8547     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8548     wxColour colour 
= attr
->GetBackgroundColour(); 
8553 wxColour 
wxGrid::GetCellTextColour( int row
, int col 
) 
8555     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8556     wxColour colour 
= attr
->GetTextColour(); 
8561 wxFont 
wxGrid::GetCellFont( int row
, int col 
) 
8563     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8564     wxFont font 
= attr
->GetFont(); 
8569 void wxGrid::GetCellAlignment( int row
, int col
, int *horiz
, int *vert 
) 
8571     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8572     attr
->GetAlignment(horiz
, vert
); 
8576 bool wxGrid::GetCellOverflow( int row
, int col 
) 
8578     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8579     bool allow 
= attr
->GetOverflow(); 
8584 void wxGrid::GetCellSize( int row
, int col
, int *num_rows
, int *num_cols 
) 
8586     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8587     attr
->GetSize( num_rows
, num_cols 
); 
8591 wxGridCellRenderer
* wxGrid::GetCellRenderer(int row
, int col
) 
8593     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
8594     wxGridCellRenderer
* renderer 
= attr
->GetRenderer(this, row
, col
); 
8600 wxGridCellEditor
* wxGrid::GetCellEditor(int row
, int col
) 
8602     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
8603     wxGridCellEditor
* editor 
= attr
->GetEditor(this, row
, col
); 
8609 bool wxGrid::IsReadOnly(int row
, int col
) const 
8611     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
8612     bool isReadOnly 
= attr
->IsReadOnly(); 
8617 // ---------------------------------------------------------------------------- 
8618 // attribute support: cache, automatic provider creation, ... 
8619 // ---------------------------------------------------------------------------- 
8621 bool wxGrid::CanHaveAttributes() 
8628     return m_table
->CanHaveAttributes(); 
8631 void wxGrid::ClearAttrCache() 
8633     if ( m_attrCache
.row 
!= -1 ) 
8635         wxSafeDecRef(m_attrCache
.attr
); 
8636         m_attrCache
.attr 
= NULL
; 
8637         m_attrCache
.row 
= -1; 
8641 void wxGrid::CacheAttr(int row
, int col
, wxGridCellAttr 
*attr
) const 
8645         wxGrid 
*self 
= (wxGrid 
*)this;  // const_cast 
8647         self
->ClearAttrCache(); 
8648         self
->m_attrCache
.row 
= row
; 
8649         self
->m_attrCache
.col 
= col
; 
8650         self
->m_attrCache
.attr 
= attr
; 
8655 bool wxGrid::LookupAttr(int row
, int col
, wxGridCellAttr 
**attr
) const 
8657     if ( row 
== m_attrCache
.row 
&& col 
== m_attrCache
.col 
) 
8659         *attr 
= m_attrCache
.attr
; 
8660         wxSafeIncRef(m_attrCache
.attr
); 
8662 #ifdef DEBUG_ATTR_CACHE 
8663         gs_nAttrCacheHits
++; 
8670 #ifdef DEBUG_ATTR_CACHE 
8671         gs_nAttrCacheMisses
++; 
8677 wxGridCellAttr 
*wxGrid::GetCellAttr(int row
, int col
) const 
8679     wxGridCellAttr 
*attr 
= NULL
; 
8680     // Additional test to avoid looking at the cache e.g. for 
8681     // wxNoCellCoords, as this will confuse memory management. 
8684         if ( !LookupAttr(row
, col
, &attr
) ) 
8686             attr 
= m_table 
? m_table
->GetAttr(row
, col 
, wxGridCellAttr::Any
) 
8687                            : (wxGridCellAttr 
*)NULL
; 
8688             CacheAttr(row
, col
, attr
); 
8693         attr
->SetDefAttr(m_defaultCellAttr
); 
8697         attr 
= m_defaultCellAttr
; 
8704 wxGridCellAttr 
*wxGrid::GetOrCreateCellAttr(int row
, int col
) const 
8706     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
8708     wxCHECK_MSG( m_table
, attr
, 
8709                   _T("we may only be called if CanHaveAttributes() returned TRUE and then m_table should be !NULL") ); 
8711     attr 
= m_table
->GetAttr(row
, col
, wxGridCellAttr::Cell
); 
8714         attr 
= new wxGridCellAttr(m_defaultCellAttr
); 
8716         // artificially inc the ref count to match DecRef() in caller 
8718         m_table
->SetAttr(attr
, row
, col
); 
8724 // ---------------------------------------------------------------------------- 
8725 // setting column attributes (wrappers around SetColAttr) 
8726 // ---------------------------------------------------------------------------- 
8728 void wxGrid::SetColFormatBool(int col
) 
8730     SetColFormatCustom(col
, wxGRID_VALUE_BOOL
); 
8733 void wxGrid::SetColFormatNumber(int col
) 
8735     SetColFormatCustom(col
, wxGRID_VALUE_NUMBER
); 
8738 void wxGrid::SetColFormatFloat(int col
, int width
, int precision
) 
8740     wxString typeName 
= wxGRID_VALUE_FLOAT
; 
8741     if ( (width 
!= -1) || (precision 
!= -1) ) 
8743         typeName 
<< _T(':') << width 
<< _T(',') << precision
; 
8746     SetColFormatCustom(col
, typeName
); 
8749 void wxGrid::SetColFormatCustom(int col
, const wxString
& typeName
) 
8751     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
8753     attr 
= m_table
->GetAttr(-1, col
, wxGridCellAttr::Col 
); 
8755         attr 
= new wxGridCellAttr
; 
8756     wxGridCellRenderer 
*renderer 
= GetDefaultRendererForType(typeName
); 
8757     attr
->SetRenderer(renderer
); 
8759     SetColAttr(col
, attr
); 
8763 // ---------------------------------------------------------------------------- 
8764 // setting cell attributes: this is forwarded to the table 
8765 // ---------------------------------------------------------------------------- 
8767 void wxGrid::SetAttr(int row
, int col
, wxGridCellAttr 
*attr
) 
8769     if ( CanHaveAttributes() ) 
8771         m_table
->SetAttr(attr
, row
, col
); 
8780 void wxGrid::SetRowAttr(int row
, wxGridCellAttr 
*attr
) 
8782     if ( CanHaveAttributes() ) 
8784         m_table
->SetRowAttr(attr
, row
); 
8793 void wxGrid::SetColAttr(int col
, wxGridCellAttr 
*attr
) 
8795     if ( CanHaveAttributes() ) 
8797         m_table
->SetColAttr(attr
, col
); 
8806 void wxGrid::SetCellBackgroundColour( int row
, int col
, const wxColour
& colour 
) 
8808     if ( CanHaveAttributes() ) 
8810         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
8811         attr
->SetBackgroundColour(colour
); 
8816 void wxGrid::SetCellTextColour( int row
, int col
, const wxColour
& colour 
) 
8818     if ( CanHaveAttributes() ) 
8820         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
8821         attr
->SetTextColour(colour
); 
8826 void wxGrid::SetCellFont( int row
, int col
, const wxFont
& font 
) 
8828     if ( CanHaveAttributes() ) 
8830         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
8831         attr
->SetFont(font
); 
8836 void wxGrid::SetCellAlignment( int row
, int col
, int horiz
, int vert 
) 
8838     if ( CanHaveAttributes() ) 
8840         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
8841         attr
->SetAlignment(horiz
, vert
); 
8846 void wxGrid::SetCellOverflow( int row
, int col
, bool allow 
) 
8848     if ( CanHaveAttributes() ) 
8850         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
8851         attr
->SetOverflow(allow
); 
8856 void wxGrid::SetCellSize( int row
, int col
, int num_rows
, int num_cols 
) 
8858     if ( CanHaveAttributes() ) 
8860         int cell_rows
, cell_cols
; 
8862         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
8863         attr
->GetSize(&cell_rows
, &cell_cols
); 
8864         attr
->SetSize(num_rows
, num_cols
); 
8867         // Cannot set the size of a cell to 0 or negative values 
8868         // While it is perfectly legal to do that, this function cannot 
8869         // handle all the possibilies, do it by hand by getting the CellAttr. 
8870         // You can only set the size of a cell to 1,1 or greater with this fn 
8871         wxASSERT_MSG( !((cell_rows 
< 1) || (cell_cols 
< 1)), 
8872                       wxT("wxGrid::SetCellSize setting cell size that is already part of another cell")); 
8873         wxASSERT_MSG( !((num_rows 
< 1) || (num_cols 
< 1)), 
8874                       wxT("wxGrid::SetCellSize setting cell size to < 1")); 
8876         // if this was already a multicell then "turn off" the other cells first 
8877         if ((cell_rows 
> 1) || (cell_rows 
> 1)) 
8880             for (j
=row
; j
<row
+cell_rows
; j
++) 
8882                 for (i
=col
; i
<col
+cell_cols
; i
++) 
8884                     if ((i 
!= col
) || (j 
!= row
)) 
8886                         wxGridCellAttr 
*attr_stub 
= GetOrCreateCellAttr(j
, i
); 
8887                         attr_stub
->SetSize( 1, 1 ); 
8888                         attr_stub
->DecRef(); 
8894         // mark the cells that will be covered by this cell to 
8895         // negative or zero values to point back at this cell 
8896         if (((num_rows 
> 1) || (num_cols 
> 1)) && (num_rows 
>= 1) && (num_cols 
>= 1)) 
8899             for (j
=row
; j
<row
+num_rows
; j
++) 
8901                 for (i
=col
; i
<col
+num_cols
; i
++) 
8903                     if ((i 
!= col
) || (j 
!= row
)) 
8905                         wxGridCellAttr 
*attr_stub 
= GetOrCreateCellAttr(j
, i
); 
8906                         attr_stub
->SetSize( row
-j
, col
-i 
); 
8907                         attr_stub
->DecRef(); 
8915 void wxGrid::SetCellRenderer(int row
, int col
, wxGridCellRenderer 
*renderer
) 
8917     if ( CanHaveAttributes() ) 
8919         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
8920         attr
->SetRenderer(renderer
); 
8925 void wxGrid::SetCellEditor(int row
, int col
, wxGridCellEditor
* editor
) 
8927     if ( CanHaveAttributes() ) 
8929         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
8930         attr
->SetEditor(editor
); 
8935 void wxGrid::SetReadOnly(int row
, int col
, bool isReadOnly
) 
8937     if ( CanHaveAttributes() ) 
8939         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
8940         attr
->SetReadOnly(isReadOnly
); 
8945 // ---------------------------------------------------------------------------- 
8946 // Data type registration 
8947 // ---------------------------------------------------------------------------- 
8949 void wxGrid::RegisterDataType(const wxString
& typeName
, 
8950                               wxGridCellRenderer
* renderer
, 
8951                               wxGridCellEditor
* editor
) 
8953     m_typeRegistry
->RegisterDataType(typeName
, renderer
, editor
); 
8957 wxGridCellEditor
* wxGrid::GetDefaultEditorForCell(int row
, int col
) const 
8959     wxString typeName 
= m_table
->GetTypeName(row
, col
); 
8960     return GetDefaultEditorForType(typeName
); 
8963 wxGridCellRenderer
* wxGrid::GetDefaultRendererForCell(int row
, int col
) const 
8965     wxString typeName 
= m_table
->GetTypeName(row
, col
); 
8966     return GetDefaultRendererForType(typeName
); 
8970 wxGrid::GetDefaultEditorForType(const wxString
& typeName
) const 
8972     int index 
= m_typeRegistry
->FindOrCloneDataType(typeName
); 
8973     if ( index 
== wxNOT_FOUND 
) 
8975         wxFAIL_MSG(wxT("Unknown data type name")); 
8980     return m_typeRegistry
->GetEditor(index
); 
8984 wxGrid::GetDefaultRendererForType(const wxString
& typeName
) const 
8986     int index 
= m_typeRegistry
->FindOrCloneDataType(typeName
); 
8987     if ( index 
== wxNOT_FOUND 
) 
8989         wxFAIL_MSG(wxT("Unknown data type name")); 
8994     return m_typeRegistry
->GetRenderer(index
); 
8998 // ---------------------------------------------------------------------------- 
9000 // ---------------------------------------------------------------------------- 
9002 void wxGrid::EnableDragRowSize( bool enable 
) 
9004     m_canDragRowSize 
= enable
; 
9008 void wxGrid::EnableDragColSize( bool enable 
) 
9010     m_canDragColSize 
= enable
; 
9013 void wxGrid::EnableDragGridSize( bool enable 
) 
9015     m_canDragGridSize 
= enable
; 
9019 void wxGrid::SetDefaultRowSize( int height
, bool resizeExistingRows 
) 
9021     m_defaultRowHeight 
= wxMax( height
, WXGRID_MIN_ROW_HEIGHT 
); 
9023     if ( resizeExistingRows 
) 
9025         // since we are resizing all rows to the default row size, 
9026         // we can simply clear the row heights and row bottoms 
9027         // arrays (which also allows us to take advantage of 
9028         // some speed optimisations) 
9029         m_rowHeights
.Empty(); 
9030         m_rowBottoms
.Empty(); 
9031         if ( !GetBatchCount() ) 
9036 void wxGrid::SetRowSize( int row
, int height 
) 
9038     wxCHECK_RET( row 
>= 0 && row 
< m_numRows
, _T("invalid row index") ); 
9040     if ( m_rowHeights
.IsEmpty() ) 
9042         // need to really create the array 
9046     int h 
= wxMax( 0, height 
); 
9047     int diff 
= h 
- m_rowHeights
[row
]; 
9049     m_rowHeights
[row
] = h
; 
9051     for ( i 
= row
;  i 
< m_numRows
;  i
++ ) 
9053         m_rowBottoms
[i
] += diff
; 
9055     if ( !GetBatchCount() ) 
9059 void wxGrid::SetDefaultColSize( int width
, bool resizeExistingCols 
) 
9061     m_defaultColWidth 
= wxMax( width
, WXGRID_MIN_COL_WIDTH 
); 
9063     if ( resizeExistingCols 
) 
9065         // since we are resizing all columns to the default column size, 
9066         // we can simply clear the col widths and col rights 
9067         // arrays (which also allows us to take advantage of 
9068         // some speed optimisations) 
9069         m_colWidths
.Empty(); 
9070         m_colRights
.Empty(); 
9071         if ( !GetBatchCount() ) 
9076 void wxGrid::SetColSize( int col
, int width 
) 
9078     wxCHECK_RET( col 
>= 0 && col 
< m_numCols
, _T("invalid column index") ); 
9080     // should we check that it's bigger than GetColMinimalWidth(col) here? 
9082     if ( m_colWidths
.IsEmpty() ) 
9084         // need to really create the array 
9088     int w 
= wxMax( 0, width 
); 
9089     int diff 
= w 
- m_colWidths
[col
]; 
9090     m_colWidths
[col
] = w
; 
9093     for ( i 
= col
;  i 
< m_numCols
;  i
++ ) 
9095         m_colRights
[i
] += diff
; 
9097     if ( !GetBatchCount() ) 
9102 void wxGrid::SetColMinimalWidth( int col
, int width 
) 
9104     m_colMinWidths
.Put(col
, width
); 
9107 void wxGrid::SetRowMinimalHeight( int row
, int width 
) 
9109     m_rowMinHeights
.Put(row
, width
); 
9112 int wxGrid::GetColMinimalWidth(int col
) const 
9114     long value 
= m_colMinWidths
.Get(col
); 
9115     return value 
!= wxNOT_FOUND 
? (int)value 
: WXGRID_MIN_COL_WIDTH
; 
9118 int wxGrid::GetRowMinimalHeight(int row
) const 
9120     long value 
= m_rowMinHeights
.Get(row
); 
9121     return value 
!= wxNOT_FOUND 
? (int)value 
: WXGRID_MIN_ROW_HEIGHT
; 
9124 // ---------------------------------------------------------------------------- 
9126 // ---------------------------------------------------------------------------- 
9128 void wxGrid::AutoSizeColOrRow( int colOrRow
, bool setAsMin
, bool column 
) 
9130     wxClientDC 
dc(m_gridWin
); 
9132     // init both of them to avoid compiler warnings, even if weo nly need one 
9140     wxCoord extent
, extentMax 
= 0; 
9141     int max 
= column 
? m_numRows 
: m_numCols
; 
9142     for ( int rowOrCol 
= 0; rowOrCol 
< max
; rowOrCol
++ ) 
9149         wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
9150         wxGridCellRenderer
* renderer 
= attr
->GetRenderer(this, row
, col
); 
9153             wxSize size 
= renderer
->GetBestSize(*this, *attr
, dc
, row
, col
); 
9154             extent 
= column 
? size
.x 
: size
.y
; 
9155             if ( extent 
> extentMax 
) 
9166     // now also compare with the column label extent 
9168     dc
.SetFont( GetLabelFont() ); 
9171         dc
.GetTextExtent( GetColLabelValue(col
), &w
, &h 
); 
9173         dc
.GetTextExtent( GetRowLabelValue(row
), &w
, &h 
); 
9175     extent 
= column 
? w 
: h
; 
9176     if ( extent 
> extentMax 
) 
9183         // empty column - give default extent (notice that if extentMax is less 
9184         // than default extent but != 0, it's ok) 
9185         extentMax 
= column 
? m_defaultColWidth 
: m_defaultRowHeight
; 
9191             // leave some space around text 
9202         SetColSize(col
, extentMax
); 
9203         if ( !GetBatchCount() ) 
9206             m_gridWin
->GetClientSize( &cw
, &ch 
); 
9207             wxRect 
rect ( CellToRect( 0, col 
) ); 
9209             CalcScrolledPosition(rect
.x
, 0, &rect
.x
, &dummy
); 
9210             rect
.width 
= cw 
- rect
.x
; 
9211             rect
.height 
= m_colLabelHeight
; 
9212             m_colLabelWin
->Refresh( TRUE
, &rect 
); 
9217         SetRowSize(row
, extentMax
); 
9218         if ( !GetBatchCount() ) 
9221             m_gridWin
->GetClientSize( &cw
, &ch 
); 
9222             wxRect 
rect ( CellToRect( row
, 0 ) ); 
9224             CalcScrolledPosition(0, rect
.y
, &dummy
, &rect
.y
); 
9225             rect
.width 
= m_rowLabelWidth
; 
9226             rect
.height 
= ch 
- rect
.y
; 
9227             m_rowLabelWin
->Refresh( TRUE
, &rect 
); 
9233             SetColMinimalWidth(col
, extentMax
); 
9235             SetRowMinimalHeight(row
, extentMax
); 
9239 int wxGrid::SetOrCalcColumnSizes(bool calcOnly
, bool setAsMin
) 
9241     int width 
= m_rowLabelWidth
; 
9246     for ( int col 
= 0; col 
< m_numCols
; col
++ ) 
9250             AutoSizeColumn(col
, setAsMin
); 
9253         width 
+= GetColWidth(col
); 
9262 int wxGrid::SetOrCalcRowSizes(bool calcOnly
, bool setAsMin
) 
9264     int height 
= m_colLabelHeight
; 
9269     for ( int row 
= 0; row 
< m_numRows
; row
++ ) 
9273             AutoSizeRow(row
, setAsMin
); 
9276         height 
+= GetRowHeight(row
); 
9285 void wxGrid::AutoSize() 
9289     wxSize 
size(SetOrCalcColumnSizes(FALSE
), SetOrCalcRowSizes(FALSE
)); 
9291     // round up the size to a multiple of scroll step - this ensures that we 
9292     // won't get the scrollbars if we're sized exactly to this width 
9293     // CalcDimension adds m_extraWidth + 1 etc. to calculate the necessary 
9295     wxSize 
sizeFit(GetScrollX(size
.x 
+ m_extraWidth 
+ 1) * GRID_SCROLL_LINE_X
, 
9296                    GetScrollY(size
.y 
+ m_extraHeight 
+ 1) * GRID_SCROLL_LINE_Y
); 
9298     // distribute the extra space between the columns/rows to avoid having 
9299     // extra white space 
9301     // Remove the extra m_extraWidth + 1 added above 
9302     wxCoord diff 
= sizeFit
.x 
- size
.x 
+ (m_extraWidth 
+ 1); 
9303     if ( diff 
&& m_numCols 
) 
9305         // try to resize the columns uniformly 
9306         wxCoord diffPerCol 
= diff 
/ m_numCols
; 
9309             for ( int col 
= 0; col 
< m_numCols
; col
++ ) 
9311                 SetColSize(col
, GetColWidth(col
) + diffPerCol
); 
9315         // add remaining amount to the last columns 
9316         diff 
-= diffPerCol 
* m_numCols
; 
9319             for ( int col 
= m_numCols 
- 1; col 
>= m_numCols 
- diff
; col
-- ) 
9321                 SetColSize(col
, GetColWidth(col
) + 1); 
9327     diff 
= sizeFit
.y 
- size
.y 
- (m_extraHeight 
+ 1); 
9328     if ( diff 
&& m_numRows 
) 
9330         // try to resize the columns uniformly 
9331         wxCoord diffPerRow 
= diff 
/ m_numRows
; 
9334             for ( int row 
= 0; row 
< m_numRows
; row
++ ) 
9336                 SetRowSize(row
, GetRowHeight(row
) + diffPerRow
); 
9340         // add remaining amount to the last rows 
9341         diff 
-= diffPerRow 
* m_numRows
; 
9344             for ( int row 
= m_numRows 
- 1; row 
>= m_numRows 
- diff
; row
-- ) 
9346                 SetRowSize(row
, GetRowHeight(row
) + 1); 
9353     SetClientSize(sizeFit
); 
9356 wxSize 
wxGrid::DoGetBestSize() const 
9358     // don't set sizes, only calculate them 
9359     wxGrid 
*self 
= (wxGrid 
*)this;  // const_cast 
9362     width 
= self
->SetOrCalcColumnSizes(TRUE
); 
9363     height 
= self
->SetOrCalcRowSizes(TRUE
); 
9365     int maxwidth
, maxheight
; 
9366     wxDisplaySize( & maxwidth
, & maxheight 
); 
9368     if ( width 
> maxwidth 
) width 
= maxwidth
; 
9369     if ( height 
> maxheight 
) height 
= maxheight
; 
9371     return wxSize( width
, height 
); 
9380 wxPen
& wxGrid::GetDividerPen() const 
9385 // ---------------------------------------------------------------------------- 
9386 // cell value accessor functions 
9387 // ---------------------------------------------------------------------------- 
9389 void wxGrid::SetCellValue( int row
, int col
, const wxString
& s 
) 
9393         m_table
->SetValue( row
, col
, s 
); 
9394         if ( !GetBatchCount() ) 
9397             wxRect 
rect( CellToRect( row
, col 
) ); 
9399             rect
.width 
= m_gridWin
->GetClientSize().GetWidth(); 
9400             CalcScrolledPosition(0, rect
.y
, &dummy
, &rect
.y
); 
9401             m_gridWin
->Refresh( FALSE
, &rect 
); 
9404         if ( m_currentCellCoords
.GetRow() == row 
&& 
9405              m_currentCellCoords
.GetCol() == col 
&& 
9406              IsCellEditControlShown()) 
9407              // Note: If we are using IsCellEditControlEnabled, 
9408              // this interacts badly with calling SetCellValue from 
9409              // an EVT_GRID_CELL_CHANGE handler. 
9411             HideCellEditControl(); 
9412             ShowCellEditControl(); // will reread data from table 
9419 // ------ Block, row and col selection 
9422 void wxGrid::SelectRow( int row
, bool addToSelected 
) 
9424     if ( IsSelection() && !addToSelected 
) 
9428         m_selection
->SelectRow( row
, FALSE
, addToSelected 
); 
9432 void wxGrid::SelectCol( int col
, bool addToSelected 
) 
9434     if ( IsSelection() && !addToSelected 
) 
9438         m_selection
->SelectCol( col
, FALSE
, addToSelected 
); 
9442 void wxGrid::SelectBlock( int topRow
, int leftCol
, int bottomRow
, int rightCol
, 
9443                           bool addToSelected 
) 
9445     if ( IsSelection() && !addToSelected 
) 
9449         m_selection
->SelectBlock( topRow
, leftCol
, bottomRow
, rightCol
, 
9450                                   FALSE
, addToSelected 
); 
9454 void wxGrid::SelectAll() 
9456     if ( m_numRows 
> 0 && m_numCols 
> 0 ) 
9459             m_selection
->SelectBlock( 0, 0, m_numRows
-1, m_numCols
-1 ); 
9464 // ------ Cell, row and col deselection 
9467 void wxGrid::DeselectRow( int row 
) 
9472     if ( m_selection
->GetSelectionMode() == wxGrid::wxGridSelectRows 
) 
9474         if ( m_selection
->IsInSelection(row
, 0 ) ) 
9475             m_selection
->ToggleCellSelection( row
, 0); 
9479         int nCols 
= GetNumberCols(); 
9480         for ( int i 
= 0; i 
< nCols 
; i
++ ) 
9482             if ( m_selection
->IsInSelection(row
, i 
) ) 
9483                 m_selection
->ToggleCellSelection( row
, i
); 
9488 void wxGrid::DeselectCol( int col 
) 
9493     if ( m_selection
->GetSelectionMode() == wxGrid::wxGridSelectColumns 
) 
9495         if ( m_selection
->IsInSelection(0, col 
) ) 
9496             m_selection
->ToggleCellSelection( 0, col
); 
9500         int nRows 
= GetNumberRows(); 
9501         for ( int i 
= 0; i 
< nRows 
; i
++ ) 
9503             if ( m_selection
->IsInSelection(i
, col 
) ) 
9504                 m_selection
->ToggleCellSelection(i
, col
); 
9509 void wxGrid::DeselectCell( int row
, int col 
) 
9511     if ( m_selection 
&& m_selection
->IsInSelection(row
, col
) ) 
9512         m_selection
->ToggleCellSelection(row
, col
); 
9515 bool wxGrid::IsSelection() 
9517     return ( m_selection 
&& (m_selection
->IsSelection() || 
9518              ( m_selectingTopLeft 
!= wxGridNoCellCoords 
&& 
9519                m_selectingBottomRight 
!= wxGridNoCellCoords
) ) ); 
9522 bool wxGrid::IsInSelection( int row
, int col 
) const 
9524     return ( m_selection 
&& (m_selection
->IsInSelection( row
, col 
) || 
9525              ( row 
>= m_selectingTopLeft
.GetRow() && 
9526                col 
>= m_selectingTopLeft
.GetCol() && 
9527                row 
<= m_selectingBottomRight
.GetRow() && 
9528                col 
<= m_selectingBottomRight
.GetCol() )) ); 
9531 wxGridCellCoordsArray 
wxGrid::GetSelectedCells() const 
9533     if (!m_selection
) { wxGridCellCoordsArray a
; return a
; } 
9534     return m_selection
->m_cellSelection
; 
9536 wxGridCellCoordsArray 
wxGrid::GetSelectionBlockTopLeft() const 
9538     if (!m_selection
) { wxGridCellCoordsArray a
; return a
; } 
9539     return m_selection
->m_blockSelectionTopLeft
; 
9541 wxGridCellCoordsArray 
wxGrid::GetSelectionBlockBottomRight() const 
9543     if (!m_selection
) { wxGridCellCoordsArray a
; return a
; } 
9544     return m_selection
->m_blockSelectionBottomRight
; 
9546 wxArrayInt 
wxGrid::GetSelectedRows() const 
9548     if (!m_selection
) { wxArrayInt a
; return a
; } 
9549     return m_selection
->m_rowSelection
; 
9551 wxArrayInt 
wxGrid::GetSelectedCols() const 
9553     if (!m_selection
) { wxArrayInt a
; return a
; } 
9554     return m_selection
->m_colSelection
; 
9558 void wxGrid::ClearSelection() 
9560     m_selectingTopLeft 
= wxGridNoCellCoords
; 
9561     m_selectingBottomRight 
= wxGridNoCellCoords
; 
9563         m_selection
->ClearSelection(); 
9567 // This function returns the rectangle that encloses the given block 
9568 // in device coords clipped to the client size of the grid window. 
9570 wxRect 
wxGrid::BlockToDeviceRect( const wxGridCellCoords 
&topLeft
, 
9571                                   const wxGridCellCoords 
&bottomRight 
) 
9573     wxRect 
rect( wxGridNoCellRect 
); 
9576     cellRect 
= CellToRect( topLeft 
); 
9577     if ( cellRect 
!= wxGridNoCellRect 
) 
9583         rect 
= wxRect( 0, 0, 0, 0 ); 
9586     cellRect 
= CellToRect( bottomRight 
); 
9587     if ( cellRect 
!= wxGridNoCellRect 
) 
9593         return wxGridNoCellRect
; 
9597     int left 
= rect
.GetLeft(); 
9598     int top 
= rect
.GetTop(); 
9599     int right 
= rect
.GetRight(); 
9600     int bottom 
= rect
.GetBottom(); 
9602     int leftCol 
= topLeft
.GetCol(); 
9603     int topRow 
= topLeft
.GetRow(); 
9604     int rightCol 
= bottomRight
.GetCol(); 
9605     int bottomRow 
= bottomRight
.GetRow(); 
9628     for ( j 
= topRow
; j 
<= bottomRow
; j
++ ) 
9630         for ( i 
= leftCol
; i 
<= rightCol
; i
++ ) 
9632             if ((j
==topRow
) || (j
==bottomRow
) || (i
==leftCol
) || (i
==rightCol
)) 
9634                 cellRect 
= CellToRect( j
, i 
); 
9636                 if (cellRect
.x 
< left
) 
9638                 if (cellRect
.y 
< top
) 
9640                 if (cellRect
.x 
+ cellRect
.width 
> right
) 
9641                     right 
= cellRect
.x 
+ cellRect
.width
; 
9642                 if (cellRect
.y 
+ cellRect
.height 
> bottom
) 
9643                     bottom 
= cellRect
.y 
+ cellRect
.height
; 
9645             else i 
= rightCol
; // jump over inner cells. 
9649     // convert to scrolled coords 
9651     CalcScrolledPosition( left
, top
, &left
, &top 
); 
9652     CalcScrolledPosition( right
, bottom
, &right
, &bottom 
); 
9655     m_gridWin
->GetClientSize( &cw
, &ch 
); 
9657     if (right 
< 0 || bottom 
< 0 || left 
> cw 
|| top 
> ch
) 
9658         return wxRect( 0, 0, 0, 0); 
9660     rect
.SetLeft( wxMax(0, left
) ); 
9661     rect
.SetTop( wxMax(0, top
) ); 
9662     rect
.SetRight( wxMin(cw
, right
) ); 
9663     rect
.SetBottom( wxMin(ch
, bottom
) ); 
9671 // ------ Grid event classes 
9674 IMPLEMENT_DYNAMIC_CLASS( wxGridEvent
, wxNotifyEvent 
) 
9676 wxGridEvent::wxGridEvent( int id
, wxEventType type
, wxObject
* obj
, 
9677                           int row
, int col
, int x
, int y
, bool sel
, 
9678                           bool control
, bool shift
, bool alt
, bool meta 
) 
9679         : wxNotifyEvent( type
, id 
) 
9686     m_control 
= control
; 
9691     SetEventObject(obj
); 
9695 IMPLEMENT_DYNAMIC_CLASS( wxGridSizeEvent
, wxNotifyEvent 
) 
9697 wxGridSizeEvent::wxGridSizeEvent( int id
, wxEventType type
, wxObject
* obj
, 
9698                                   int rowOrCol
, int x
, int y
, 
9699                                   bool control
, bool shift
, bool alt
, bool meta 
) 
9700         : wxNotifyEvent( type
, id 
) 
9702     m_rowOrCol 
= rowOrCol
; 
9705     m_control 
= control
; 
9710     SetEventObject(obj
); 
9714 IMPLEMENT_DYNAMIC_CLASS( wxGridRangeSelectEvent
, wxNotifyEvent 
) 
9716 wxGridRangeSelectEvent::wxGridRangeSelectEvent(int id
, wxEventType type
, wxObject
* obj
, 
9717                                                const wxGridCellCoords
& topLeft
, 
9718                                                const wxGridCellCoords
& bottomRight
, 
9719                                                bool sel
, bool control
, 
9720                                                bool shift
, bool alt
, bool meta 
) 
9721         : wxNotifyEvent( type
, id 
) 
9723     m_topLeft     
= topLeft
; 
9724     m_bottomRight 
= bottomRight
; 
9726     m_control     
= control
; 
9731     SetEventObject(obj
); 
9735 IMPLEMENT_DYNAMIC_CLASS(wxGridEditorCreatedEvent
, wxCommandEvent
) 
9737 wxGridEditorCreatedEvent::wxGridEditorCreatedEvent(int id
, wxEventType type
, 
9738                                                    wxObject
* obj
, int row
, 
9739                                                    int col
, wxControl
* ctrl
) 
9740     : wxCommandEvent(type
, id
) 
9742     SetEventObject(obj
); 
9749 #endif // !wxUSE_NEW_GRID/wxUSE_NEW_GRID 
9751 #endif // wxUSE_GRID