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 // ---------------------------------------------------------------------------- 
  20 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA) 
  21     #pragma implementation "grid.h" 
  24 // For compilers that support precompilatixon, includes "wx/wx.h". 
  25 #include "wx/wxprec.h" 
  37     #include "wx/dcclient.h" 
  38     #include "wx/settings.h" 
  40     #include "wx/textctrl.h" 
  41     #include "wx/checkbox.h" 
  42     #include "wx/combobox.h" 
  43     #include "wx/valtext.h" 
  47 #include "wx/textfile.h" 
  48 #include "wx/spinctrl.h" 
  49 #include "wx/tokenzr.h" 
  52 #include "wx/generic/gridsel.h" 
  54 #if defined(__WXMOTIF__) 
  55     #define WXUNUSED_MOTIF(identifier)  WXUNUSED(identifier) 
  57     #define WXUNUSED_MOTIF(identifier)  identifier 
  60 #if defined(__WXGTK__) 
  61     #define WXUNUSED_GTK(identifier)    WXUNUSED(identifier) 
  63     #define WXUNUSED_GTK(identifier)    identifier 
  66 // Required for wxIs... functions 
  69 // ---------------------------------------------------------------------------- 
  71 // ---------------------------------------------------------------------------- 
  73 WX_DEFINE_ARRAY_WITH_DECL_PTR(wxGridCellAttr 
*, wxArrayAttrs
, 
  74                                  class WXDLLIMPEXP_ADV
); 
  76 struct wxGridCellWithAttr
 
  78     wxGridCellWithAttr(int row
, int col
, wxGridCellAttr 
*attr_
) 
  79         : coords(row
, col
), attr(attr_
) 
  88     wxGridCellCoords coords
; 
  92 //  DECLARE_NO_COPY_CLASS(wxGridCellWithAttr) 
  93 // without rewriting the macros, which require a public copy constructor. 
  96 WX_DECLARE_OBJARRAY_WITH_DECL(wxGridCellWithAttr
, wxGridCellWithAttrArray
, 
  97                               class WXDLLIMPEXP_ADV
); 
  99 #include "wx/arrimpl.cpp" 
 101 WX_DEFINE_OBJARRAY(wxGridCellCoordsArray
) 
 102 WX_DEFINE_OBJARRAY(wxGridCellWithAttrArray
) 
 104 // ---------------------------------------------------------------------------- 
 106 // ---------------------------------------------------------------------------- 
 108 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_LEFT_CLICK
) 
 109 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_RIGHT_CLICK
) 
 110 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_LEFT_DCLICK
) 
 111 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_RIGHT_DCLICK
) 
 112 DEFINE_EVENT_TYPE(wxEVT_GRID_LABEL_LEFT_CLICK
) 
 113 DEFINE_EVENT_TYPE(wxEVT_GRID_LABEL_RIGHT_CLICK
) 
 114 DEFINE_EVENT_TYPE(wxEVT_GRID_LABEL_LEFT_DCLICK
) 
 115 DEFINE_EVENT_TYPE(wxEVT_GRID_LABEL_RIGHT_DCLICK
) 
 116 DEFINE_EVENT_TYPE(wxEVT_GRID_ROW_SIZE
) 
 117 DEFINE_EVENT_TYPE(wxEVT_GRID_COL_SIZE
) 
 118 DEFINE_EVENT_TYPE(wxEVT_GRID_RANGE_SELECT
) 
 119 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_CHANGE
) 
 120 DEFINE_EVENT_TYPE(wxEVT_GRID_SELECT_CELL
) 
 121 DEFINE_EVENT_TYPE(wxEVT_GRID_EDITOR_SHOWN
) 
 122 DEFINE_EVENT_TYPE(wxEVT_GRID_EDITOR_HIDDEN
) 
 123 DEFINE_EVENT_TYPE(wxEVT_GRID_EDITOR_CREATED
) 
 125 // ---------------------------------------------------------------------------- 
 127 // ---------------------------------------------------------------------------- 
 129 class WXDLLIMPEXP_ADV wxGridRowLabelWindow 
: public wxWindow
 
 132     wxGridRowLabelWindow() { m_owner 
= (wxGrid 
*)NULL
; } 
 133     wxGridRowLabelWindow( wxGrid 
*parent
, wxWindowID id
, 
 134                           const wxPoint 
&pos
, const wxSize 
&size 
); 
 139     void OnPaint( wxPaintEvent
& event 
); 
 140     void OnMouseEvent( wxMouseEvent
& event 
); 
 141     void OnMouseWheel( wxMouseEvent
& event 
); 
 142     void OnKeyDown( wxKeyEvent
& event 
); 
 143     void OnKeyUp( wxKeyEvent
& ); 
 145     DECLARE_DYNAMIC_CLASS(wxGridRowLabelWindow
) 
 146     DECLARE_EVENT_TABLE() 
 147     DECLARE_NO_COPY_CLASS(wxGridRowLabelWindow
) 
 151 class WXDLLIMPEXP_ADV wxGridColLabelWindow 
: public wxWindow
 
 154     wxGridColLabelWindow() { m_owner 
= (wxGrid 
*)NULL
; } 
 155     wxGridColLabelWindow( wxGrid 
*parent
, wxWindowID id
, 
 156                           const wxPoint 
&pos
, const wxSize 
&size 
); 
 161     void OnPaint( wxPaintEvent 
&event 
); 
 162     void OnMouseEvent( wxMouseEvent
& event 
); 
 163     void OnMouseWheel( wxMouseEvent
& event 
); 
 164     void OnKeyDown( wxKeyEvent
& event 
); 
 165     void OnKeyUp( wxKeyEvent
& ); 
 167     DECLARE_DYNAMIC_CLASS(wxGridColLabelWindow
) 
 168     DECLARE_EVENT_TABLE() 
 169     DECLARE_NO_COPY_CLASS(wxGridColLabelWindow
) 
 173 class WXDLLIMPEXP_ADV wxGridCornerLabelWindow 
: public wxWindow
 
 176     wxGridCornerLabelWindow() { m_owner 
= (wxGrid 
*)NULL
; } 
 177     wxGridCornerLabelWindow( wxGrid 
*parent
, wxWindowID id
, 
 178                              const wxPoint 
&pos
, const wxSize 
&size 
); 
 183     void OnMouseEvent( wxMouseEvent
& event 
); 
 184     void OnMouseWheel( wxMouseEvent
& event 
); 
 185     void OnKeyDown( wxKeyEvent
& event 
); 
 186     void OnKeyUp( wxKeyEvent
& ); 
 187     void OnPaint( wxPaintEvent
& event 
); 
 189     DECLARE_DYNAMIC_CLASS(wxGridCornerLabelWindow
) 
 190     DECLARE_EVENT_TABLE() 
 191     DECLARE_NO_COPY_CLASS(wxGridCornerLabelWindow
) 
 194 class WXDLLIMPEXP_ADV wxGridWindow 
: public wxWindow
 
 199         m_owner 
= (wxGrid 
*)NULL
; 
 200         m_rowLabelWin 
= (wxGridRowLabelWindow 
*)NULL
; 
 201         m_colLabelWin 
= (wxGridColLabelWindow 
*)NULL
; 
 204     wxGridWindow( wxGrid 
*parent
, 
 205                   wxGridRowLabelWindow 
*rowLblWin
, 
 206                   wxGridColLabelWindow 
*colLblWin
, 
 207                   wxWindowID id
, const wxPoint 
&pos
, const wxSize 
&size 
); 
 210     void ScrollWindow( int dx
, int dy
, const wxRect 
*rect 
); 
 212     wxGrid
* GetOwner() { return m_owner
; } 
 216     wxGridRowLabelWindow     
*m_rowLabelWin
; 
 217     wxGridColLabelWindow     
*m_colLabelWin
; 
 219     void OnPaint( wxPaintEvent 
&event 
); 
 220     void OnMouseWheel( wxMouseEvent
& event 
); 
 221     void OnMouseEvent( wxMouseEvent
& event 
); 
 222     void OnKeyDown( wxKeyEvent
& ); 
 223     void OnKeyUp( wxKeyEvent
& ); 
 224     void OnEraseBackground( wxEraseEvent
& ); 
 225     void OnFocus( wxFocusEvent
& ); 
 227     DECLARE_DYNAMIC_CLASS(wxGridWindow
) 
 228     DECLARE_EVENT_TABLE() 
 229     DECLARE_NO_COPY_CLASS(wxGridWindow
) 
 234 class wxGridCellEditorEvtHandler 
: public wxEvtHandler
 
 237     wxGridCellEditorEvtHandler() 
 238         : m_grid(0), m_editor(0) 
 240     wxGridCellEditorEvtHandler(wxGrid
* grid
, wxGridCellEditor
* editor
) 
 241         : m_grid(grid
), m_editor(editor
) 
 244     void OnKeyDown(wxKeyEvent
& event
); 
 245     void OnChar(wxKeyEvent
& event
); 
 249     wxGridCellEditor
*   m_editor
; 
 250     DECLARE_DYNAMIC_CLASS(wxGridCellEditorEvtHandler
) 
 251     DECLARE_EVENT_TABLE() 
 252     DECLARE_NO_COPY_CLASS(wxGridCellEditorEvtHandler
) 
 256 IMPLEMENT_DYNAMIC_CLASS( wxGridCellEditorEvtHandler
, wxEvtHandler 
) 
 257 BEGIN_EVENT_TABLE( wxGridCellEditorEvtHandler
, wxEvtHandler 
) 
 258     EVT_KEY_DOWN( wxGridCellEditorEvtHandler::OnKeyDown 
) 
 259     EVT_CHAR( wxGridCellEditorEvtHandler::OnChar 
) 
 264 // ---------------------------------------------------------------------------- 
 265 // the internal data representation used by wxGridCellAttrProvider 
 266 // ---------------------------------------------------------------------------- 
 268 // this class stores attributes set for cells 
 269 class WXDLLIMPEXP_ADV wxGridCellAttrData
 
 272     void SetAttr(wxGridCellAttr 
*attr
, int row
, int col
); 
 273     wxGridCellAttr 
*GetAttr(int row
, int col
) const; 
 274     void UpdateAttrRows( size_t pos
, int numRows 
); 
 275     void UpdateAttrCols( size_t pos
, int numCols 
); 
 278     // searches for the attr for given cell, returns wxNOT_FOUND if not found 
 279     int FindIndex(int row
, int col
) const; 
 281     wxGridCellWithAttrArray m_attrs
; 
 284 // this class stores attributes set for rows or columns 
 285 class WXDLLIMPEXP_ADV wxGridRowOrColAttrData
 
 288     // empty ctor to suppress warnings 
 289     wxGridRowOrColAttrData() { } 
 290     ~wxGridRowOrColAttrData(); 
 292     void SetAttr(wxGridCellAttr 
*attr
, int rowOrCol
); 
 293     wxGridCellAttr 
*GetAttr(int rowOrCol
) const; 
 294     void UpdateAttrRowsOrCols( size_t pos
, int numRowsOrCols 
); 
 297     wxArrayInt m_rowsOrCols
; 
 298     wxArrayAttrs m_attrs
; 
 301 // NB: this is just a wrapper around 3 objects: one which stores cell 
 302 //     attributes, and 2 others for row/col ones 
 303 class WXDLLIMPEXP_ADV wxGridCellAttrProviderData
 
 306     wxGridCellAttrData m_cellAttrs
; 
 307     wxGridRowOrColAttrData m_rowAttrs
, 
 312 // ---------------------------------------------------------------------------- 
 313 // data structures used for the data type registry 
 314 // ---------------------------------------------------------------------------- 
 316 struct wxGridDataTypeInfo
 
 318     wxGridDataTypeInfo(const wxString
& typeName
, 
 319                        wxGridCellRenderer
* renderer
, 
 320                        wxGridCellEditor
* editor
) 
 321         : m_typeName(typeName
), m_renderer(renderer
), m_editor(editor
) 
 324     ~wxGridDataTypeInfo() 
 326         wxSafeDecRef(m_renderer
); 
 327         wxSafeDecRef(m_editor
); 
 331     wxGridCellRenderer
* m_renderer
; 
 332     wxGridCellEditor
*   m_editor
; 
 334     DECLARE_NO_COPY_CLASS(wxGridDataTypeInfo
) 
 338 WX_DEFINE_ARRAY_WITH_DECL_PTR(wxGridDataTypeInfo
*, wxGridDataTypeInfoArray
, 
 339                                  class WXDLLIMPEXP_ADV
); 
 342 class WXDLLIMPEXP_ADV wxGridTypeRegistry
 
 345   wxGridTypeRegistry() {} 
 346     ~wxGridTypeRegistry(); 
 348     void RegisterDataType(const wxString
& typeName
, 
 349                      wxGridCellRenderer
* renderer
, 
 350                      wxGridCellEditor
* editor
); 
 352     // find one of already registered data types 
 353     int FindRegisteredDataType(const wxString
& typeName
); 
 355     // try to FindRegisteredDataType(), if this fails and typeName is one of 
 356     // standard typenames, register it and return its index 
 357     int FindDataType(const wxString
& typeName
); 
 359     // try to FindDataType(), if it fails see if it is not one of already 
 360     // registered data types with some params in which case clone the 
 361     // registered data type and set params for it 
 362     int FindOrCloneDataType(const wxString
& typeName
); 
 364     wxGridCellRenderer
* GetRenderer(int index
); 
 365     wxGridCellEditor
*   GetEditor(int index
); 
 368     wxGridDataTypeInfoArray m_typeinfo
; 
 371 // ---------------------------------------------------------------------------- 
 372 // conditional compilation 
 373 // ---------------------------------------------------------------------------- 
 375 #ifndef WXGRID_DRAW_LINES 
 376 #define WXGRID_DRAW_LINES 1 
 379 // ---------------------------------------------------------------------------- 
 381 // ---------------------------------------------------------------------------- 
 383 //#define DEBUG_ATTR_CACHE 
 384 #ifdef DEBUG_ATTR_CACHE 
 385     static size_t gs_nAttrCacheHits 
= 0; 
 386     static size_t gs_nAttrCacheMisses 
= 0; 
 387 #endif // DEBUG_ATTR_CACHE 
 389 // ---------------------------------------------------------------------------- 
 391 // ---------------------------------------------------------------------------- 
 393 wxGridCellCoords 
wxGridNoCellCoords( -1, -1 ); 
 394 wxRect           
wxGridNoCellRect( -1, -1, -1, -1 ); 
 397 // TODO: this doesn't work at all, grid cells have different sizes and approx 
 398 //       calculations don't work as because of the size mismatch scrollbars 
 399 //       sometimes fail to be shown when they should be or vice versa 
 401 //       The scroll bars may be a little flakey once in a while, but that is 
 402 //       surely much less horrible than having scroll lines of only 1!!! 
 405 //       Well, it's still seriously broken so it might be better but needs 
 408 static const size_t GRID_SCROLL_LINE_X 
= 15;  // 1; 
 409 static const size_t GRID_SCROLL_LINE_Y 
= GRID_SCROLL_LINE_X
; 
 411 // the size of hash tables used a bit everywhere (the max number of elements 
 412 // in these hash tables is the number of rows/columns) 
 413 static const int GRID_HASH_SIZE 
= 100; 
 415 // ---------------------------------------------------------------------------- 
 417 // ---------------------------------------------------------------------------- 
 419 static inline int GetScrollX(int x
) 
 421     return (x 
+ GRID_SCROLL_LINE_X 
- 1) / GRID_SCROLL_LINE_X
; 
 424 static inline int GetScrollY(int y
) 
 426     return (y 
+ GRID_SCROLL_LINE_Y 
- 1) / GRID_SCROLL_LINE_Y
; 
 429 // ============================================================================ 
 431 // ============================================================================ 
 433 // ---------------------------------------------------------------------------- 
 435 // ---------------------------------------------------------------------------- 
 437 wxGridCellEditor::wxGridCellEditor() 
 444 wxGridCellEditor::~wxGridCellEditor() 
 449 void wxGridCellEditor::Create(wxWindow
* WXUNUSED(parent
), 
 450                               wxWindowID 
WXUNUSED(id
), 
 451                               wxEvtHandler
* evtHandler
) 
 454         m_control
->PushEventHandler(evtHandler
); 
 457 void wxGridCellEditor::PaintBackground(const wxRect
& rectCell
, 
 458                                        wxGridCellAttr 
*attr
) 
 460     // erase the background because we might not fill the cell 
 461     wxClientDC 
dc(m_control
->GetParent()); 
 462     wxGridWindow
* gridWindow 
= wxDynamicCast(m_control
->GetParent(), wxGridWindow
); 
 464         gridWindow
->GetOwner()->PrepareDC(dc
); 
 466     dc
.SetPen(*wxTRANSPARENT_PEN
); 
 467     dc
.SetBrush(wxBrush(attr
->GetBackgroundColour(), wxSOLID
)); 
 468     dc
.DrawRectangle(rectCell
); 
 470     // redraw the control we just painted over 
 471     m_control
->Refresh(); 
 474 void wxGridCellEditor::Destroy() 
 478         m_control
->PopEventHandler(TRUE 
/* delete it*/); 
 480         m_control
->Destroy(); 
 485 void wxGridCellEditor::Show(bool show
, wxGridCellAttr 
*attr
) 
 487     wxASSERT_MSG(m_control
, 
 488                  wxT("The wxGridCellEditor must be Created first!")); 
 489     m_control
->Show(show
); 
 493         // set the colours/fonts if we have any 
 496             m_colFgOld 
= m_control
->GetForegroundColour(); 
 497             m_control
->SetForegroundColour(attr
->GetTextColour()); 
 499             m_colBgOld 
= m_control
->GetBackgroundColour(); 
 500             m_control
->SetBackgroundColour(attr
->GetBackgroundColour()); 
 502             m_fontOld 
= m_control
->GetFont(); 
 503             m_control
->SetFont(attr
->GetFont()); 
 505             // can't do anything more in the base class version, the other 
 506             // attributes may only be used by the derived classes 
 511         // restore the standard colours fonts 
 512         if ( m_colFgOld
.Ok() ) 
 514             m_control
->SetForegroundColour(m_colFgOld
); 
 515             m_colFgOld 
= wxNullColour
; 
 518         if ( m_colBgOld
.Ok() ) 
 520             m_control
->SetBackgroundColour(m_colBgOld
); 
 521             m_colBgOld 
= wxNullColour
; 
 524         if ( m_fontOld
.Ok() ) 
 526             m_control
->SetFont(m_fontOld
); 
 527             m_fontOld 
= wxNullFont
; 
 532 void wxGridCellEditor::SetSize(const wxRect
& rect
) 
 534     wxASSERT_MSG(m_control
, 
 535                  wxT("The wxGridCellEditor must be Created first!")); 
 536     m_control
->SetSize(rect
, wxSIZE_ALLOW_MINUS_ONE
); 
 539 void wxGridCellEditor::HandleReturn(wxKeyEvent
& event
) 
 544 bool wxGridCellEditor::IsAcceptedKey(wxKeyEvent
& event
) 
 546     // accept the simple key presses, not anything with Ctrl/Alt/Meta 
 547     return !(event
.ControlDown() || event
.AltDown()); 
 550 void wxGridCellEditor::StartingKey(wxKeyEvent
& event
) 
 555 void wxGridCellEditor::StartingClick() 
 561 // ---------------------------------------------------------------------------- 
 562 // wxGridCellTextEditor 
 563 // ---------------------------------------------------------------------------- 
 565 wxGridCellTextEditor::wxGridCellTextEditor() 
 570 void wxGridCellTextEditor::Create(wxWindow
* parent
, 
 572                                   wxEvtHandler
* evtHandler
) 
 574     m_control 
= new wxTextCtrl(parent
, id
, wxEmptyString
, 
 575                                wxDefaultPosition
, wxDefaultSize
 
 576 #if defined(__WXMSW__) 
 577                                , wxTE_PROCESS_TAB 
| wxTE_AUTO_SCROLL
 
 581     // set max length allowed in the textctrl, if the parameter was set 
 584         ((wxTextCtrl
*)m_control
)->SetMaxLength(m_maxChars
); 
 587     wxGridCellEditor::Create(parent
, id
, evtHandler
); 
 590 void wxGridCellTextEditor::PaintBackground(const wxRect
& WXUNUSED(rectCell
), 
 591                                            wxGridCellAttr 
* WXUNUSED(attr
)) 
 593     // as we fill the entire client area, don't do anything here to minimize 
 597 void wxGridCellTextEditor::SetSize(const wxRect
& rectOrig
) 
 599     wxRect 
rect(rectOrig
); 
 601     // Make the edit control large enough to allow for internal 
 604     // TODO: remove this if the text ctrl sizing is improved esp. for 
 607 #if defined(__WXGTK__) 
 616     int extra_x 
= ( rect
.x 
> 2 )? 2 : 1; 
 618 // MB: treat MSW separately here otherwise the caret doesn't show 
 619 // when the editor is in the first row. 
 620 #if defined(__WXMSW__) 
 623     int extra_y 
= ( rect
.y 
> 2 )? 2 : 1; 
 626 #if defined(__WXMOTIF__) 
 630     rect
.SetLeft( wxMax(0, rect
.x 
- extra_x
) ); 
 631     rect
.SetTop( wxMax(0, rect
.y 
- extra_y
) ); 
 632     rect
.SetRight( rect
.GetRight() + 2*extra_x 
); 
 633     rect
.SetBottom( rect
.GetBottom() + 2*extra_y 
); 
 636     wxGridCellEditor::SetSize(rect
); 
 639 void wxGridCellTextEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
 641     wxASSERT_MSG(m_control
, 
 642                  wxT("The wxGridCellEditor must be Created first!")); 
 644     m_startValue 
= grid
->GetTable()->GetValue(row
, col
); 
 646     DoBeginEdit(m_startValue
); 
 649 void wxGridCellTextEditor::DoBeginEdit(const wxString
& startValue
) 
 651     Text()->SetValue(startValue
); 
 652     Text()->SetInsertionPointEnd(); 
 653     Text()->SetSelection(-1,-1); 
 657 bool wxGridCellTextEditor::EndEdit(int row
, int col
, 
 660     wxASSERT_MSG(m_control
, 
 661                  wxT("The wxGridCellEditor must be Created first!")); 
 663     bool changed 
= FALSE
; 
 664     wxString value 
= Text()->GetValue(); 
 665     if (value 
!= m_startValue
) 
 669         grid
->GetTable()->SetValue(row
, col
, value
); 
 671     m_startValue 
= wxEmptyString
; 
 672     // No point in setting the text of the hidden control 
 673     //Text()->SetValue(m_startValue); 
 679 void wxGridCellTextEditor::Reset() 
 681     wxASSERT_MSG(m_control
, 
 682                  wxT("The wxGridCellEditor must be Created first!")); 
 684     DoReset(m_startValue
); 
 687 void wxGridCellTextEditor::DoReset(const wxString
& startValue
) 
 689     Text()->SetValue(startValue
); 
 690     Text()->SetInsertionPointEnd(); 
 693 bool wxGridCellTextEditor::IsAcceptedKey(wxKeyEvent
& event
) 
 695     if ( wxGridCellEditor::IsAcceptedKey(event
) ) 
 697         int keycode 
= event
.GetKeyCode(); 
 711             case WXK_NUMPAD_MULTIPLY
: 
 715             case WXK_NUMPAD_SUBTRACT
: 
 717             case WXK_NUMPAD_DECIMAL
: 
 719             case WXK_NUMPAD_DIVIDE
: 
 723                 // accept 8 bit chars too if isprint() agrees 
 724                 if ( (keycode 
< 255) && (wxIsprint(keycode
)) ) 
 732 void wxGridCellTextEditor::StartingKey(wxKeyEvent
& event
) 
 734     if ( !Text()->EmulateKeyPress(event
) ) 
 740 void wxGridCellTextEditor::HandleReturn( wxKeyEvent
& 
 741                                          WXUNUSED_GTK(WXUNUSED_MOTIF(event
)) ) 
 743 #if defined(__WXMOTIF__) || defined(__WXGTK__) 
 744     // wxMotif needs a little extra help... 
 745     size_t pos 
= (size_t)( Text()->GetInsertionPoint() ); 
 746     wxString 
s( Text()->GetValue() ); 
 747     s 
= s
.Left(pos
) + wxT("\n") + s
.Mid(pos
); 
 749     Text()->SetInsertionPoint( pos 
); 
 751     // the other ports can handle a Return key press 
 757 void wxGridCellTextEditor::SetParameters(const wxString
& params
) 
 767         if ( !params
.ToLong(&tmp
) ) 
 769             wxLogDebug(_T("Invalid wxGridCellTextEditor parameter string '%s' ignored"), params
.c_str()); 
 773             m_maxChars 
= (size_t)tmp
; 
 778 // return the value in the text control 
 779 wxString 
wxGridCellTextEditor::GetValue() const 
 781   return Text()->GetValue(); 
 784 // ---------------------------------------------------------------------------- 
 785 // wxGridCellNumberEditor 
 786 // ---------------------------------------------------------------------------- 
 788 wxGridCellNumberEditor::wxGridCellNumberEditor(int min
, int max
) 
 794 void wxGridCellNumberEditor::Create(wxWindow
* parent
, 
 796                                     wxEvtHandler
* evtHandler
) 
 800         // create a spin ctrl 
 801         m_control 
= new wxSpinCtrl(parent
, -1, wxEmptyString
, 
 802                                    wxDefaultPosition
, wxDefaultSize
, 
 806         wxGridCellEditor::Create(parent
, id
, evtHandler
); 
 810         // just a text control 
 811         wxGridCellTextEditor::Create(parent
, id
, evtHandler
); 
 814         Text()->SetValidator(wxTextValidator(wxFILTER_NUMERIC
)); 
 815 #endif // wxUSE_VALIDATORS 
 819 void wxGridCellNumberEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
 821     // first get the value 
 822     wxGridTableBase 
*table 
= grid
->GetTable(); 
 823     if ( table
->CanGetValueAs(row
, col
, wxGRID_VALUE_NUMBER
) ) 
 825         m_valueOld 
= table
->GetValueAsLong(row
, col
); 
 830         wxString sValue 
= table
->GetValue(row
, col
); 
 831         if (! sValue
.ToLong(&m_valueOld
) && ! sValue
.IsEmpty()) 
 833             wxFAIL_MSG( _T("this cell doesn't have numeric value") ); 
 840         Spin()->SetValue((int)m_valueOld
); 
 845         DoBeginEdit(GetString()); 
 849 bool wxGridCellNumberEditor::EndEdit(int row
, int col
, 
 858         value 
= Spin()->GetValue(); 
 859         changed 
= value 
!= m_valueOld
; 
 861             text 
= wxString::Format(wxT("%ld"), value
); 
 865         text 
= Text()->GetValue(); 
 866         changed 
= (text
.IsEmpty() || text
.ToLong(&value
)) && (value 
!= m_valueOld
); 
 871         if (grid
->GetTable()->CanSetValueAs(row
, col
, wxGRID_VALUE_NUMBER
)) 
 872             grid
->GetTable()->SetValueAsLong(row
, col
, value
); 
 874             grid
->GetTable()->SetValue(row
, col
, text
); 
 880 void wxGridCellNumberEditor::Reset() 
 884         Spin()->SetValue((int)m_valueOld
); 
 888         DoReset(GetString()); 
 892 bool wxGridCellNumberEditor::IsAcceptedKey(wxKeyEvent
& event
) 
 894     if ( wxGridCellEditor::IsAcceptedKey(event
) ) 
 896         int keycode 
= event
.GetKeyCode(); 
 912             case WXK_NUMPAD_SUBTRACT
: 
 918                 if ( (keycode 
< 128) && wxIsdigit(keycode
) ) 
 926 void wxGridCellNumberEditor::StartingKey(wxKeyEvent
& event
) 
 930         int keycode 
= event
.GetKeyCode(); 
 931         if ( wxIsdigit(keycode
) || keycode 
== '+' || keycode 
== '-' 
 932             || keycode 
==  WXK_NUMPAD0
 
 933             || keycode 
==  WXK_NUMPAD1
 
 934             || keycode 
==  WXK_NUMPAD2
 
 935             || keycode 
==  WXK_NUMPAD3
 
 936             || keycode 
==  WXK_NUMPAD4
 
 937             || keycode 
==  WXK_NUMPAD5
 
 938             || keycode 
==  WXK_NUMPAD6
 
 939             || keycode 
==  WXK_NUMPAD7
 
 940             || keycode 
==  WXK_NUMPAD8
 
 941             || keycode 
==  WXK_NUMPAD9
 
 942             || keycode 
==  WXK_ADD
 
 943             || keycode 
==  WXK_NUMPAD_ADD
 
 944             || keycode 
==  WXK_SUBTRACT
 
 945             || keycode 
==  WXK_NUMPAD_SUBTRACT
) 
 947             wxGridCellTextEditor::StartingKey(event
); 
 957 void wxGridCellNumberEditor::SetParameters(const wxString
& params
) 
 968         if ( params
.BeforeFirst(_T(',')).ToLong(&tmp
) ) 
 972             if ( params
.AfterFirst(_T(',')).ToLong(&tmp
) ) 
 976                 // skip the error message below 
 981         wxLogDebug(_T("Invalid wxGridCellNumberEditor parameter string '%s' ignored"), params
.c_str()); 
 985 // return the value in the spin control if it is there (the text control otherwise) 
 986 wxString 
wxGridCellNumberEditor::GetValue() const 
 992     long value 
= Spin()->GetValue(); 
 993     s
.Printf(wxT("%ld"), value
); 
 997     s 
= Text()->GetValue(); 
1002 // ---------------------------------------------------------------------------- 
1003 // wxGridCellFloatEditor 
1004 // ---------------------------------------------------------------------------- 
1006 wxGridCellFloatEditor::wxGridCellFloatEditor(int width
, int precision
) 
1009     m_precision 
= precision
; 
1012 void wxGridCellFloatEditor::Create(wxWindow
* parent
, 
1014                                    wxEvtHandler
* evtHandler
) 
1016     wxGridCellTextEditor::Create(parent
, id
, evtHandler
); 
1018 #if wxUSE_VALIDATORS 
1019     Text()->SetValidator(wxTextValidator(wxFILTER_NUMERIC
)); 
1020 #endif // wxUSE_VALIDATORS 
1023 void wxGridCellFloatEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
1025     // first get the value 
1026     wxGridTableBase 
*table 
= grid
->GetTable(); 
1027     if ( table
->CanGetValueAs(row
, col
, wxGRID_VALUE_FLOAT
) ) 
1029         m_valueOld 
= table
->GetValueAsDouble(row
, col
); 
1034         wxString sValue 
= table
->GetValue(row
, col
); 
1035         if (! sValue
.ToDouble(&m_valueOld
) && ! sValue
.IsEmpty()) 
1037             wxFAIL_MSG( _T("this cell doesn't have float value") ); 
1042     DoBeginEdit(GetString()); 
1045 bool wxGridCellFloatEditor::EndEdit(int row
, int col
, 
1049     wxString 
text(Text()->GetValue()); 
1051     if ( (text
.IsEmpty() || text
.ToDouble(&value
)) && (value 
!= m_valueOld
) ) 
1053         if (grid
->GetTable()->CanSetValueAs(row
, col
, wxGRID_VALUE_FLOAT
)) 
1054             grid
->GetTable()->SetValueAsDouble(row
, col
, value
); 
1056             grid
->GetTable()->SetValue(row
, col
, text
); 
1063 void wxGridCellFloatEditor::Reset() 
1065     DoReset(GetString()); 
1068 void wxGridCellFloatEditor::StartingKey(wxKeyEvent
& event
) 
1070     int keycode 
= event
.GetKeyCode(); 
1072     tmpbuf
[0] = (char) keycode
; 
1074     bool is_decimal_point 
= ( wxString(tmpbuf
, *wxConvCurrent
) ==  
1075       wxLocale::GetInfo(wxLOCALE_DECIMAL_POINT
, wxLOCALE_CAT_NUMBER
) ); 
1076         if ( wxIsdigit(keycode
) || keycode 
== '+' || keycode 
== '-' 
1078             || keycode 
==  WXK_NUMPAD0
 
1079             || keycode 
==  WXK_NUMPAD1
 
1080             || keycode 
==  WXK_NUMPAD2
 
1081             || keycode 
==  WXK_NUMPAD3
 
1082             || keycode 
==  WXK_NUMPAD4
 
1083             || keycode 
==  WXK_NUMPAD5
 
1084             || keycode 
==  WXK_NUMPAD6
 
1085             || keycode 
==  WXK_NUMPAD7
 
1086             || keycode 
==  WXK_NUMPAD8
 
1087             || keycode 
==  WXK_NUMPAD9
 
1088             || keycode 
==  WXK_ADD
 
1089             || keycode 
==  WXK_NUMPAD_ADD
 
1090             || keycode 
==  WXK_SUBTRACT
 
1091             || keycode 
==  WXK_NUMPAD_SUBTRACT
) 
1093         wxGridCellTextEditor::StartingKey(event
); 
1095         // skip Skip() below 
1102 void wxGridCellFloatEditor::SetParameters(const wxString
& params
) 
1113         if ( params
.BeforeFirst(_T(',')).ToLong(&tmp
) ) 
1117             if ( params
.AfterFirst(_T(',')).ToLong(&tmp
) ) 
1119                 m_precision 
= (int)tmp
; 
1121                 // skip the error message below 
1126         wxLogDebug(_T("Invalid wxGridCellFloatEditor parameter string '%s' ignored"), params
.c_str()); 
1130 wxString 
wxGridCellFloatEditor::GetString() const 
1133     if ( m_width 
== -1 ) 
1135         // default width/precision 
1138     else if ( m_precision 
== -1 ) 
1140         // default precision 
1141         fmt
.Printf(_T("%%%d.f"), m_width
); 
1145         fmt
.Printf(_T("%%%d.%df"), m_width
, m_precision
); 
1148     return wxString::Format(fmt
, m_valueOld
); 
1151 bool wxGridCellFloatEditor::IsAcceptedKey(wxKeyEvent
& event
) 
1153     if ( wxGridCellEditor::IsAcceptedKey(event
) ) 
1155         int keycode 
= event
.GetKeyCode(); 
1169             case WXK_NUMPAD_ADD
: 
1171             case WXK_NUMPAD_SUBTRACT
: 
1173             case WXK_NUMPAD_DECIMAL
: 
1178                // additionally accept 'e' as in '1e+6', also '-', '+', and '.' 
1180                 tmpbuf
[0] = (char) keycode
; 
1182                 bool is_decimal_point 
= 
1183                   ( wxString(tmpbuf
, *wxConvCurrent
) ==  
1184                     wxLocale::GetInfo(wxLOCALE_DECIMAL_POINT
, 
1185                                       wxLOCALE_CAT_NUMBER
) ); 
1186                 if ( (keycode 
< 128) && 
1187                      (wxIsdigit(keycode
) || tolower(keycode
) == 'e' || 
1188                       is_decimal_point 
|| keycode 
== '+' || keycode 
== '-') ) 
1197 #endif // wxUSE_TEXTCTRL 
1201 // ---------------------------------------------------------------------------- 
1202 // wxGridCellBoolEditor 
1203 // ---------------------------------------------------------------------------- 
1205 void wxGridCellBoolEditor::Create(wxWindow
* parent
, 
1207                                   wxEvtHandler
* evtHandler
) 
1209     m_control 
= new wxCheckBox(parent
, id
, wxEmptyString
, 
1210                                wxDefaultPosition
, wxDefaultSize
, 
1213     wxGridCellEditor::Create(parent
, id
, evtHandler
); 
1216 void wxGridCellBoolEditor::SetSize(const wxRect
& r
) 
1218     bool resize 
= FALSE
; 
1219     wxSize size 
= m_control
->GetSize(); 
1220     wxCoord minSize 
= wxMin(r
.width
, r
.height
); 
1222     // check if the checkbox is not too big/small for this cell 
1223     wxSize sizeBest 
= m_control
->GetBestSize(); 
1224     if ( !(size 
== sizeBest
) ) 
1226         // reset to default size if it had been made smaller 
1232     if ( size
.x 
>= minSize 
|| size
.y 
>= minSize 
) 
1234         // leave 1 pixel margin 
1235         size
.x 
= size
.y 
= minSize 
- 2; 
1242         m_control
->SetSize(size
); 
1245     // position it in the centre of the rectangle (TODO: support alignment?) 
1247 #if defined(__WXGTK__) || defined (__WXMOTIF__) 
1248     // the checkbox without label still has some space to the right in wxGTK, 
1249     // so shift it to the right 
1251 #elif defined(__WXMSW__) 
1252     // here too, but in other way 
1257     int hAlign 
= wxALIGN_CENTRE
; 
1258     int vAlign 
= wxALIGN_CENTRE
; 
1260         GetCellAttr()->GetAlignment(& hAlign
, & vAlign
); 
1263     if (hAlign 
== wxALIGN_LEFT
) 
1269         y 
= r
.y 
+ r
.height
/2 - size
.y
/2; 
1271     else if (hAlign 
== wxALIGN_RIGHT
) 
1273         x 
= r
.x 
+ r
.width 
- size
.x 
- 2; 
1274         y 
= r
.y 
+ r
.height
/2 - size
.y
/2; 
1276     else if (hAlign 
== wxALIGN_CENTRE
) 
1278         x 
= r
.x 
+ r
.width
/2 - size
.x
/2; 
1279         y 
= r
.y 
+ r
.height
/2 - size
.y
/2; 
1282     m_control
->Move(x
, y
); 
1285 void wxGridCellBoolEditor::Show(bool show
, wxGridCellAttr 
*attr
) 
1287     m_control
->Show(show
); 
1291         wxColour colBg 
= attr 
? attr
->GetBackgroundColour() : *wxLIGHT_GREY
; 
1292         CBox()->SetBackgroundColour(colBg
); 
1296 void wxGridCellBoolEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
1298     wxASSERT_MSG(m_control
, 
1299                  wxT("The wxGridCellEditor must be Created first!")); 
1301     if (grid
->GetTable()->CanGetValueAs(row
, col
, wxGRID_VALUE_BOOL
)) 
1302         m_startValue 
= grid
->GetTable()->GetValueAsBool(row
, col
); 
1305         wxString 
cellval( grid
->GetTable()->GetValue(row
, col
) ); 
1306         m_startValue 
= !( !cellval 
|| (cellval 
== wxT("0")) ); 
1308     CBox()->SetValue(m_startValue
); 
1312 bool wxGridCellBoolEditor::EndEdit(int row
, int col
, 
1315     wxASSERT_MSG(m_control
, 
1316                  wxT("The wxGridCellEditor must be Created first!")); 
1318     bool changed 
= FALSE
; 
1319     bool value 
= CBox()->GetValue(); 
1320     if ( value 
!= m_startValue 
) 
1325         if (grid
->GetTable()->CanGetValueAs(row
, col
, wxGRID_VALUE_BOOL
)) 
1326             grid
->GetTable()->SetValueAsBool(row
, col
, value
); 
1328             grid
->GetTable()->SetValue(row
, col
, value 
? _T("1") : wxEmptyString
); 
1334 void wxGridCellBoolEditor::Reset() 
1336     wxASSERT_MSG(m_control
, 
1337                  wxT("The wxGridCellEditor must be Created first!")); 
1339     CBox()->SetValue(m_startValue
); 
1342 void wxGridCellBoolEditor::StartingClick() 
1344     CBox()->SetValue(!CBox()->GetValue()); 
1347 bool wxGridCellBoolEditor::IsAcceptedKey(wxKeyEvent
& event
) 
1349     if ( wxGridCellEditor::IsAcceptedKey(event
) ) 
1351         int keycode 
= event
.GetKeyCode(); 
1355             case WXK_NUMPAD_MULTIPLY
: 
1357             case WXK_NUMPAD_ADD
: 
1359             case WXK_NUMPAD_SUBTRACT
: 
1370 // return the value as "1" for true and the empty string for false 
1371 wxString 
wxGridCellBoolEditor::GetValue() const 
1373   bool bSet 
= CBox()->GetValue(); 
1374   return bSet 
? _T("1") : wxEmptyString
; 
1377 #endif // wxUSE_CHECKBOX 
1381 // ---------------------------------------------------------------------------- 
1382 // wxGridCellChoiceEditor 
1383 // ---------------------------------------------------------------------------- 
1385 wxGridCellChoiceEditor::wxGridCellChoiceEditor(const wxArrayString
& choices
, 
1387     : m_choices(choices
), 
1388       m_allowOthers(allowOthers
) { } 
1390 wxGridCellChoiceEditor::wxGridCellChoiceEditor(size_t count
, 
1391                                                const wxString choices
[], 
1393                       : m_allowOthers(allowOthers
) 
1397         m_choices
.Alloc(count
); 
1398         for ( size_t n 
= 0; n 
< count
; n
++ ) 
1400             m_choices
.Add(choices
[n
]); 
1405 wxGridCellEditor 
*wxGridCellChoiceEditor::Clone() const 
1407     wxGridCellChoiceEditor 
*editor 
= new wxGridCellChoiceEditor
; 
1408     editor
->m_allowOthers 
= m_allowOthers
; 
1409     editor
->m_choices 
= m_choices
; 
1414 void wxGridCellChoiceEditor::Create(wxWindow
* parent
, 
1416                                     wxEvtHandler
* evtHandler
) 
1418     m_control 
= new wxComboBox(parent
, id
, wxEmptyString
, 
1419                                wxDefaultPosition
, wxDefaultSize
, 
1421                                m_allowOthers 
? 0 : wxCB_READONLY
); 
1423     wxGridCellEditor::Create(parent
, id
, evtHandler
); 
1426 void wxGridCellChoiceEditor::PaintBackground(const wxRect
& rectCell
, 
1427                                              wxGridCellAttr 
* attr
) 
1429     // as we fill the entire client area, don't do anything here to minimize 
1432     // TODO: It doesn't actually fill the client area since the height of a 
1433     // combo always defaults to the standard...  Until someone has time to 
1434     // figure out the right rectangle to paint, just do it the normal way... 
1435     wxGridCellEditor::PaintBackground(rectCell
, attr
); 
1438 void wxGridCellChoiceEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
1440     wxASSERT_MSG(m_control
, 
1441                  wxT("The wxGridCellEditor must be Created first!")); 
1443     m_startValue 
= grid
->GetTable()->GetValue(row
, col
); 
1446         Combo()->SetValue(m_startValue
); 
1449         // find the right position, or default to the first if not found 
1450         int pos 
= Combo()->FindString(m_startValue
); 
1453         Combo()->SetSelection(pos
); 
1455     Combo()->SetInsertionPointEnd(); 
1456     Combo()->SetFocus(); 
1459 bool wxGridCellChoiceEditor::EndEdit(int row
, int col
, 
1462     wxString value 
= Combo()->GetValue(); 
1463     bool changed 
= value 
!= m_startValue
; 
1466         grid
->GetTable()->SetValue(row
, col
, value
); 
1468     m_startValue 
= wxEmptyString
; 
1470         Combo()->SetValue(m_startValue
); 
1472         Combo()->SetSelection(0); 
1477 void wxGridCellChoiceEditor::Reset() 
1479     Combo()->SetValue(m_startValue
); 
1480     Combo()->SetInsertionPointEnd(); 
1483 void wxGridCellChoiceEditor::SetParameters(const wxString
& params
) 
1493     wxStringTokenizer 
tk(params
, _T(',')); 
1494     while ( tk
.HasMoreTokens() ) 
1496         m_choices
.Add(tk
.GetNextToken()); 
1500 // return the value in the text control 
1501 wxString 
wxGridCellChoiceEditor::GetValue() const 
1503   return Combo()->GetValue(); 
1506 #endif // wxUSE_COMBOBOX 
1508 // ---------------------------------------------------------------------------- 
1509 // wxGridCellEditorEvtHandler 
1510 // ---------------------------------------------------------------------------- 
1512 void wxGridCellEditorEvtHandler::OnKeyDown(wxKeyEvent
& event
) 
1514     switch ( event
.GetKeyCode() ) 
1518             m_grid
->DisableCellEditControl(); 
1522             m_grid
->GetEventHandler()->ProcessEvent( event 
); 
1526         case WXK_NUMPAD_ENTER
: 
1527             if (!m_grid
->GetEventHandler()->ProcessEvent(event
)) 
1528                 m_editor
->HandleReturn(event
); 
1537 void wxGridCellEditorEvtHandler::OnChar(wxKeyEvent
& event
) 
1539     switch ( event
.GetKeyCode() ) 
1544         case WXK_NUMPAD_ENTER
: 
1552 // ---------------------------------------------------------------------------- 
1553 // wxGridCellWorker is an (almost) empty common base class for 
1554 // wxGridCellRenderer and wxGridCellEditor managing ref counting 
1555 // ---------------------------------------------------------------------------- 
1557 void wxGridCellWorker::SetParameters(const wxString
& WXUNUSED(params
)) 
1562 wxGridCellWorker::~wxGridCellWorker() 
1566 // ============================================================================ 
1568 // ============================================================================ 
1570 // ---------------------------------------------------------------------------- 
1571 // wxGridCellRenderer 
1572 // ---------------------------------------------------------------------------- 
1574 void wxGridCellRenderer::Draw(wxGrid
& grid
, 
1575                               wxGridCellAttr
& attr
, 
1578                               int WXUNUSED(row
), int WXUNUSED(col
), 
1581     dc
.SetBackgroundMode( wxSOLID 
); 
1583     // grey out fields if the grid is disabled 
1584     if( grid
.IsEnabled() ) 
1588           dc
.SetBrush( wxBrush(grid
.GetSelectionBackground(), wxSOLID
) ); 
1592           dc
.SetBrush( wxBrush(attr
.GetBackgroundColour(), wxSOLID
) ); 
1597       dc
.SetBrush(wxBrush(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_BTNFACE
), wxSOLID
)); 
1600     dc
.SetPen( *wxTRANSPARENT_PEN 
); 
1601     dc
.DrawRectangle(rect
); 
1604 // ---------------------------------------------------------------------------- 
1605 // wxGridCellStringRenderer 
1606 // ---------------------------------------------------------------------------- 
1608 void wxGridCellStringRenderer::SetTextColoursAndFont(wxGrid
& grid
, 
1609                                                      wxGridCellAttr
& attr
, 
1613     dc
.SetBackgroundMode( wxTRANSPARENT 
); 
1615     // TODO some special colours for attr.IsReadOnly() case? 
1617     // different coloured text when the grid is disabled 
1618     if( grid
.IsEnabled() ) 
1622           dc
.SetTextBackground( grid
.GetSelectionBackground() ); 
1623           dc
.SetTextForeground( grid
.GetSelectionForeground() ); 
1627           dc
.SetTextBackground( attr
.GetBackgroundColour() ); 
1628           dc
.SetTextForeground( attr
.GetTextColour() ); 
1633       dc
.SetTextBackground(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_BTNFACE
)); 
1634       dc
.SetTextForeground(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_GRAYTEXT
)); 
1637     dc
.SetFont( attr
.GetFont() ); 
1640 wxSize 
wxGridCellStringRenderer::DoGetBestSize(wxGridCellAttr
& attr
, 
1642                                                const wxString
& text
) 
1644     wxCoord x 
= 0, y 
= 0, max_x 
= 0; 
1645     dc
.SetFont(attr
.GetFont()); 
1646     wxStringTokenizer 
tk(text
, _T('\n')); 
1647     while ( tk
.HasMoreTokens() ) 
1649         dc
.GetTextExtent(tk
.GetNextToken(), &x
, &y
); 
1650         max_x 
= wxMax(max_x
, x
); 
1653     y 
*= 1 + text
.Freq(wxT('\n')); // multiply by the number of lines. 
1655     return wxSize(max_x
, y
); 
1658 wxSize 
wxGridCellStringRenderer::GetBestSize(wxGrid
& grid
, 
1659                                              wxGridCellAttr
& attr
, 
1663     return DoGetBestSize(attr
, dc
, grid
.GetCellValue(row
, col
)); 
1666 void wxGridCellStringRenderer::Draw(wxGrid
& grid
, 
1667                                     wxGridCellAttr
& attr
, 
1669                                     const wxRect
& rectCell
, 
1673     wxRect rect 
= rectCell
; 
1676     // erase only this cells background, overflow cells should have been erased 
1677     wxGridCellRenderer::Draw(grid
, attr
, dc
, rectCell
, row
, col
, isSelected
); 
1680     attr
.GetAlignment(&hAlign
, &vAlign
); 
1682     int overflowCols 
= 0; 
1684     if (attr
.GetOverflow()) 
1686         int cols 
= grid
.GetNumberCols(); 
1687         int best_width 
= GetBestSize(grid
,attr
,dc
,row
,col
).GetWidth(); 
1688         int cell_rows
, cell_cols
; 
1689         attr
.GetSize( &cell_rows
, &cell_cols 
); // shouldn't get here if <=0 
1690         if ((best_width 
> rectCell
.width
) && (col 
< cols
) && grid
.GetTable()) 
1692             int i
, c_cols
, c_rows
; 
1693             for (i 
= col
+cell_cols
; i 
< cols
; i
++) 
1695                 bool is_empty 
= TRUE
; 
1696                 for (int j
=row
; j
<row
+cell_rows
; j
++) 
1698                     // check w/ anchor cell for multicell block 
1699                     grid
.GetCellSize(j
, i
, &c_rows
, &c_cols
); 
1700                     if (c_rows 
> 0) c_rows 
= 0; 
1701                     if (!grid
.GetTable()->IsEmptyCell(j
+c_rows
, i
)) 
1708                     rect
.width 
+= grid
.GetColSize(i
); 
1714                 if (rect
.width 
>= best_width
) break; 
1716             overflowCols 
= i 
- col 
- cell_cols 
+ 1; 
1717             if (overflowCols 
>= cols
) overflowCols 
= cols 
- 1; 
1720         if (overflowCols 
> 0) // redraw overflow cells w/ proper hilight 
1722             hAlign 
= wxALIGN_LEFT
; // if oveflowed then it's left aligned 
1724             clip
.x 
+= rectCell
.width
; 
1725             // draw each overflow cell individually 
1726             int col_end 
= col
+cell_cols
+overflowCols
; 
1727             if (col_end 
>= grid
.GetNumberCols()) 
1728                 col_end 
= grid
.GetNumberCols() - 1; 
1729             for (int i 
= col
+cell_cols
; i 
<= col_end
; i
++) 
1731                 clip
.width 
= grid
.GetColSize(i
) - 1; 
1732                 dc
.DestroyClippingRegion(); 
1733                 dc
.SetClippingRegion(clip
); 
1735                 SetTextColoursAndFont(grid
, attr
, dc
, 
1736                         grid
.IsInSelection(row
,i
)); 
1738                 grid
.DrawTextRectangle(dc
, grid
.GetCellValue(row
, col
), 
1739                         rect
, hAlign
, vAlign
); 
1740                 clip
.x 
+= grid
.GetColSize(i
) - 1; 
1746             dc
.DestroyClippingRegion(); 
1750     // now we only have to draw the text 
1751     SetTextColoursAndFont(grid
, attr
, dc
, isSelected
); 
1753     grid
.DrawTextRectangle(dc
, grid
.GetCellValue(row
, col
), 
1754                            rect
, hAlign
, vAlign
); 
1757 // ---------------------------------------------------------------------------- 
1758 // wxGridCellNumberRenderer 
1759 // ---------------------------------------------------------------------------- 
1761 wxString 
wxGridCellNumberRenderer::GetString(wxGrid
& grid
, int row
, int col
) 
1763     wxGridTableBase 
*table 
= grid
.GetTable(); 
1765     if ( table
->CanGetValueAs(row
, col
, wxGRID_VALUE_NUMBER
) ) 
1767         text
.Printf(_T("%ld"), table
->GetValueAsLong(row
, col
)); 
1771         text 
= table
->GetValue(row
, col
); 
1777 void wxGridCellNumberRenderer::Draw(wxGrid
& grid
, 
1778                                     wxGridCellAttr
& attr
, 
1780                                     const wxRect
& rectCell
, 
1784     wxGridCellRenderer::Draw(grid
, attr
, dc
, rectCell
, row
, col
, isSelected
); 
1786     SetTextColoursAndFont(grid
, attr
, dc
, isSelected
); 
1788     // draw the text right aligned by default 
1790     attr
.GetAlignment(&hAlign
, &vAlign
); 
1791     hAlign 
= wxALIGN_RIGHT
; 
1793     wxRect rect 
= rectCell
; 
1796     grid
.DrawTextRectangle(dc
, GetString(grid
, row
, col
), rect
, hAlign
, vAlign
); 
1799 wxSize 
wxGridCellNumberRenderer::GetBestSize(wxGrid
& grid
, 
1800                                              wxGridCellAttr
& attr
, 
1804     return DoGetBestSize(attr
, dc
, GetString(grid
, row
, col
)); 
1807 // ---------------------------------------------------------------------------- 
1808 // wxGridCellFloatRenderer 
1809 // ---------------------------------------------------------------------------- 
1811 wxGridCellFloatRenderer::wxGridCellFloatRenderer(int width
, int precision
) 
1814     SetPrecision(precision
); 
1817 wxGridCellRenderer 
*wxGridCellFloatRenderer::Clone() const 
1819     wxGridCellFloatRenderer 
*renderer 
= new wxGridCellFloatRenderer
; 
1820     renderer
->m_width 
= m_width
; 
1821     renderer
->m_precision 
= m_precision
; 
1822     renderer
->m_format 
= m_format
; 
1827 wxString 
wxGridCellFloatRenderer::GetString(wxGrid
& grid
, int row
, int col
) 
1829     wxGridTableBase 
*table 
= grid
.GetTable(); 
1834     if ( table
->CanGetValueAs(row
, col
, wxGRID_VALUE_FLOAT
) ) 
1836         val 
= table
->GetValueAsDouble(row
, col
); 
1841         text 
= table
->GetValue(row
, col
); 
1842         hasDouble 
= text
.ToDouble(&val
); 
1849             if ( m_width 
== -1 ) 
1851                 if ( m_precision 
== -1 ) 
1853                     // default width/precision 
1854                     m_format 
= _T("%f"); 
1858                     m_format
.Printf(_T("%%.%df"), m_precision
); 
1861             else if ( m_precision 
== -1 ) 
1863                 // default precision 
1864                 m_format
.Printf(_T("%%%d.f"), m_width
); 
1868                 m_format
.Printf(_T("%%%d.%df"), m_width
, m_precision
); 
1872         text
.Printf(m_format
, val
); 
1875     //else: text already contains the string 
1880 void wxGridCellFloatRenderer::Draw(wxGrid
& grid
, 
1881                                    wxGridCellAttr
& attr
, 
1883                                    const wxRect
& rectCell
, 
1887     wxGridCellRenderer::Draw(grid
, attr
, dc
, rectCell
, row
, col
, isSelected
); 
1889     SetTextColoursAndFont(grid
, attr
, dc
, isSelected
); 
1891     // draw the text right aligned by default 
1893     attr
.GetAlignment(&hAlign
, &vAlign
); 
1894     hAlign 
= wxALIGN_RIGHT
; 
1896     wxRect rect 
= rectCell
; 
1899     grid
.DrawTextRectangle(dc
, GetString(grid
, row
, col
), rect
, hAlign
, vAlign
); 
1902 wxSize 
wxGridCellFloatRenderer::GetBestSize(wxGrid
& grid
, 
1903                                             wxGridCellAttr
& attr
, 
1907     return DoGetBestSize(attr
, dc
, GetString(grid
, row
, col
)); 
1910 void wxGridCellFloatRenderer::SetParameters(const wxString
& params
) 
1914         // reset to defaults 
1920         wxString tmp 
= params
.BeforeFirst(_T(',')); 
1924             if ( tmp
.ToLong(&width
) ) 
1926                 SetWidth((int)width
); 
1930                 wxLogDebug(_T("Invalid wxGridCellFloatRenderer width parameter string '%s ignored"), params
.c_str()); 
1934                 tmp 
= params
.AfterFirst(_T(',')); 
1938             if ( tmp
.ToLong(&precision
) ) 
1940                 SetPrecision((int)precision
); 
1944                 wxLogDebug(_T("Invalid wxGridCellFloatRenderer precision parameter string '%s ignored"), params
.c_str()); 
1952 // ---------------------------------------------------------------------------- 
1953 // wxGridCellBoolRenderer 
1954 // ---------------------------------------------------------------------------- 
1956 wxSize 
wxGridCellBoolRenderer::ms_sizeCheckMark
; 
1958 // FIXME these checkbox size calculations are really ugly... 
1960 // between checkmark and box 
1961 static const wxCoord wxGRID_CHECKMARK_MARGIN 
= 2; 
1963 wxSize 
wxGridCellBoolRenderer::GetBestSize(wxGrid
& grid
, 
1964                                            wxGridCellAttr
& WXUNUSED(attr
), 
1969     // compute it only once (no locks for MT safeness in GUI thread...) 
1970     if ( !ms_sizeCheckMark
.x 
) 
1972         // get checkbox size 
1973         wxCheckBox 
*checkbox 
= new wxCheckBox(&grid
, -1, wxEmptyString
); 
1974         wxSize size 
= checkbox
->GetBestSize(); 
1975         wxCoord checkSize 
= size
.y 
+ 2*wxGRID_CHECKMARK_MARGIN
; 
1977         // FIXME wxGTK::wxCheckBox::GetBestSize() gives "wrong" result 
1978 #if defined(__WXGTK__) || defined(__WXMOTIF__) 
1979         checkSize 
-= size
.y 
/ 2; 
1984         ms_sizeCheckMark
.x 
= ms_sizeCheckMark
.y 
= checkSize
; 
1987     return ms_sizeCheckMark
; 
1990 void wxGridCellBoolRenderer::Draw(wxGrid
& grid
, 
1991                                   wxGridCellAttr
& attr
, 
1997     wxGridCellRenderer::Draw(grid
, attr
, dc
, rect
, row
, col
, isSelected
); 
1999     // draw a check mark in the centre (ignoring alignment - TODO) 
2000     wxSize size 
= GetBestSize(grid
, attr
, dc
, row
, col
); 
2002     // don't draw outside the cell 
2003     wxCoord minSize 
= wxMin(rect
.width
, rect
.height
); 
2004     if ( size
.x 
>= minSize 
|| size
.y 
>= minSize 
) 
2006         // and even leave (at least) 1 pixel margin 
2007         size
.x 
= size
.y 
= minSize 
- 2; 
2010     // draw a border around checkmark 
2012     attr
.GetAlignment(& hAlign
, &vAlign
); 
2015     if (hAlign 
== wxALIGN_CENTRE
) 
2017         rectBorder
.x 
= rect
.x 
+ rect
.width
/2 - size
.x
/2; 
2018         rectBorder
.y 
= rect
.y 
+ rect
.height
/2 - size
.y
/2; 
2019         rectBorder
.width 
= size
.x
; 
2020         rectBorder
.height 
= size
.y
; 
2022     else if (hAlign 
== wxALIGN_LEFT
) 
2024         rectBorder
.x 
= rect
.x 
+ 2; 
2025         rectBorder
.y 
= rect
.y 
+ rect
.height
/2 - size
.y
/2; 
2026         rectBorder
.width 
= size
.x
; 
2027         rectBorder
.height 
= size
.y
; 
2029     else if (hAlign 
== wxALIGN_RIGHT
) 
2031         rectBorder
.x 
= rect
.x 
+ rect
.width 
- size
.x 
- 2; 
2032         rectBorder
.y 
= rect
.y 
+ rect
.height
/2 - size
.y
/2; 
2033         rectBorder
.width 
= size
.x
; 
2034         rectBorder
.height 
= size
.y
; 
2038     if ( grid
.GetTable()->CanGetValueAs(row
, col
, wxGRID_VALUE_BOOL
) ) 
2039         value 
= grid
.GetTable()->GetValueAsBool(row
, col
); 
2042         wxString 
cellval( grid
.GetTable()->GetValue(row
, col
) ); 
2043         value 
= !( !cellval 
|| (cellval 
== wxT("0")) ); 
2048         wxRect rectMark 
= rectBorder
; 
2050         // MSW DrawCheckMark() is weird (and should probably be changed...) 
2051         rectMark
.Inflate(-wxGRID_CHECKMARK_MARGIN
/2); 
2055         rectMark
.Inflate(-wxGRID_CHECKMARK_MARGIN
); 
2058         dc
.SetTextForeground(attr
.GetTextColour()); 
2059         dc
.DrawCheckMark(rectMark
); 
2062     dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
2063     dc
.SetPen(wxPen(attr
.GetTextColour(), 1, wxSOLID
)); 
2064     dc
.DrawRectangle(rectBorder
); 
2067 // ---------------------------------------------------------------------------- 
2069 // ---------------------------------------------------------------------------- 
2071 void wxGridCellAttr::Init(wxGridCellAttr 
*attrDefault
) 
2075     m_isReadOnly 
= Unset
; 
2080     m_attrkind 
= wxGridCellAttr::Cell
; 
2082     m_sizeRows 
= m_sizeCols 
= 1; 
2083     m_overflow 
= UnsetOverflow
; 
2085     SetDefAttr(attrDefault
); 
2088 wxGridCellAttr 
*wxGridCellAttr::Clone() const 
2090     wxGridCellAttr 
*attr 
= new wxGridCellAttr(m_defGridAttr
); 
2092     if ( HasTextColour() ) 
2093         attr
->SetTextColour(GetTextColour()); 
2094     if ( HasBackgroundColour() ) 
2095         attr
->SetBackgroundColour(GetBackgroundColour()); 
2097         attr
->SetFont(GetFont()); 
2098     if ( HasAlignment() ) 
2099         attr
->SetAlignment(m_hAlign
, m_vAlign
); 
2101     attr
->SetSize( m_sizeRows
, m_sizeCols 
); 
2105         attr
->SetRenderer(m_renderer
); 
2106         m_renderer
->IncRef(); 
2110         attr
->SetEditor(m_editor
); 
2115         attr
->SetReadOnly(); 
2117     attr
->SetKind( m_attrkind 
); 
2122 void wxGridCellAttr::MergeWith(wxGridCellAttr 
*mergefrom
) 
2124     if ( !HasTextColour() && mergefrom
->HasTextColour() ) 
2125         SetTextColour(mergefrom
->GetTextColour()); 
2126     if ( !HasBackgroundColour() && mergefrom
->HasBackgroundColour() ) 
2127         SetBackgroundColour(mergefrom
->GetBackgroundColour()); 
2128     if ( !HasFont() && mergefrom
->HasFont() ) 
2129         SetFont(mergefrom
->GetFont()); 
2130     if ( !HasAlignment() && mergefrom
->HasAlignment() ){ 
2132         mergefrom
->GetAlignment( &hAlign
, &vAlign
); 
2133         SetAlignment(hAlign
, vAlign
); 
2136     mergefrom
->GetSize( &m_sizeRows
, &m_sizeCols 
); 
2138     // Directly access member functions as GetRender/Editor don't just return 
2139     // m_renderer/m_editor 
2141     // Maybe add support for merge of Render and Editor? 
2142     if (!HasRenderer() && mergefrom
->HasRenderer() ) 
2144         m_renderer 
= mergefrom
->m_renderer
; 
2145         m_renderer
->IncRef(); 
2147     if ( !HasEditor() && mergefrom
->HasEditor() ) 
2149         m_editor 
=  mergefrom
->m_editor
; 
2152     if ( !HasReadWriteMode()  && mergefrom
->HasReadWriteMode() ) 
2153         SetReadOnly(mergefrom
->IsReadOnly()); 
2155     if (!HasOverflowMode()  && mergefrom
->HasOverflowMode() ) 
2156         SetOverflow(mergefrom
->GetOverflow()); 
2158     SetDefAttr(mergefrom
->m_defGridAttr
); 
2161 void wxGridCellAttr::SetSize(int num_rows
, int num_cols
) 
2163     // The size of a cell is normally 1,1 
2165     // If this cell is larger (2,2) then this is the top left cell 
2166     // the other cells that will be covered (lower right cells) must be 
2167     // set to negative or zero values such that 
2168     // row + num_rows of the covered cell points to the larger cell (this cell) 
2169     // same goes for the col + num_cols. 
2171     // Size of 0,0 is NOT valid, neither is <=0 and any positive value 
2173     wxASSERT_MSG( (!((num_rows
>0)&&(num_cols
<=0)) || 
2174                   !((num_rows
<=0)&&(num_cols
>0)) || 
2175                   !((num_rows
==0)&&(num_cols
==0))), 
2176                   wxT("wxGridCellAttr::SetSize only takes two postive values or negative/zero values")); 
2178     m_sizeRows 
= num_rows
; 
2179     m_sizeCols 
= num_cols
; 
2182 const wxColour
& wxGridCellAttr::GetTextColour() const 
2184     if (HasTextColour()) 
2188     else if (m_defGridAttr 
&& m_defGridAttr 
!= this) 
2190         return m_defGridAttr
->GetTextColour(); 
2194         wxFAIL_MSG(wxT("Missing default cell attribute")); 
2195         return wxNullColour
; 
2200 const wxColour
& wxGridCellAttr::GetBackgroundColour() const 
2202     if (HasBackgroundColour()) 
2204     else if (m_defGridAttr 
&& m_defGridAttr 
!= this) 
2205         return m_defGridAttr
->GetBackgroundColour(); 
2208         wxFAIL_MSG(wxT("Missing default cell attribute")); 
2209         return wxNullColour
; 
2214 const wxFont
& wxGridCellAttr::GetFont() const 
2218     else if (m_defGridAttr 
&& m_defGridAttr 
!= this) 
2219         return m_defGridAttr
->GetFont(); 
2222         wxFAIL_MSG(wxT("Missing default cell attribute")); 
2228 void wxGridCellAttr::GetAlignment(int *hAlign
, int *vAlign
) const 
2232         if ( hAlign 
) *hAlign 
= m_hAlign
; 
2233         if ( vAlign 
) *vAlign 
= m_vAlign
; 
2235     else if (m_defGridAttr 
&& m_defGridAttr 
!= this) 
2236         m_defGridAttr
->GetAlignment(hAlign
, vAlign
); 
2239         wxFAIL_MSG(wxT("Missing default cell attribute")); 
2243 void wxGridCellAttr::GetSize( int *num_rows
, int *num_cols 
) const 
2245     if ( num_rows 
) *num_rows 
= m_sizeRows
; 
2246     if ( num_cols 
) *num_cols 
= m_sizeCols
; 
2249 // GetRenderer and GetEditor use a slightly different decision path about 
2250 // which attribute to use.  If a non-default attr object has one then it is 
2251 // used, otherwise the default editor or renderer is fetched from the grid and 
2252 // used.  It should be the default for the data type of the cell.  If it is 
2253 // NULL (because the table has a type that the grid does not have in its 
2254 // registry,) then the grid's default editor or renderer is used. 
2256 wxGridCellRenderer
* wxGridCellAttr::GetRenderer(wxGrid
* grid
, int row
, int col
) const 
2258     wxGridCellRenderer 
*renderer
; 
2260     if ( m_renderer 
&& this != m_defGridAttr 
) 
2262         // use the cells renderer if it has one 
2263         renderer 
= m_renderer
; 
2266     else // no non default cell renderer 
2268         // get default renderer for the data type 
2271             // GetDefaultRendererForCell() will do IncRef() for us 
2272             renderer 
= grid
->GetDefaultRendererForCell(row
, col
); 
2281             if (m_defGridAttr 
&&  this != m_defGridAttr 
) 
2283                 // if we still don't have one then use the grid default 
2284                 // (no need for IncRef() here neither) 
2285                 renderer 
= m_defGridAttr
->GetRenderer(NULL
, 0, 0); 
2287             else // default grid attr 
2289                 // use m_renderer which we had decided not to use initially 
2290                 renderer 
= m_renderer
; 
2297     // we're supposed to always find something 
2298     wxASSERT_MSG(renderer
, wxT("Missing default cell renderer")); 
2303 // same as above, except for s/renderer/editor/g 
2304 wxGridCellEditor
* wxGridCellAttr::GetEditor(wxGrid
* grid
, int row
, int col
) const 
2306     wxGridCellEditor 
*editor
; 
2308     if ( m_editor 
&& this != m_defGridAttr 
) 
2310         // use the cells editor if it has one 
2314     else // no non default cell editor 
2316         // get default editor for the data type 
2319             // GetDefaultEditorForCell() will do IncRef() for us 
2320             editor 
= grid
->GetDefaultEditorForCell(row
, col
); 
2329             if ( m_defGridAttr 
&& this != m_defGridAttr 
) 
2331                 // if we still don't have one then use the grid default 
2332                 // (no need for IncRef() here neither) 
2333                 editor 
= m_defGridAttr
->GetEditor(NULL
, 0, 0); 
2335             else // default grid attr 
2337                 // use m_editor which we had decided not to use initially 
2345     // we're supposed to always find something 
2346     wxASSERT_MSG(editor
, wxT("Missing default cell editor")); 
2351 // ---------------------------------------------------------------------------- 
2352 // wxGridCellAttrData 
2353 // ---------------------------------------------------------------------------- 
2355 void wxGridCellAttrData::SetAttr(wxGridCellAttr 
*attr
, int row
, int col
) 
2357     int n 
= FindIndex(row
, col
); 
2358     if ( n 
== wxNOT_FOUND 
) 
2360         // add the attribute 
2361         m_attrs
.Add(new wxGridCellWithAttr(row
, col
, attr
)); 
2365         // free the old attribute 
2366         m_attrs
[(size_t)n
].attr
->DecRef(); 
2370             // change the attribute 
2371             m_attrs
[(size_t)n
].attr 
= attr
; 
2375             // remove this attribute 
2376             m_attrs
.RemoveAt((size_t)n
); 
2381 wxGridCellAttr 
*wxGridCellAttrData::GetAttr(int row
, int col
) const 
2383     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
2385     int n 
= FindIndex(row
, col
); 
2386     if ( n 
!= wxNOT_FOUND 
) 
2388         attr 
= m_attrs
[(size_t)n
].attr
; 
2395 void wxGridCellAttrData::UpdateAttrRows( size_t pos
, int numRows 
) 
2397     size_t count 
= m_attrs
.GetCount(); 
2398     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2400         wxGridCellCoords
& coords 
= m_attrs
[n
].coords
; 
2401         wxCoord row 
= coords
.GetRow(); 
2402         if ((size_t)row 
>= pos
) 
2406                 // If rows inserted, include row counter where necessary 
2407                 coords
.SetRow(row 
+ numRows
); 
2409             else if (numRows 
< 0) 
2411                 // If rows deleted ... 
2412                 if ((size_t)row 
>= pos 
- numRows
) 
2414                     // ...either decrement row counter (if row still exists)... 
2415                     coords
.SetRow(row 
+ numRows
); 
2419                     // ...or remove the attribute 
2420                     m_attrs
.RemoveAt((size_t)n
); 
2428 void wxGridCellAttrData::UpdateAttrCols( size_t pos
, int numCols 
) 
2430     size_t count 
= m_attrs
.GetCount(); 
2431     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2433         wxGridCellCoords
& coords 
= m_attrs
[n
].coords
; 
2434         wxCoord col 
= coords
.GetCol(); 
2435         if ( (size_t)col 
>= pos 
) 
2439                 // If rows inserted, include row counter where necessary 
2440                 coords
.SetCol(col 
+ numCols
); 
2442             else if (numCols 
< 0) 
2444                 // If rows deleted ... 
2445                 if ((size_t)col 
>= pos 
- numCols
) 
2447                     // ...either decrement row counter (if row still exists)... 
2448                     coords
.SetCol(col 
+ numCols
); 
2452                     // ...or remove the attribute 
2453                     m_attrs
.RemoveAt((size_t)n
); 
2461 int wxGridCellAttrData::FindIndex(int row
, int col
) const 
2463     size_t count 
= m_attrs
.GetCount(); 
2464     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2466         const wxGridCellCoords
& coords 
= m_attrs
[n
].coords
; 
2467         if ( (coords
.GetRow() == row
) && (coords
.GetCol() == col
) ) 
2476 // ---------------------------------------------------------------------------- 
2477 // wxGridRowOrColAttrData 
2478 // ---------------------------------------------------------------------------- 
2480 wxGridRowOrColAttrData::~wxGridRowOrColAttrData() 
2482     size_t count 
= m_attrs
.Count(); 
2483     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2485         m_attrs
[n
]->DecRef(); 
2489 wxGridCellAttr 
*wxGridRowOrColAttrData::GetAttr(int rowOrCol
) const 
2491     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
2493     int n 
= m_rowsOrCols
.Index(rowOrCol
); 
2494     if ( n 
!= wxNOT_FOUND 
) 
2496         attr 
= m_attrs
[(size_t)n
]; 
2503 void wxGridRowOrColAttrData::SetAttr(wxGridCellAttr 
*attr
, int rowOrCol
) 
2505     int i 
= m_rowsOrCols
.Index(rowOrCol
); 
2506     if ( i 
== wxNOT_FOUND 
) 
2508         // add the attribute 
2509         m_rowsOrCols
.Add(rowOrCol
); 
2514         size_t n 
= (size_t)i
; 
2517             // change the attribute 
2518             m_attrs
[n
]->DecRef(); 
2523             // remove this attribute 
2524             m_attrs
[n
]->DecRef(); 
2525             m_rowsOrCols
.RemoveAt(n
); 
2526             m_attrs
.RemoveAt(n
); 
2531 void wxGridRowOrColAttrData::UpdateAttrRowsOrCols( size_t pos
, int numRowsOrCols 
) 
2533     size_t count 
= m_attrs
.GetCount(); 
2534     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2536         int & rowOrCol 
= m_rowsOrCols
[n
]; 
2537         if ( (size_t)rowOrCol 
>= pos 
) 
2539             if ( numRowsOrCols 
> 0 ) 
2541                 // If rows inserted, include row counter where necessary 
2542                 rowOrCol 
+= numRowsOrCols
; 
2544             else if ( numRowsOrCols 
< 0) 
2546                 // If rows deleted, either decrement row counter (if row still exists) 
2547                 if ((size_t)rowOrCol 
>= pos 
- numRowsOrCols
) 
2548                     rowOrCol 
+= numRowsOrCols
; 
2551                     m_rowsOrCols
.RemoveAt((size_t)n
); 
2552                     m_attrs
.RemoveAt((size_t)n
); 
2560 // ---------------------------------------------------------------------------- 
2561 // wxGridCellAttrProvider 
2562 // ---------------------------------------------------------------------------- 
2564 wxGridCellAttrProvider::wxGridCellAttrProvider() 
2566     m_data 
= (wxGridCellAttrProviderData 
*)NULL
; 
2569 wxGridCellAttrProvider::~wxGridCellAttrProvider() 
2574 void wxGridCellAttrProvider::InitData() 
2576     m_data 
= new wxGridCellAttrProviderData
; 
2579 wxGridCellAttr 
*wxGridCellAttrProvider::GetAttr(int row
, int col
, 
2580                                                 wxGridCellAttr::wxAttrKind  kind 
) const 
2582     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
2587             case (wxGridCellAttr::Any
): 
2588                 //Get cached merge attributes. 
2589                 // Currenlty not used as no cache implemented as not mutiable 
2590                 // attr = m_data->m_mergeAttr.GetAttr(row, col); 
2593                     //Basicaly implement old version. 
2594                     //Also check merge cache, so we don't have to re-merge every time.. 
2595                     wxGridCellAttr 
*attrcell 
= m_data
->m_cellAttrs
.GetAttr(row
, col
); 
2596                     wxGridCellAttr 
*attrrow 
= m_data
->m_rowAttrs
.GetAttr(row
); 
2597                     wxGridCellAttr 
*attrcol 
= m_data
->m_colAttrs
.GetAttr(col
); 
2599                     if((attrcell 
!= attrrow
) && (attrrow 
!=attrcol
) && (attrcell 
!= attrcol
)){ 
2600                         // Two or move are non NULL 
2601                         attr 
= new wxGridCellAttr
; 
2602                         attr
->SetKind(wxGridCellAttr::Merged
); 
2606                             attr
->MergeWith(attrcell
); 
2610                             attr
->MergeWith(attrcol
); 
2614                             attr
->MergeWith(attrrow
); 
2617                         //store merge attr if cache implemented 
2619                         //m_data->m_mergeAttr.SetAttr(attr, row, col); 
2623                         // one or none is non null return it or null. 
2624                         if(attrrow
) attr 
= attrrow
; 
2625                         if(attrcol
) attr 
= attrcol
; 
2626                         if(attrcell
) attr 
= attrcell
; 
2630             case (wxGridCellAttr::Cell
): 
2631                 attr 
= m_data
->m_cellAttrs
.GetAttr(row
, col
); 
2633             case (wxGridCellAttr::Col
): 
2634                  attr 
= m_data
->m_colAttrs
.GetAttr(col
); 
2636             case (wxGridCellAttr::Row
): 
2637             attr 
= m_data
->m_rowAttrs
.GetAttr(row
); 
2641                 // (wxGridCellAttr::Default): 
2642                 // (wxGridCellAttr::Merged): 
2649 void wxGridCellAttrProvider::SetAttr(wxGridCellAttr 
*attr
, 
2655     m_data
->m_cellAttrs
.SetAttr(attr
, row
, col
); 
2658 void wxGridCellAttrProvider::SetRowAttr(wxGridCellAttr 
*attr
, int row
) 
2663     m_data
->m_rowAttrs
.SetAttr(attr
, row
); 
2666 void wxGridCellAttrProvider::SetColAttr(wxGridCellAttr 
*attr
, int col
) 
2671     m_data
->m_colAttrs
.SetAttr(attr
, col
); 
2674 void wxGridCellAttrProvider::UpdateAttrRows( size_t pos
, int numRows 
) 
2678         m_data
->m_cellAttrs
.UpdateAttrRows( pos
, numRows 
); 
2680         m_data
->m_rowAttrs
.UpdateAttrRowsOrCols( pos
, numRows 
); 
2684 void wxGridCellAttrProvider::UpdateAttrCols( size_t pos
, int numCols 
) 
2688         m_data
->m_cellAttrs
.UpdateAttrCols( pos
, numCols 
); 
2690         m_data
->m_colAttrs
.UpdateAttrRowsOrCols( pos
, numCols 
); 
2694 // ---------------------------------------------------------------------------- 
2695 // wxGridTypeRegistry 
2696 // ---------------------------------------------------------------------------- 
2698 wxGridTypeRegistry::~wxGridTypeRegistry() 
2700     size_t count 
= m_typeinfo
.Count(); 
2701     for ( size_t i 
= 0; i 
< count
; i
++ ) 
2702         delete m_typeinfo
[i
]; 
2706 void wxGridTypeRegistry::RegisterDataType(const wxString
& typeName
, 
2707                                           wxGridCellRenderer
* renderer
, 
2708                                           wxGridCellEditor
* editor
) 
2710     wxGridDataTypeInfo
* info 
= new wxGridDataTypeInfo(typeName
, renderer
, editor
); 
2712     // is it already registered? 
2713     int loc 
= FindRegisteredDataType(typeName
); 
2714     if ( loc 
!= wxNOT_FOUND 
) 
2716         delete m_typeinfo
[loc
]; 
2717         m_typeinfo
[loc
] = info
; 
2721         m_typeinfo
.Add(info
); 
2725 int wxGridTypeRegistry::FindRegisteredDataType(const wxString
& typeName
) 
2727     size_t count 
= m_typeinfo
.GetCount(); 
2728     for ( size_t i 
= 0; i 
< count
; i
++ ) 
2730         if ( typeName 
== m_typeinfo
[i
]->m_typeName 
) 
2739 int wxGridTypeRegistry::FindDataType(const wxString
& typeName
) 
2741     int index 
= FindRegisteredDataType(typeName
); 
2742     if ( index 
== wxNOT_FOUND 
) 
2744         // check whether this is one of the standard ones, in which case 
2745         // register it "on the fly" 
2747         if ( typeName 
== wxGRID_VALUE_STRING 
) 
2749             RegisterDataType(wxGRID_VALUE_STRING
, 
2750                              new wxGridCellStringRenderer
, 
2751                              new wxGridCellTextEditor
); 
2753 #endif // wxUSE_TEXTCTRL 
2755         if ( typeName 
== wxGRID_VALUE_BOOL 
) 
2757             RegisterDataType(wxGRID_VALUE_BOOL
, 
2758                              new wxGridCellBoolRenderer
, 
2759                              new wxGridCellBoolEditor
); 
2761 #endif // wxUSE_CHECKBOX 
2763         if ( typeName 
== wxGRID_VALUE_NUMBER 
) 
2765             RegisterDataType(wxGRID_VALUE_NUMBER
, 
2766                              new wxGridCellNumberRenderer
, 
2767                              new wxGridCellNumberEditor
); 
2769         else if ( typeName 
== wxGRID_VALUE_FLOAT 
) 
2771             RegisterDataType(wxGRID_VALUE_FLOAT
, 
2772                              new wxGridCellFloatRenderer
, 
2773                              new wxGridCellFloatEditor
); 
2775 #endif // wxUSE_TEXTCTRL 
2777         if ( typeName 
== wxGRID_VALUE_CHOICE 
) 
2779             RegisterDataType(wxGRID_VALUE_CHOICE
, 
2780                              new wxGridCellStringRenderer
, 
2781                              new wxGridCellChoiceEditor
); 
2783 #endif // wxUSE_COMBOBOX 
2788         // we get here only if just added the entry for this type, so return 
2790         index 
= m_typeinfo
.GetCount() - 1; 
2796 int wxGridTypeRegistry::FindOrCloneDataType(const wxString
& typeName
) 
2798     int index 
= FindDataType(typeName
); 
2799     if ( index 
== wxNOT_FOUND 
) 
2801         // the first part of the typename is the "real" type, anything after ':' 
2802         // are the parameters for the renderer 
2803         index 
= FindDataType(typeName
.BeforeFirst(_T(':'))); 
2804         if ( index 
== wxNOT_FOUND 
) 
2809         wxGridCellRenderer 
*renderer 
= GetRenderer(index
); 
2810         wxGridCellRenderer 
*rendererOld 
= renderer
; 
2811         renderer 
= renderer
->Clone(); 
2812         rendererOld
->DecRef(); 
2814         wxGridCellEditor 
*editor 
= GetEditor(index
); 
2815         wxGridCellEditor 
*editorOld 
= editor
; 
2816         editor 
= editor
->Clone(); 
2817         editorOld
->DecRef(); 
2819         // do it even if there are no parameters to reset them to defaults 
2820         wxString params 
= typeName
.AfterFirst(_T(':')); 
2821         renderer
->SetParameters(params
); 
2822         editor
->SetParameters(params
); 
2824         // register the new typename 
2825         RegisterDataType(typeName
, renderer
, editor
); 
2827         // we just registered it, it's the last one 
2828         index 
= m_typeinfo
.GetCount() - 1; 
2834 wxGridCellRenderer
* wxGridTypeRegistry::GetRenderer(int index
) 
2836     wxGridCellRenderer
* renderer 
= m_typeinfo
[index
]->m_renderer
; 
2842 wxGridCellEditor
* wxGridTypeRegistry::GetEditor(int index
) 
2844     wxGridCellEditor
* editor 
= m_typeinfo
[index
]->m_editor
; 
2850 // ---------------------------------------------------------------------------- 
2852 // ---------------------------------------------------------------------------- 
2854 IMPLEMENT_ABSTRACT_CLASS( wxGridTableBase
, wxObject 
) 
2857 wxGridTableBase::wxGridTableBase() 
2859     m_view 
= (wxGrid 
*) NULL
; 
2860     m_attrProvider 
= (wxGridCellAttrProvider 
*) NULL
; 
2863 wxGridTableBase::~wxGridTableBase() 
2865     delete m_attrProvider
; 
2868 void wxGridTableBase::SetAttrProvider(wxGridCellAttrProvider 
*attrProvider
) 
2870     delete m_attrProvider
; 
2871     m_attrProvider 
= attrProvider
; 
2874 bool wxGridTableBase::CanHaveAttributes() 
2876     if ( ! GetAttrProvider() ) 
2878         // use the default attr provider by default 
2879         SetAttrProvider(new wxGridCellAttrProvider
); 
2884 wxGridCellAttr 
*wxGridTableBase::GetAttr(int row
, int col
, wxGridCellAttr::wxAttrKind  kind
) 
2886     if ( m_attrProvider 
) 
2887         return m_attrProvider
->GetAttr(row
, col
, kind
); 
2889         return (wxGridCellAttr 
*)NULL
; 
2892 void wxGridTableBase::SetAttr(wxGridCellAttr
* attr
, int row
, int col
) 
2894     if ( m_attrProvider 
) 
2896         attr
->SetKind(wxGridCellAttr::Cell
); 
2897         m_attrProvider
->SetAttr(attr
, row
, col
); 
2901         // as we take ownership of the pointer and don't store it, we must 
2907 void wxGridTableBase::SetRowAttr(wxGridCellAttr 
*attr
, int row
) 
2909     if ( m_attrProvider 
) 
2911         attr
->SetKind(wxGridCellAttr::Row
); 
2912         m_attrProvider
->SetRowAttr(attr
, row
); 
2916         // as we take ownership of the pointer and don't store it, we must 
2922 void wxGridTableBase::SetColAttr(wxGridCellAttr 
*attr
, int col
) 
2924     if ( m_attrProvider 
) 
2926         attr
->SetKind(wxGridCellAttr::Col
); 
2927         m_attrProvider
->SetColAttr(attr
, col
); 
2931         // as we take ownership of the pointer and don't store it, we must 
2937 bool wxGridTableBase::InsertRows( size_t WXUNUSED(pos
), 
2938                                   size_t WXUNUSED(numRows
) ) 
2940     wxFAIL_MSG( wxT("Called grid table class function InsertRows\nbut your derived table class does not override this function") ); 
2945 bool wxGridTableBase::AppendRows( size_t WXUNUSED(numRows
) ) 
2947     wxFAIL_MSG( wxT("Called grid table class function AppendRows\nbut your derived table class does not override this function")); 
2952 bool wxGridTableBase::DeleteRows( size_t WXUNUSED(pos
), 
2953                                   size_t WXUNUSED(numRows
) ) 
2955     wxFAIL_MSG( wxT("Called grid table class function DeleteRows\nbut your derived table class does not override this function")); 
2960 bool wxGridTableBase::InsertCols( size_t WXUNUSED(pos
), 
2961                                   size_t WXUNUSED(numCols
) ) 
2963     wxFAIL_MSG( wxT("Called grid table class function InsertCols\nbut your derived table class does not override this function")); 
2968 bool wxGridTableBase::AppendCols( size_t WXUNUSED(numCols
) ) 
2970     wxFAIL_MSG(wxT("Called grid table class function AppendCols\nbut your derived table class does not override this function")); 
2975 bool wxGridTableBase::DeleteCols( size_t WXUNUSED(pos
), 
2976                                   size_t WXUNUSED(numCols
) ) 
2978     wxFAIL_MSG( wxT("Called grid table class function DeleteCols\nbut your derived table class does not override this function")); 
2984 wxString 
wxGridTableBase::GetRowLabelValue( int row 
) 
2987     s 
<< row 
+ 1; // RD: Starting the rows at zero confuses users, no matter 
2988                   //     how much it makes sense to us geeks. 
2992 wxString 
wxGridTableBase::GetColLabelValue( int col 
) 
2994     // default col labels are: 
2995     //   cols 0 to 25   : A-Z 
2996     //   cols 26 to 675 : AA-ZZ 
3001     for ( n 
= 1; ; n
++ ) 
3003         s 
+= (wxChar
) (_T('A') + (wxChar
)( col%26 
)); 
3005         if ( col 
< 0 ) break; 
3008     // reverse the string... 
3010     for ( i 
= 0;  i 
< n
;  i
++ ) 
3019 wxString 
wxGridTableBase::GetTypeName( int WXUNUSED(row
), int WXUNUSED(col
) ) 
3021     return wxGRID_VALUE_STRING
; 
3024 bool wxGridTableBase::CanGetValueAs( int WXUNUSED(row
), int WXUNUSED(col
), 
3025                                      const wxString
& typeName 
) 
3027     return typeName 
== wxGRID_VALUE_STRING
; 
3030 bool wxGridTableBase::CanSetValueAs( int row
, int col
, const wxString
& typeName 
) 
3032     return CanGetValueAs(row
, col
, typeName
); 
3035 long wxGridTableBase::GetValueAsLong( int WXUNUSED(row
), int WXUNUSED(col
) ) 
3040 double wxGridTableBase::GetValueAsDouble( int WXUNUSED(row
), int WXUNUSED(col
) ) 
3045 bool wxGridTableBase::GetValueAsBool( int WXUNUSED(row
), int WXUNUSED(col
) ) 
3050 void wxGridTableBase::SetValueAsLong( int WXUNUSED(row
), int WXUNUSED(col
), 
3051                                       long WXUNUSED(value
) ) 
3055 void wxGridTableBase::SetValueAsDouble( int WXUNUSED(row
), int WXUNUSED(col
), 
3056                                         double WXUNUSED(value
) ) 
3060 void wxGridTableBase::SetValueAsBool( int WXUNUSED(row
), int WXUNUSED(col
), 
3061                                       bool WXUNUSED(value
) ) 
3066 void* wxGridTableBase::GetValueAsCustom( int WXUNUSED(row
), int WXUNUSED(col
), 
3067                                          const wxString
& WXUNUSED(typeName
) ) 
3072 void  wxGridTableBase::SetValueAsCustom( int WXUNUSED(row
), int WXUNUSED(col
), 
3073                                          const wxString
& WXUNUSED(typeName
), 
3074                                          void* WXUNUSED(value
) ) 
3078 ////////////////////////////////////////////////////////////////////// 
3080 // Message class for the grid table to send requests and notifications 
3084 wxGridTableMessage::wxGridTableMessage() 
3086     m_table 
= (wxGridTableBase 
*) NULL
; 
3092 wxGridTableMessage::wxGridTableMessage( wxGridTableBase 
*table
, int id
, 
3093                                         int commandInt1
, int commandInt2 
) 
3097     m_comInt1 
= commandInt1
; 
3098     m_comInt2 
= commandInt2
; 
3103 ////////////////////////////////////////////////////////////////////// 
3105 // A basic grid table for string data. An object of this class will 
3106 // created by wxGrid if you don't specify an alternative table class. 
3109 WX_DEFINE_OBJARRAY(wxGridStringArray
) 
3111 IMPLEMENT_DYNAMIC_CLASS( wxGridStringTable
, wxGridTableBase 
) 
3113 wxGridStringTable::wxGridStringTable() 
3118 wxGridStringTable::wxGridStringTable( int numRows
, int numCols 
) 
3121     m_data
.Alloc( numRows 
); 
3124     sa
.Alloc( numCols 
); 
3125     sa
.Add( wxEmptyString
, numCols 
); 
3127     m_data
.Add( sa
, numRows 
); 
3130 wxGridStringTable::~wxGridStringTable() 
3134 int wxGridStringTable::GetNumberRows() 
3136     return m_data
.GetCount(); 
3139 int wxGridStringTable::GetNumberCols() 
3141     if ( m_data
.GetCount() > 0 ) 
3142         return m_data
[0].GetCount(); 
3147 wxString 
wxGridStringTable::GetValue( int row
, int col 
) 
3149     wxCHECK_MSG( (row 
< GetNumberRows()) && (col 
< GetNumberCols()), 
3151                  _T("invalid row or column index in wxGridStringTable") ); 
3153     return m_data
[row
][col
]; 
3156 void wxGridStringTable::SetValue( int row
, int col
, const wxString
& value 
) 
3158     wxCHECK_RET( (row 
< GetNumberRows()) && (col 
< GetNumberCols()), 
3159                  _T("invalid row or column index in wxGridStringTable") ); 
3161     m_data
[row
][col
] = value
; 
3164 bool wxGridStringTable::IsEmptyCell( int row
, int col 
) 
3166     wxCHECK_MSG( (row 
< GetNumberRows()) && (col 
< GetNumberCols()), 
3168                   _T("invalid row or column index in wxGridStringTable") ); 
3170     return (m_data
[row
][col
] == wxEmptyString
); 
3173 void wxGridStringTable::Clear() 
3176     int numRows
, numCols
; 
3178     numRows 
= m_data
.GetCount(); 
3181         numCols 
= m_data
[0].GetCount(); 
3183         for ( row 
= 0;  row 
< numRows
;  row
++ ) 
3185             for ( col 
= 0;  col 
< numCols
;  col
++ ) 
3187                 m_data
[row
][col
] = wxEmptyString
; 
3194 bool wxGridStringTable::InsertRows( size_t pos
, size_t numRows 
) 
3196     size_t curNumRows 
= m_data
.GetCount(); 
3197     size_t curNumCols 
= ( curNumRows 
> 0 ? m_data
[0].GetCount() : 
3198                           ( GetView() ? GetView()->GetNumberCols() : 0 ) ); 
3200     if ( pos 
>= curNumRows 
) 
3202         return AppendRows( numRows 
); 
3206     sa
.Alloc( curNumCols 
); 
3207     sa
.Add( wxEmptyString
, curNumCols 
); 
3208     m_data
.Insert( sa
, pos
, numRows 
); 
3211         wxGridTableMessage 
msg( this, 
3212                                 wxGRIDTABLE_NOTIFY_ROWS_INSERTED
, 
3216         GetView()->ProcessTableMessage( msg 
); 
3222 bool wxGridStringTable::AppendRows( size_t numRows 
) 
3224     size_t curNumRows 
= m_data
.GetCount(); 
3225     size_t curNumCols 
= ( curNumRows 
> 0 ? m_data
[0].GetCount() : 
3226                           ( GetView() ? GetView()->GetNumberCols() : 0 ) ); 
3229     if ( curNumCols 
> 0 ) 
3231         sa
.Alloc( curNumCols 
); 
3232         sa
.Add( wxEmptyString
, curNumCols 
); 
3235     m_data
.Add( sa
, numRows 
); 
3239         wxGridTableMessage 
msg( this, 
3240                                 wxGRIDTABLE_NOTIFY_ROWS_APPENDED
, 
3243         GetView()->ProcessTableMessage( msg 
); 
3249 bool wxGridStringTable::DeleteRows( size_t pos
, size_t numRows 
) 
3251     size_t curNumRows 
= m_data
.GetCount(); 
3253     if ( pos 
>= curNumRows 
) 
3255         wxFAIL_MSG( wxString::Format
 
3257                         wxT("Called wxGridStringTable::DeleteRows(pos=%lu, N=%lu)\nPos value is invalid for present table with %lu rows"), 
3259                         (unsigned long)numRows
, 
3260                         (unsigned long)curNumRows
 
3266     if ( numRows 
> curNumRows 
- pos 
) 
3268         numRows 
= curNumRows 
- pos
; 
3271     if ( numRows 
>= curNumRows 
) 
3277         m_data
.RemoveAt( pos
, numRows 
); 
3281         wxGridTableMessage 
msg( this, 
3282                                 wxGRIDTABLE_NOTIFY_ROWS_DELETED
, 
3286         GetView()->ProcessTableMessage( msg 
); 
3292 bool wxGridStringTable::InsertCols( size_t pos
, size_t numCols 
) 
3296     size_t curNumRows 
= m_data
.GetCount(); 
3297     size_t curNumCols 
= ( curNumRows 
> 0 ? m_data
[0].GetCount() : 
3298                           ( GetView() ? GetView()->GetNumberCols() : 0 ) ); 
3300     if ( pos 
>= curNumCols 
) 
3302         return AppendCols( numCols 
); 
3305     for ( row 
= 0;  row 
< curNumRows
;  row
++ ) 
3307         for ( col 
= pos
;  col 
< pos 
+ numCols
;  col
++ ) 
3309             m_data
[row
].Insert( wxEmptyString
, col 
); 
3314         wxGridTableMessage 
msg( this, 
3315                                 wxGRIDTABLE_NOTIFY_COLS_INSERTED
, 
3319         GetView()->ProcessTableMessage( msg 
); 
3325 bool wxGridStringTable::AppendCols( size_t numCols 
) 
3329     size_t curNumRows 
= m_data
.GetCount(); 
3333         // TODO: something better than this ? 
3335         wxFAIL_MSG( wxT("Unable to append cols to a grid table with no rows.\nCall AppendRows() first") ); 
3340     for ( row 
= 0;  row 
< curNumRows
;  row
++ ) 
3342         m_data
[row
].Add( wxEmptyString
, numCols 
); 
3347         wxGridTableMessage 
msg( this, 
3348                                 wxGRIDTABLE_NOTIFY_COLS_APPENDED
, 
3351         GetView()->ProcessTableMessage( msg 
); 
3357 bool wxGridStringTable::DeleteCols( size_t pos
, size_t numCols 
) 
3361     size_t curNumRows 
= m_data
.GetCount(); 
3362     size_t curNumCols 
= ( curNumRows 
> 0 ? m_data
[0].GetCount() : 
3363                           ( GetView() ? GetView()->GetNumberCols() : 0 ) ); 
3365     if ( pos 
>= curNumCols 
) 
3367         wxFAIL_MSG( wxString::Format
 
3369                         wxT("Called wxGridStringTable::DeleteCols(pos=%lu, N=%lu)\nPos value is invalid for present table with %lu cols"), 
3371                         (unsigned long)numCols
, 
3372                         (unsigned long)curNumCols
 
3377     if ( numCols 
> curNumCols 
- pos 
) 
3379         numCols 
= curNumCols 
- pos
; 
3382     for ( row 
= 0;  row 
< curNumRows
;  row
++ ) 
3384         if ( numCols 
>= curNumCols 
) 
3386             m_data
[row
].Clear(); 
3390             m_data
[row
].RemoveAt( pos
, numCols 
); 
3395         wxGridTableMessage 
msg( this, 
3396                                 wxGRIDTABLE_NOTIFY_COLS_DELETED
, 
3400         GetView()->ProcessTableMessage( msg 
); 
3406 wxString 
wxGridStringTable::GetRowLabelValue( int row 
) 
3408     if ( row 
> (int)(m_rowLabels
.GetCount()) - 1 ) 
3410         // using default label 
3412         return wxGridTableBase::GetRowLabelValue( row 
); 
3416         return m_rowLabels
[ row 
]; 
3420 wxString 
wxGridStringTable::GetColLabelValue( int col 
) 
3422     if ( col 
> (int)(m_colLabels
.GetCount()) - 1 ) 
3424         // using default label 
3426         return wxGridTableBase::GetColLabelValue( col 
); 
3430         return m_colLabels
[ col 
]; 
3434 void wxGridStringTable::SetRowLabelValue( int row
, const wxString
& value 
) 
3436     if ( row 
> (int)(m_rowLabels
.GetCount()) - 1 ) 
3438         int n 
= m_rowLabels
.GetCount(); 
3440         for ( i 
= n
;  i 
<= row
;  i
++ ) 
3442             m_rowLabels
.Add( wxGridTableBase::GetRowLabelValue(i
) ); 
3446     m_rowLabels
[row
] = value
; 
3449 void wxGridStringTable::SetColLabelValue( int col
, const wxString
& value 
) 
3451     if ( col 
> (int)(m_colLabels
.GetCount()) - 1 ) 
3453         int n 
= m_colLabels
.GetCount(); 
3455         for ( i 
= n
;  i 
<= col
;  i
++ ) 
3457             m_colLabels
.Add( wxGridTableBase::GetColLabelValue(i
) ); 
3461     m_colLabels
[col
] = value
; 
3466 ////////////////////////////////////////////////////////////////////// 
3467 ////////////////////////////////////////////////////////////////////// 
3469 IMPLEMENT_DYNAMIC_CLASS( wxGridRowLabelWindow
, wxWindow 
) 
3471 BEGIN_EVENT_TABLE( wxGridRowLabelWindow
, wxWindow 
) 
3472     EVT_PAINT( wxGridRowLabelWindow::OnPaint 
) 
3473     EVT_MOUSEWHEEL( wxGridRowLabelWindow::OnMouseWheel
) 
3474     EVT_MOUSE_EVENTS( wxGridRowLabelWindow::OnMouseEvent 
) 
3475     EVT_KEY_DOWN( wxGridRowLabelWindow::OnKeyDown 
) 
3476     EVT_KEY_UP( wxGridRowLabelWindow::OnKeyUp 
) 
3479 wxGridRowLabelWindow::wxGridRowLabelWindow( wxGrid 
*parent
, 
3481                                             const wxPoint 
&pos
, const wxSize 
&size 
) 
3482   : wxWindow( parent
, id
, pos
, size
, wxWANTS_CHARS
|wxBORDER_NONE
|wxFULL_REPAINT_ON_RESIZE 
) 
3487 void wxGridRowLabelWindow::OnPaint( wxPaintEvent
& WXUNUSED(event
) ) 
3491     // NO - don't do this because it will set both the x and y origin 
3492     // coords to match the parent scrolled window and we just want to 
3493     // set the y coord  - MB 
3495     // m_owner->PrepareDC( dc ); 
3498     m_owner
->CalcUnscrolledPosition( 0, 0, &x
, &y 
); 
3499     dc
.SetDeviceOrigin( 0, -y 
); 
3501     wxArrayInt rows 
= m_owner
->CalcRowLabelsExposed( GetUpdateRegion() ); 
3502     m_owner
->DrawRowLabels( dc 
, rows 
); 
3506 void wxGridRowLabelWindow::OnMouseEvent( wxMouseEvent
& event 
) 
3508     m_owner
->ProcessRowLabelMouseEvent( event 
); 
3512 void wxGridRowLabelWindow::OnMouseWheel( wxMouseEvent
& event 
) 
3514     m_owner
->GetEventHandler()->ProcessEvent(event
); 
3518 // This seems to be required for wxMotif otherwise the mouse 
3519 // cursor must be in the cell edit control to get key events 
3521 void wxGridRowLabelWindow::OnKeyDown( wxKeyEvent
& event 
) 
3523     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3526 void wxGridRowLabelWindow::OnKeyUp( wxKeyEvent
& event 
) 
3528     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3533 ////////////////////////////////////////////////////////////////////// 
3535 IMPLEMENT_DYNAMIC_CLASS( wxGridColLabelWindow
, wxWindow 
) 
3537 BEGIN_EVENT_TABLE( wxGridColLabelWindow
, wxWindow 
) 
3538     EVT_PAINT( wxGridColLabelWindow::OnPaint 
) 
3539     EVT_MOUSEWHEEL( wxGridColLabelWindow::OnMouseWheel
) 
3540     EVT_MOUSE_EVENTS( wxGridColLabelWindow::OnMouseEvent 
) 
3541     EVT_KEY_DOWN( wxGridColLabelWindow::OnKeyDown 
) 
3542     EVT_KEY_UP( wxGridColLabelWindow::OnKeyUp 
) 
3545 wxGridColLabelWindow::wxGridColLabelWindow( wxGrid 
*parent
, 
3547                                             const wxPoint 
&pos
, const wxSize 
&size 
) 
3548   : wxWindow( parent
, id
, pos
, size
, wxWANTS_CHARS
|wxBORDER_NONE
|wxFULL_REPAINT_ON_RESIZE 
) 
3553 void wxGridColLabelWindow::OnPaint( wxPaintEvent
& WXUNUSED(event
) ) 
3557     // NO - don't do this because it will set both the x and y origin 
3558     // coords to match the parent scrolled window and we just want to 
3559     // set the x coord  - MB 
3561     // m_owner->PrepareDC( dc ); 
3564     m_owner
->CalcUnscrolledPosition( 0, 0, &x
, &y 
); 
3565     dc
.SetDeviceOrigin( -x
, 0 ); 
3567     wxArrayInt cols 
= m_owner
->CalcColLabelsExposed( GetUpdateRegion() ); 
3568     m_owner
->DrawColLabels( dc 
, cols 
); 
3572 void wxGridColLabelWindow::OnMouseEvent( wxMouseEvent
& event 
) 
3574     m_owner
->ProcessColLabelMouseEvent( event 
); 
3577 void wxGridColLabelWindow::OnMouseWheel( wxMouseEvent
& event 
) 
3579     m_owner
->GetEventHandler()->ProcessEvent(event
); 
3583 // This seems to be required for wxMotif otherwise the mouse 
3584 // cursor must be in the cell edit control to get key events 
3586 void wxGridColLabelWindow::OnKeyDown( wxKeyEvent
& event 
) 
3588     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3591 void wxGridColLabelWindow::OnKeyUp( wxKeyEvent
& event 
) 
3593     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3598 ////////////////////////////////////////////////////////////////////// 
3600 IMPLEMENT_DYNAMIC_CLASS( wxGridCornerLabelWindow
, wxWindow 
) 
3602 BEGIN_EVENT_TABLE( wxGridCornerLabelWindow
, wxWindow 
) 
3603     EVT_MOUSEWHEEL( wxGridCornerLabelWindow::OnMouseWheel
) 
3604     EVT_MOUSE_EVENTS( wxGridCornerLabelWindow::OnMouseEvent 
) 
3605     EVT_PAINT( wxGridCornerLabelWindow::OnPaint
) 
3606     EVT_KEY_DOWN( wxGridCornerLabelWindow::OnKeyDown 
) 
3607     EVT_KEY_UP( wxGridCornerLabelWindow::OnKeyUp 
) 
3610 wxGridCornerLabelWindow::wxGridCornerLabelWindow( wxGrid 
*parent
, 
3612                                                   const wxPoint 
&pos
, const wxSize 
&size 
) 
3613   : wxWindow( parent
, id
, pos
, size
, wxWANTS_CHARS
|wxBORDER_NONE
|wxFULL_REPAINT_ON_RESIZE 
) 
3618 void wxGridCornerLabelWindow::OnPaint( wxPaintEvent
& WXUNUSED(event
) ) 
3622     int client_height 
= 0; 
3623     int client_width 
= 0; 
3624     GetClientSize( &client_width
, &client_height 
); 
3626     dc
.SetPen( wxPen(wxSystemSettings::GetColour(wxSYS_COLOUR_3DDKSHADOW
),1, wxSOLID
) ); 
3627     dc
.DrawLine( client_width
-1, client_height
-1, client_width
-1, 0 ); 
3628     dc
.DrawLine( client_width
-1, client_height
-1, 0, client_height
-1 ); 
3629     dc
.DrawLine( 0, 0, client_width
, 0 ); 
3630     dc
.DrawLine( 0, 0, 0, client_height 
); 
3632     dc
.SetPen( *wxWHITE_PEN 
); 
3633     dc
.DrawLine( 1, 1, client_width
-1, 1 ); 
3634     dc
.DrawLine( 1, 1, 1, client_height
-1 ); 
3638 void wxGridCornerLabelWindow::OnMouseEvent( wxMouseEvent
& event 
) 
3640     m_owner
->ProcessCornerLabelMouseEvent( event 
); 
3644 void wxGridCornerLabelWindow::OnMouseWheel( wxMouseEvent
& event 
) 
3646     m_owner
->GetEventHandler()->ProcessEvent(event
); 
3649 // This seems to be required for wxMotif otherwise the mouse 
3650 // cursor must be in the cell edit control to get key events 
3652 void wxGridCornerLabelWindow::OnKeyDown( wxKeyEvent
& event 
) 
3654     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3657 void wxGridCornerLabelWindow::OnKeyUp( wxKeyEvent
& event 
) 
3659     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3664 ////////////////////////////////////////////////////////////////////// 
3666 IMPLEMENT_DYNAMIC_CLASS( wxGridWindow
, wxWindow 
) 
3668 BEGIN_EVENT_TABLE( wxGridWindow
, wxWindow 
) 
3669     EVT_PAINT( wxGridWindow::OnPaint 
) 
3670     EVT_MOUSEWHEEL( wxGridWindow::OnMouseWheel
) 
3671     EVT_MOUSE_EVENTS( wxGridWindow::OnMouseEvent 
) 
3672     EVT_KEY_DOWN( wxGridWindow::OnKeyDown 
) 
3673     EVT_KEY_UP( wxGridWindow::OnKeyUp 
) 
3674     EVT_SET_FOCUS( wxGridWindow::OnFocus 
) 
3675     EVT_KILL_FOCUS( wxGridWindow::OnFocus 
) 
3676     EVT_ERASE_BACKGROUND( wxGridWindow::OnEraseBackground 
) 
3679 wxGridWindow::wxGridWindow( wxGrid 
*parent
, 
3680                             wxGridRowLabelWindow 
*rowLblWin
, 
3681                             wxGridColLabelWindow 
*colLblWin
, 
3684                             const wxSize 
&size 
) 
3685             : wxWindow( parent
, id
, pos
, size
, wxWANTS_CHARS 
| wxBORDER_NONE 
| wxCLIP_CHILDREN
|wxFULL_REPAINT_ON_RESIZE
, 
3686                         wxT("grid window") ) 
3690     m_rowLabelWin 
= rowLblWin
; 
3691     m_colLabelWin 
= colLblWin
; 
3692     SetBackgroundColour(_T("WHITE")); 
3696 wxGridWindow::~wxGridWindow() 
3701 void wxGridWindow::OnPaint( wxPaintEvent 
&WXUNUSED(event
) ) 
3703     wxPaintDC 
dc( this ); 
3704     m_owner
->PrepareDC( dc 
); 
3705     wxRegion reg 
= GetUpdateRegion(); 
3706     wxGridCellCoordsArray DirtyCells 
= m_owner
->CalcCellsExposed( reg 
); 
3707     m_owner
->DrawGridCellArea( dc 
, DirtyCells
); 
3708 #if WXGRID_DRAW_LINES 
3709     m_owner
->DrawAllGridLines( dc
, reg 
); 
3711     m_owner
->DrawGridSpace( dc 
); 
3712     m_owner
->DrawHighlight( dc 
, DirtyCells 
); 
3716 void wxGridWindow::ScrollWindow( int dx
, int dy
, const wxRect 
*rect 
) 
3718     wxWindow::ScrollWindow( dx
, dy
, rect 
); 
3719     m_rowLabelWin
->ScrollWindow( 0, dy
, rect 
); 
3720     m_colLabelWin
->ScrollWindow( dx
, 0, rect 
); 
3724 void wxGridWindow::OnMouseEvent( wxMouseEvent
& event 
) 
3726     m_owner
->ProcessGridCellMouseEvent( event 
); 
3729 void wxGridWindow::OnMouseWheel( wxMouseEvent
& event 
) 
3731     m_owner
->GetEventHandler()->ProcessEvent(event
); 
3734 // This seems to be required for wxMotif/wxGTK otherwise the mouse 
3735 // cursor must be in the cell edit control to get key events 
3737 void wxGridWindow::OnKeyDown( wxKeyEvent
& event 
) 
3739     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3742 void wxGridWindow::OnKeyUp( wxKeyEvent
& event 
) 
3744     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3747 void wxGridWindow::OnEraseBackground( wxEraseEvent
& WXUNUSED(event
) ) 
3751 void wxGridWindow::OnFocus(wxFocusEvent
& event
) 
3753     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) 
3757 ////////////////////////////////////////////////////////////////////// 
3759 // Internal Helper function for computing row or column from some 
3760 // (unscrolled) coordinate value, using either 
3761 // m_defaultRowHeight/m_defaultColWidth or binary search on array 
3762 // of m_rowBottoms/m_ColRights to speed up the search! 
3764 // Internal helper macros for simpler use of that function 
3766 static int CoordToRowOrCol(int coord
, int defaultDist
, int minDist
, 
3767                            const wxArrayInt
& BorderArray
, int nMax
, 
3770 #define internalXToCol(x) CoordToRowOrCol(x, m_defaultColWidth, \ 
3771                                           m_minAcceptableColWidth, \ 
3772                                           m_colRights, m_numCols, TRUE) 
3773 #define internalYToRow(y) CoordToRowOrCol(y, m_defaultRowHeight, \ 
3774                                           m_minAcceptableRowHeight, \ 
3775                                           m_rowBottoms, m_numRows, TRUE) 
3776 ///////////////////////////////////////////////////////////////////// 
3778 #if wxUSE_EXTENDED_RTTI 
3779 WX_DEFINE_FLAGS( wxGridStyle 
) 
3781 wxBEGIN_FLAGS( wxGridStyle 
) 
3782     // new style border flags, we put them first to 
3783     // use them for streaming out 
3784     wxFLAGS_MEMBER(wxBORDER_SIMPLE
) 
3785     wxFLAGS_MEMBER(wxBORDER_SUNKEN
) 
3786     wxFLAGS_MEMBER(wxBORDER_DOUBLE
) 
3787     wxFLAGS_MEMBER(wxBORDER_RAISED
) 
3788     wxFLAGS_MEMBER(wxBORDER_STATIC
) 
3789     wxFLAGS_MEMBER(wxBORDER_NONE
) 
3791     // old style border flags 
3792     wxFLAGS_MEMBER(wxSIMPLE_BORDER
) 
3793     wxFLAGS_MEMBER(wxSUNKEN_BORDER
) 
3794     wxFLAGS_MEMBER(wxDOUBLE_BORDER
) 
3795     wxFLAGS_MEMBER(wxRAISED_BORDER
) 
3796     wxFLAGS_MEMBER(wxSTATIC_BORDER
) 
3797     wxFLAGS_MEMBER(wxBORDER
) 
3799     // standard window styles 
3800     wxFLAGS_MEMBER(wxTAB_TRAVERSAL
) 
3801     wxFLAGS_MEMBER(wxCLIP_CHILDREN
) 
3802     wxFLAGS_MEMBER(wxTRANSPARENT_WINDOW
) 
3803     wxFLAGS_MEMBER(wxWANTS_CHARS
) 
3804     wxFLAGS_MEMBER(wxFULL_REPAINT_ON_RESIZE
) 
3805     wxFLAGS_MEMBER(wxALWAYS_SHOW_SB 
) 
3806     wxFLAGS_MEMBER(wxVSCROLL
) 
3807     wxFLAGS_MEMBER(wxHSCROLL
) 
3809 wxEND_FLAGS( wxGridStyle 
) 
3811 IMPLEMENT_DYNAMIC_CLASS_XTI(wxGrid
, wxScrolledWindow
,"wx/grid.h") 
3813 wxBEGIN_PROPERTIES_TABLE(wxGrid
) 
3814     wxHIDE_PROPERTY( Children 
) 
3815     wxPROPERTY_FLAGS( WindowStyle 
, wxGridStyle 
, long , SetWindowStyleFlag 
, GetWindowStyleFlag 
, , 0 /*flags*/ , wxT("Helpstring") , wxT("group")) // style 
3816 wxEND_PROPERTIES_TABLE() 
3818 wxBEGIN_HANDLERS_TABLE(wxGrid
) 
3819 wxEND_HANDLERS_TABLE() 
3821 wxCONSTRUCTOR_5( wxGrid 
, wxWindow
* , Parent 
, wxWindowID 
, Id 
, wxPoint 
, Position 
, wxSize 
, Size 
, long , WindowStyle 
)  
3824  TODO : Expose more information of a list's layout etc. via appropriate objects (à la NotebookPageInfo) 
3827 IMPLEMENT_DYNAMIC_CLASS( wxGrid
, wxScrolledWindow 
) 
3830 BEGIN_EVENT_TABLE( wxGrid
, wxScrolledWindow 
) 
3831     EVT_PAINT( wxGrid::OnPaint 
) 
3832     EVT_SIZE( wxGrid::OnSize 
) 
3833     EVT_KEY_DOWN( wxGrid::OnKeyDown 
) 
3834     EVT_KEY_UP( wxGrid::OnKeyUp 
) 
3835     EVT_ERASE_BACKGROUND( wxGrid::OnEraseBackground 
) 
3840     // in order to make sure that a size event is not 
3841     // trigerred in a unfinished state 
3842     m_cornerLabelWin 
= NULL 
; 
3843     m_rowLabelWin 
= NULL 
; 
3844     m_colLabelWin 
= NULL 
; 
3848 wxGrid::wxGrid( wxWindow 
*parent
, 
3853                  const wxString
& name 
) 
3854   : wxScrolledWindow( parent
, id
, pos
, size
, (style 
| wxWANTS_CHARS
), name 
), 
3855     m_colMinWidths(GRID_HASH_SIZE
), 
3856     m_rowMinHeights(GRID_HASH_SIZE
) 
3861 bool wxGrid::Create(wxWindow 
*parent
, wxWindowID id
, 
3862                           const wxPoint
& pos
, const wxSize
& size
, 
3863                           long style
, const wxString
& name
) 
3865     if (!wxScrolledWindow::Create(parent
, id
, pos
, size
, 
3866                                   style 
| wxWANTS_CHARS 
, name
)) 
3869     m_colMinWidths 
= wxLongToLongHashMap(GRID_HASH_SIZE
) ; 
3870     m_rowMinHeights 
= wxLongToLongHashMap(GRID_HASH_SIZE
) ; 
3881     // Must do this or ~wxScrollHelper will pop the wrong event handler 
3882     SetTargetWindow(this); 
3884     wxSafeDecRef(m_defaultCellAttr
); 
3886 #ifdef DEBUG_ATTR_CACHE 
3887     size_t total 
= gs_nAttrCacheHits 
+ gs_nAttrCacheMisses
; 
3888     wxPrintf(_T("wxGrid attribute cache statistics: " 
3889                 "total: %u, hits: %u (%u%%)\n"), 
3890              total
, gs_nAttrCacheHits
, 
3891              total 
? (gs_nAttrCacheHits
*100) / total 
: 0); 
3897     delete m_typeRegistry
; 
3903 // ----- internal init and update functions 
3906 void wxGrid::Create() 
3908     m_created 
= FALSE
;    // set to TRUE by CreateGrid 
3910     m_table        
= (wxGridTableBase 
*) NULL
; 
3913     m_cellEditCtrlEnabled 
= FALSE
; 
3915     m_defaultCellAttr 
= new wxGridCellAttr(); 
3917     // Set default cell attributes 
3918     m_defaultCellAttr
->SetDefAttr(m_defaultCellAttr
); 
3919     m_defaultCellAttr
->SetKind(wxGridCellAttr::Default
); 
3920     m_defaultCellAttr
->SetFont(GetFont()); 
3921     m_defaultCellAttr
->SetAlignment(wxALIGN_LEFT
, wxALIGN_TOP
); 
3922     m_defaultCellAttr
->SetTextColour( 
3923         wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT
)); 
3924     m_defaultCellAttr
->SetBackgroundColour( 
3925         wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW
)); 
3926     m_defaultCellAttr
->SetRenderer(new wxGridCellStringRenderer
); 
3927     m_defaultCellAttr
->SetEditor(new wxGridCellTextEditor
); 
3932     m_currentCellCoords 
= wxGridNoCellCoords
; 
3934     m_rowLabelWidth 
= WXGRID_DEFAULT_ROW_LABEL_WIDTH
; 
3935     m_colLabelHeight 
= WXGRID_DEFAULT_COL_LABEL_HEIGHT
; 
3937     // create the type registry 
3938     m_typeRegistry 
= new wxGridTypeRegistry
; 
3941     // subwindow components that make up the wxGrid 
3942     m_cornerLabelWin 
= new wxGridCornerLabelWindow( this, 
3947     m_rowLabelWin 
= new wxGridRowLabelWindow( this, 
3952     m_colLabelWin 
= new wxGridColLabelWindow( this, 
3957     m_gridWin 
= new wxGridWindow( this, 
3964     SetTargetWindow( m_gridWin 
); 
3970 bool wxGrid::CreateGrid( int numRows
, int numCols
, 
3971                          wxGrid::wxGridSelectionModes selmode 
) 
3973     wxCHECK_MSG( !m_created
, 
3975                  wxT("wxGrid::CreateGrid or wxGrid::SetTable called more than once") ); 
3977     m_numRows 
= numRows
; 
3978     m_numCols 
= numCols
; 
3980     m_table 
= new wxGridStringTable( m_numRows
, m_numCols 
); 
3981     m_table
->SetView( this ); 
3983     m_selection 
= new wxGridSelection( this, selmode 
); 
3992 void wxGrid::SetSelectionMode(wxGrid::wxGridSelectionModes selmode
) 
3994     wxCHECK_RET( m_created
, 
3995                  wxT("Called wxGrid::SetSelectionMode() before calling CreateGrid()") ); 
3997     m_selection
->SetSelectionMode( selmode 
); 
4000 wxGrid::wxGridSelectionModes 
wxGrid::GetSelectionMode() const 
4002     wxCHECK_MSG( m_created
, wxGrid::wxGridSelectCells
, 
4003                  wxT("Called wxGrid::GetSelectionMode() before calling CreateGrid()") ); 
4005     return m_selection
->GetSelectionMode(); 
4008 bool wxGrid::SetTable( wxGridTableBase 
*table
, bool takeOwnership
, 
4009                        wxGrid::wxGridSelectionModes selmode 
) 
4013         // stop all processing 
4018             wxGridTableBase 
*t
=m_table
; 
4031         m_numRows 
= table
->GetNumberRows(); 
4032         m_numCols 
= table
->GetNumberCols(); 
4035         m_table
->SetView( this ); 
4038         m_selection 
= new wxGridSelection( this, selmode 
); 
4051     m_rowLabelWidth  
= WXGRID_DEFAULT_ROW_LABEL_WIDTH
; 
4052     m_colLabelHeight 
= WXGRID_DEFAULT_COL_LABEL_HEIGHT
; 
4054     if ( m_rowLabelWin 
) 
4056         m_labelBackgroundColour 
= m_rowLabelWin
->GetBackgroundColour(); 
4060         m_labelBackgroundColour 
= wxColour( _T("WHITE") ); 
4063     m_labelTextColour 
= wxColour( _T("BLACK") ); 
4066     m_attrCache
.row 
= -1; 
4067     m_attrCache
.col 
= -1; 
4068     m_attrCache
.attr 
= NULL
; 
4070     // TODO: something better than this ? 
4072     m_labelFont 
= this->GetFont(); 
4073     m_labelFont
.SetWeight( wxBOLD 
); 
4075     m_rowLabelHorizAlign 
= wxALIGN_CENTRE
; 
4076     m_rowLabelVertAlign  
= wxALIGN_CENTRE
; 
4078     m_colLabelHorizAlign 
= wxALIGN_CENTRE
; 
4079     m_colLabelVertAlign  
= wxALIGN_CENTRE
; 
4080     m_colLabelTextOrientation 
= wxHORIZONTAL
; 
4082     m_defaultColWidth  
= WXGRID_DEFAULT_COL_WIDTH
; 
4083     m_defaultRowHeight 
= m_gridWin
->GetCharHeight(); 
4085     m_minAcceptableColWidth  
= WXGRID_MIN_COL_WIDTH
; 
4086     m_minAcceptableRowHeight 
= WXGRID_MIN_ROW_HEIGHT
; 
4088 #if defined(__WXMOTIF__) || defined(__WXGTK__)  // see also text ctrl sizing in ShowCellEditControl() 
4089     m_defaultRowHeight 
+= 8; 
4091     m_defaultRowHeight 
+= 4; 
4094     m_gridLineColour 
= wxColour( 192,192,192 ); 
4095     m_gridLinesEnabled 
= TRUE
; 
4096     m_cellHighlightColour 
= *wxBLACK
; 
4097     m_cellHighlightPenWidth 
= 2; 
4098     m_cellHighlightROPenWidth 
= 1; 
4100     m_cursorMode  
= WXGRID_CURSOR_SELECT_CELL
; 
4101     m_winCapture 
= (wxWindow 
*)NULL
; 
4102     m_canDragRowSize 
= TRUE
; 
4103     m_canDragColSize 
= TRUE
; 
4104     m_canDragGridSize 
= TRUE
; 
4106     m_dragRowOrCol 
= -1; 
4107     m_isDragging 
= FALSE
; 
4108     m_startDragPos 
= wxDefaultPosition
; 
4110     m_waitForSlowClick 
= FALSE
; 
4112     m_rowResizeCursor 
= wxCursor( wxCURSOR_SIZENS 
); 
4113     m_colResizeCursor 
= wxCursor( wxCURSOR_SIZEWE 
); 
4115     m_currentCellCoords 
= wxGridNoCellCoords
; 
4117     m_selectingTopLeft 
= wxGridNoCellCoords
; 
4118     m_selectingBottomRight 
= wxGridNoCellCoords
; 
4119     m_selectionBackground 
= wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHT
); 
4120     m_selectionForeground 
= wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHTTEXT
); 
4122     m_editable 
= TRUE
;  // default for whole grid 
4124     m_inOnKeyDown 
= FALSE
; 
4131 // ---------------------------------------------------------------------------- 
4132 // the idea is to call these functions only when necessary because they create 
4133 // quite big arrays which eat memory mostly unnecessary - in particular, if 
4134 // default widths/heights are used for all rows/columns, we may not use these 
4137 // with some extra code, it should be possible to only store the 
4138 // widths/heights different from default ones but this will be done later... 
4139 // ---------------------------------------------------------------------------- 
4141 void wxGrid::InitRowHeights() 
4143     m_rowHeights
.Empty(); 
4144     m_rowBottoms
.Empty(); 
4146     m_rowHeights
.Alloc( m_numRows 
); 
4147     m_rowBottoms
.Alloc( m_numRows 
); 
4151     m_rowHeights
.Add( m_defaultRowHeight
, m_numRows 
); 
4153     for ( int i 
= 0;  i 
< m_numRows
;  i
++ ) 
4155         rowBottom 
+= m_defaultRowHeight
; 
4156         m_rowBottoms
.Add( rowBottom 
); 
4160 void wxGrid::InitColWidths() 
4162     m_colWidths
.Empty(); 
4163     m_colRights
.Empty(); 
4165     m_colWidths
.Alloc( m_numCols 
); 
4166     m_colRights
.Alloc( m_numCols 
); 
4169     m_colWidths
.Add( m_defaultColWidth
, m_numCols 
); 
4171     for ( int i 
= 0;  i 
< m_numCols
;  i
++ ) 
4173         colRight 
+= m_defaultColWidth
; 
4174         m_colRights
.Add( colRight 
); 
4178 int wxGrid::GetColWidth(int col
) const 
4180     return m_colWidths
.IsEmpty() ? m_defaultColWidth 
: m_colWidths
[col
]; 
4183 int wxGrid::GetColLeft(int col
) const 
4185     return m_colRights
.IsEmpty() ? col 
* m_defaultColWidth
 
4186                                  : m_colRights
[col
] - m_colWidths
[col
]; 
4189 int wxGrid::GetColRight(int col
) const 
4191     return m_colRights
.IsEmpty() ? (col 
+ 1) * m_defaultColWidth
 
4195 int wxGrid::GetRowHeight(int row
) const 
4197     return m_rowHeights
.IsEmpty() ? m_defaultRowHeight 
: m_rowHeights
[row
]; 
4200 int wxGrid::GetRowTop(int row
) const 
4202     return m_rowBottoms
.IsEmpty() ? row 
* m_defaultRowHeight
 
4203                                   : m_rowBottoms
[row
] - m_rowHeights
[row
]; 
4206 int wxGrid::GetRowBottom(int row
) const 
4208     return m_rowBottoms
.IsEmpty() ? (row 
+ 1) * m_defaultRowHeight
 
4209                                   : m_rowBottoms
[row
]; 
4212 void wxGrid::CalcDimensions() 
4215     GetClientSize( &cw
, &ch 
); 
4217     if ( m_rowLabelWin
->IsShown() ) 
4218         cw 
-= m_rowLabelWidth
; 
4219     if ( m_colLabelWin
->IsShown() ) 
4220         ch 
-= m_colLabelHeight
; 
4223     int w 
= m_numCols 
> 0 ? GetColRight(m_numCols 
- 1) + m_extraWidth 
+ 1 : 0; 
4224     int h 
= m_numRows 
> 0 ? GetRowBottom(m_numRows 
- 1) + m_extraHeight 
+ 1 : 0; 
4226     // take into account editor if shown 
4227     if( IsCellEditControlShown() ) 
4230       int r 
= m_currentCellCoords
.GetRow(); 
4231       int c 
= m_currentCellCoords
.GetCol(); 
4232       int x 
= GetColLeft(c
); 
4233       int y 
= GetRowTop(r
); 
4235       // how big is the editor 
4236       wxGridCellAttr
* attr 
= GetCellAttr(r
, c
); 
4237       wxGridCellEditor
* editor 
= attr
->GetEditor(this, r
, c
); 
4238       editor
->GetControl()->GetSize(&w2
, &h2
); 
4241       if( w2 
> w 
) w 
= w2
; 
4242       if( h2 
> h 
) h 
= h2
; 
4247     // preserve (more or less) the previous position 
4249     GetViewStart( &x
, &y 
); 
4251     // ensure the position is valid for the new scroll ranges 
4253         x 
= wxMax( w 
- 1, 0 ); 
4255         y 
= wxMax( h 
- 1, 0 ); 
4257     // do set scrollbar parameters 
4258     SetScrollbars( GRID_SCROLL_LINE_X
, GRID_SCROLL_LINE_Y
, 
4259                    GetScrollX(w
), GetScrollY(h
), x
, y
, 
4260                    GetBatchCount() != 0); 
4262     // if our OnSize() hadn't been called (it would if we have scrollbars), we 
4263     // still must reposition the children 
4268 void wxGrid::CalcWindowSizes() 
4270     // escape if the window is has not been fully created yet 
4272     if ( m_cornerLabelWin 
== NULL 
) 
4276     GetClientSize( &cw
, &ch 
); 
4278     if ( m_cornerLabelWin
->IsShown() ) 
4279         m_cornerLabelWin
->SetSize( 0, 0, m_rowLabelWidth
, m_colLabelHeight 
); 
4281     if ( m_colLabelWin
->IsShown() ) 
4282         m_colLabelWin
->SetSize( m_rowLabelWidth
, 0, cw
-m_rowLabelWidth
, m_colLabelHeight
); 
4284     if ( m_rowLabelWin
->IsShown() ) 
4285         m_rowLabelWin
->SetSize( 0, m_colLabelHeight
, m_rowLabelWidth
, ch
-m_colLabelHeight
); 
4287     if ( m_gridWin
->IsShown() ) 
4288         m_gridWin
->SetSize( m_rowLabelWidth
, m_colLabelHeight
, cw
-m_rowLabelWidth
, ch
-m_colLabelHeight
); 
4292 // this is called when the grid table sends a message to say that it 
4293 // has been redimensioned 
4295 bool wxGrid::Redimension( wxGridTableMessage
& msg 
) 
4298     bool result 
= FALSE
; 
4300     // Clear the attribute cache as the attribute might refer to a different 
4301     // cell than stored in the cache after adding/removing rows/columns. 
4303     // By the same reasoning, the editor should be dismissed if columns are 
4304     // added or removed. And for consistency, it should IMHO always be 
4305     // removed, not only if the cell "underneath" it actually changes. 
4306     // For now, I intentionally do not save the editor's content as the 
4307     // cell it might want to save that stuff to might no longer exist. 
4308     HideCellEditControl(); 
4310     // if we were using the default widths/heights so far, we must change them 
4312     if ( m_colWidths
.IsEmpty() ) 
4317     if ( m_rowHeights
.IsEmpty() ) 
4323     switch ( msg
.GetId() ) 
4325         case wxGRIDTABLE_NOTIFY_ROWS_INSERTED
: 
4327             size_t pos 
= msg
.GetCommandInt(); 
4328             int numRows 
= msg
.GetCommandInt2(); 
4330             m_numRows 
+= numRows
; 
4332             if ( !m_rowHeights
.IsEmpty() ) 
4334                 m_rowHeights
.Insert( m_defaultRowHeight
, pos
, numRows 
); 
4335                 m_rowBottoms
.Insert( 0, pos
, numRows 
); 
4338                 if ( pos 
> 0 ) bottom 
= m_rowBottoms
[pos
-1]; 
4340                 for ( i 
= pos
;  i 
< m_numRows
;  i
++ ) 
4342                     bottom 
+= m_rowHeights
[i
]; 
4343                     m_rowBottoms
[i
] = bottom
; 
4346             if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
4348                 // if we have just inserted cols into an empty grid the current 
4349                 // cell will be undefined... 
4351                 SetCurrentCell( 0, 0 ); 
4355                 m_selection
->UpdateRows( pos
, numRows 
); 
4356             wxGridCellAttrProvider 
* attrProvider 
= m_table
->GetAttrProvider(); 
4358                 attrProvider
->UpdateAttrRows( pos
, numRows 
); 
4360             if ( !GetBatchCount() ) 
4363                 m_rowLabelWin
->Refresh(); 
4369         case wxGRIDTABLE_NOTIFY_ROWS_APPENDED
: 
4371             int numRows 
= msg
.GetCommandInt(); 
4372             int oldNumRows 
= m_numRows
; 
4373             m_numRows 
+= numRows
; 
4375             if ( !m_rowHeights
.IsEmpty() ) 
4377                 m_rowHeights
.Add( m_defaultRowHeight
, numRows 
); 
4378                 m_rowBottoms
.Add( 0, numRows 
); 
4381                 if ( oldNumRows 
> 0 ) bottom 
= m_rowBottoms
[oldNumRows
-1]; 
4383                 for ( i 
= oldNumRows
;  i 
< m_numRows
;  i
++ ) 
4385                     bottom 
+= m_rowHeights
[i
]; 
4386                     m_rowBottoms
[i
] = bottom
; 
4389             if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
4391                 // if we have just inserted cols into an empty grid the current 
4392                 // cell will be undefined... 
4394                 SetCurrentCell( 0, 0 ); 
4396             if ( !GetBatchCount() ) 
4399                 m_rowLabelWin
->Refresh(); 
4405         case wxGRIDTABLE_NOTIFY_ROWS_DELETED
: 
4407             size_t pos 
= msg
.GetCommandInt(); 
4408             int numRows 
= msg
.GetCommandInt2(); 
4409             m_numRows 
-= numRows
; 
4411             if ( !m_rowHeights
.IsEmpty() ) 
4413                 m_rowHeights
.RemoveAt( pos
, numRows 
); 
4414                 m_rowBottoms
.RemoveAt( pos
, numRows 
); 
4417                 for ( i 
= 0;  i 
< m_numRows
;  i
++ ) 
4419                     h 
+= m_rowHeights
[i
]; 
4420                     m_rowBottoms
[i
] = h
; 
4425                 m_currentCellCoords 
= wxGridNoCellCoords
; 
4429                 if ( m_currentCellCoords
.GetRow() >= m_numRows 
) 
4430                     m_currentCellCoords
.Set( 0, 0 ); 
4434                 m_selection
->UpdateRows( pos
, -((int)numRows
) ); 
4435             wxGridCellAttrProvider 
* attrProvider 
= m_table
->GetAttrProvider(); 
4437                 attrProvider
->UpdateAttrRows( pos
, -((int)numRows
) ); 
4438 // ifdef'd out following patch from Paul Gammans 
4440                 // No need to touch column attributes, unless we 
4441                 // removed _all_ rows, in this case, we remove 
4442                 // all column attributes. 
4443                 // I hate to do this here, but the 
4444                 // needed data is not available inside UpdateAttrRows. 
4445                 if ( !GetNumberRows() ) 
4446                     attrProvider
->UpdateAttrCols( 0, -GetNumberCols() ); 
4449             if ( !GetBatchCount() ) 
4452                 m_rowLabelWin
->Refresh(); 
4458         case wxGRIDTABLE_NOTIFY_COLS_INSERTED
: 
4460             size_t pos 
= msg
.GetCommandInt(); 
4461             int numCols 
= msg
.GetCommandInt2(); 
4462             m_numCols 
+= numCols
; 
4464             if ( !m_colWidths
.IsEmpty() ) 
4466                 m_colWidths
.Insert( m_defaultColWidth
, pos
, numCols 
); 
4467                 m_colRights
.Insert( 0, pos
, numCols 
); 
4470                 if ( pos 
> 0 ) right 
= m_colRights
[pos
-1]; 
4472                 for ( i 
= pos
;  i 
< m_numCols
;  i
++ ) 
4474                     right 
+= m_colWidths
[i
]; 
4475                     m_colRights
[i
] = right
; 
4478             if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
4480                 // if we have just inserted cols into an empty grid the current 
4481                 // cell will be undefined... 
4483                 SetCurrentCell( 0, 0 ); 
4487                 m_selection
->UpdateCols( pos
, numCols 
); 
4488             wxGridCellAttrProvider 
* attrProvider 
= m_table
->GetAttrProvider(); 
4490                 attrProvider
->UpdateAttrCols( pos
, numCols 
); 
4491             if ( !GetBatchCount() ) 
4494                 m_colLabelWin
->Refresh(); 
4501         case wxGRIDTABLE_NOTIFY_COLS_APPENDED
: 
4503             int numCols 
= msg
.GetCommandInt(); 
4504             int oldNumCols 
= m_numCols
; 
4505             m_numCols 
+= numCols
; 
4506             if ( !m_colWidths
.IsEmpty() ) 
4508                 m_colWidths
.Add( m_defaultColWidth
, numCols 
); 
4509                 m_colRights
.Add( 0, numCols 
); 
4512                 if ( oldNumCols 
> 0 ) right 
= m_colRights
[oldNumCols
-1]; 
4514                 for ( i 
= oldNumCols
;  i 
< m_numCols
;  i
++ ) 
4516                     right 
+= m_colWidths
[i
]; 
4517                     m_colRights
[i
] = right
; 
4520             if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
4522                 // if we have just inserted cols into an empty grid the current 
4523                 // cell will be undefined... 
4525                 SetCurrentCell( 0, 0 ); 
4527             if ( !GetBatchCount() ) 
4530                 m_colLabelWin
->Refresh(); 
4536         case wxGRIDTABLE_NOTIFY_COLS_DELETED
: 
4538             size_t pos 
= msg
.GetCommandInt(); 
4539             int numCols 
= msg
.GetCommandInt2(); 
4540             m_numCols 
-= numCols
; 
4542             if ( !m_colWidths
.IsEmpty() ) 
4544                 m_colWidths
.RemoveAt( pos
, numCols 
); 
4545                 m_colRights
.RemoveAt( pos
, numCols 
); 
4548                 for ( i 
= 0;  i 
< m_numCols
;  i
++ ) 
4550                     w 
+= m_colWidths
[i
]; 
4556                 m_currentCellCoords 
= wxGridNoCellCoords
; 
4560                 if ( m_currentCellCoords
.GetCol() >= m_numCols 
) 
4561                   m_currentCellCoords
.Set( 0, 0 ); 
4565                 m_selection
->UpdateCols( pos
, -((int)numCols
) ); 
4566             wxGridCellAttrProvider 
* attrProvider 
= m_table
->GetAttrProvider(); 
4568                 attrProvider
->UpdateAttrCols( pos
, -((int)numCols
) ); 
4569 // ifdef'd out following patch from Paul Gammans 
4571                 // No need to touch row attributes, unless we 
4572                 // removed _all_ columns, in this case, we remove 
4573                 // all row attributes. 
4574                 // I hate to do this here, but the 
4575                 // needed data is not available inside UpdateAttrCols. 
4576                 if ( !GetNumberCols() ) 
4577                     attrProvider
->UpdateAttrRows( 0, -GetNumberRows() ); 
4580             if ( !GetBatchCount() ) 
4583                 m_colLabelWin
->Refresh(); 
4590     if (result 
&& !GetBatchCount() ) 
4591         m_gridWin
->Refresh(); 
4596 wxArrayInt 
wxGrid::CalcRowLabelsExposed( const wxRegion
& reg 
) 
4598     wxRegionIterator 
iter( reg 
); 
4601     wxArrayInt  rowlabels
; 
4608         // TODO: remove this when we can... 
4609         // There is a bug in wxMotif that gives garbage update 
4610         // rectangles if you jump-scroll a long way by clicking the 
4611         // scrollbar with middle button.  This is a work-around 
4613 #if defined(__WXMOTIF__) 
4615         m_gridWin
->GetClientSize( &cw
, &ch 
); 
4616         if ( r
.GetTop() > ch 
) r
.SetTop( 0 ); 
4617         r
.SetBottom( wxMin( r
.GetBottom(), ch 
) ); 
4620         // logical bounds of update region 
4623         CalcUnscrolledPosition( 0, r
.GetTop(), &dummy
, &top 
); 
4624         CalcUnscrolledPosition( 0, r
.GetBottom(), &dummy
, &bottom 
); 
4626         // find the row labels within these bounds 
4629         for ( row 
= internalYToRow(top
);  row 
< m_numRows
;  row
++ ) 
4631             if ( GetRowBottom(row
) < top 
) 
4634             if ( GetRowTop(row
) > bottom 
) 
4637             rowlabels
.Add( row 
); 
4647 wxArrayInt 
wxGrid::CalcColLabelsExposed( const wxRegion
& reg 
) 
4649     wxRegionIterator 
iter( reg 
); 
4652     wxArrayInt colLabels
; 
4659         // TODO: remove this when we can... 
4660         // There is a bug in wxMotif that gives garbage update 
4661         // rectangles if you jump-scroll a long way by clicking the 
4662         // scrollbar with middle button.  This is a work-around 
4664 #if defined(__WXMOTIF__) 
4666         m_gridWin
->GetClientSize( &cw
, &ch 
); 
4667         if ( r
.GetLeft() > cw 
) r
.SetLeft( 0 ); 
4668         r
.SetRight( wxMin( r
.GetRight(), cw 
) ); 
4671         // logical bounds of update region 
4674         CalcUnscrolledPosition( r
.GetLeft(), 0, &left
, &dummy 
); 
4675         CalcUnscrolledPosition( r
.GetRight(), 0, &right
, &dummy 
); 
4677         // find the cells within these bounds 
4680         for ( col 
= internalXToCol(left
);  col 
< m_numCols
;  col
++ ) 
4682             if ( GetColRight(col
) < left 
) 
4685             if ( GetColLeft(col
) > right 
) 
4688             colLabels
.Add( col 
); 
4697 wxGridCellCoordsArray 
wxGrid::CalcCellsExposed( const wxRegion
& reg 
) 
4699     wxRegionIterator 
iter( reg 
); 
4702     wxGridCellCoordsArray  cellsExposed
; 
4704     int left
, top
, right
, bottom
; 
4709         // TODO: remove this when we can... 
4710         // There is a bug in wxMotif that gives garbage update 
4711         // rectangles if you jump-scroll a long way by clicking the 
4712         // scrollbar with middle button.  This is a work-around 
4714 #if defined(__WXMOTIF__) 
4716         m_gridWin
->GetClientSize( &cw
, &ch 
); 
4717         if ( r
.GetTop() > ch 
) r
.SetTop( 0 ); 
4718         if ( r
.GetLeft() > cw 
) r
.SetLeft( 0 ); 
4719         r
.SetRight( wxMin( r
.GetRight(), cw 
) ); 
4720         r
.SetBottom( wxMin( r
.GetBottom(), ch 
) ); 
4723         // logical bounds of update region 
4725         CalcUnscrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top 
); 
4726         CalcUnscrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom 
); 
4728         // find the cells within these bounds 
4731         for ( row 
= internalYToRow(top
);  row 
< m_numRows
;  row
++ ) 
4733             if ( GetRowBottom(row
) <= top 
) 
4736             if ( GetRowTop(row
) > bottom 
) 
4739             for ( col 
= internalXToCol(left
);  col 
< m_numCols
;  col
++ ) 
4741                 if ( GetColRight(col
) <= left 
) 
4744                 if ( GetColLeft(col
) > right 
) 
4747                 cellsExposed
.Add( wxGridCellCoords( row
, col 
) ); 
4754     return cellsExposed
; 
4758 void wxGrid::ProcessRowLabelMouseEvent( wxMouseEvent
& event 
) 
4761     wxPoint 
pos( event
.GetPosition() ); 
4762     CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y 
); 
4764     if ( event
.Dragging() ) 
4768             m_isDragging 
= TRUE
; 
4769             m_rowLabelWin
->CaptureMouse(); 
4772         if ( event
.LeftIsDown() ) 
4774             switch( m_cursorMode 
) 
4776                 case WXGRID_CURSOR_RESIZE_ROW
: 
4778                     int cw
, ch
, left
, dummy
; 
4779                     m_gridWin
->GetClientSize( &cw
, &ch 
); 
4780                     CalcUnscrolledPosition( 0, 0, &left
, &dummy 
); 
4782                     wxClientDC 
dc( m_gridWin 
); 
4785                                GetRowTop(m_dragRowOrCol
) + 
4786                                GetRowMinimalHeight(m_dragRowOrCol
) ); 
4787                     dc
.SetLogicalFunction(wxINVERT
); 
4788                     if ( m_dragLastPos 
>= 0 ) 
4790                         dc
.DrawLine( left
, m_dragLastPos
, left
+cw
, m_dragLastPos 
); 
4792                     dc
.DrawLine( left
, y
, left
+cw
, y 
); 
4797                 case WXGRID_CURSOR_SELECT_ROW
: 
4798                     if ( (row 
= YToRow( y 
)) >= 0 ) 
4802                             m_selection
->SelectRow( row
, 
4803                                                     event
.ControlDown(), 
4810                 // default label to suppress warnings about "enumeration value 
4811                 // 'xxx' not handled in switch 
4819     if ( m_isDragging 
&& (event
.Entering() || event
.Leaving()) ) 
4824         if (m_rowLabelWin
->HasCapture()) m_rowLabelWin
->ReleaseMouse(); 
4825         m_isDragging 
= FALSE
; 
4828     // ------------ Entering or leaving the window 
4830     if ( event
.Entering() || event
.Leaving() ) 
4832         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_rowLabelWin
); 
4836     // ------------ Left button pressed 
4838     else if ( event
.LeftDown() ) 
4840         // don't send a label click event for a hit on the 
4841         // edge of the row label - this is probably the user 
4842         // wanting to resize the row 
4844         if ( YToEdgeOfRow(y
) < 0 ) 
4848                  !SendEvent( wxEVT_GRID_LABEL_LEFT_CLICK
, row
, -1, event 
) ) 
4850                 if ( !event
.ShiftDown() && !event
.ControlDown() ) 
4854                     if ( event
.ShiftDown() ) 
4856                         m_selection
->SelectBlock( m_currentCellCoords
.GetRow(), 
4859                                                   GetNumberCols() - 1, 
4860                                                   event
.ControlDown(), 
4867                         m_selection
->SelectRow( row
, 
4868                                                 event
.ControlDown(), 
4875                 ChangeCursorMode(WXGRID_CURSOR_SELECT_ROW
, m_rowLabelWin
); 
4880             // starting to drag-resize a row 
4882             if ( CanDragRowSize() ) 
4883                 ChangeCursorMode(WXGRID_CURSOR_RESIZE_ROW
, m_rowLabelWin
); 
4888     // ------------ Left double click 
4890     else if (event
.LeftDClick() ) 
4892         int row 
= YToEdgeOfRow(y
); 
4897                  !SendEvent( wxEVT_GRID_LABEL_LEFT_DCLICK
, row
, -1, event 
) ) 
4899                 // no default action at the moment 
4904             // adjust row height depending on label text 
4905             AutoSizeRowLabelSize( row 
); 
4907             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
); 
4913     // ------------ Left button released 
4915     else if ( event
.LeftUp() ) 
4917         if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_ROW 
) 
4919             DoEndDragResizeRow(); 
4921             // Note: we are ending the event *after* doing 
4922             // default processing in this case 
4924             SendEvent( wxEVT_GRID_ROW_SIZE
, m_dragRowOrCol
, -1, event 
); 
4927         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_rowLabelWin
); 
4932     // ------------ Right button down 
4934     else if ( event
.RightDown() ) 
4938             !SendEvent( wxEVT_GRID_LABEL_RIGHT_CLICK
, row
, -1, event 
) ) 
4940             // no default action at the moment 
4945     // ------------ Right double click 
4947     else if ( event
.RightDClick() ) 
4951              !SendEvent( wxEVT_GRID_LABEL_RIGHT_DCLICK
, row
, -1, event 
) ) 
4953             // no default action at the moment 
4958     // ------------ No buttons down and mouse moving 
4960     else if ( event
.Moving() ) 
4962         m_dragRowOrCol 
= YToEdgeOfRow( y 
); 
4963         if ( m_dragRowOrCol 
>= 0 ) 
4965             if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
4967                 // don't capture the mouse yet 
4968                 if ( CanDragRowSize() ) 
4969                     ChangeCursorMode(WXGRID_CURSOR_RESIZE_ROW
, m_rowLabelWin
, FALSE
); 
4972         else if ( m_cursorMode 
!= WXGRID_CURSOR_SELECT_CELL 
) 
4974             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_rowLabelWin
, FALSE
); 
4980 void wxGrid::ProcessColLabelMouseEvent( wxMouseEvent
& event 
) 
4983     wxPoint 
pos( event
.GetPosition() ); 
4984     CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y 
); 
4986     if ( event
.Dragging() ) 
4990             m_isDragging 
= TRUE
; 
4991             m_colLabelWin
->CaptureMouse(); 
4994         if ( event
.LeftIsDown() ) 
4996             switch( m_cursorMode 
) 
4998                 case WXGRID_CURSOR_RESIZE_COL
: 
5000                     int cw
, ch
, dummy
, top
; 
5001                     m_gridWin
->GetClientSize( &cw
, &ch 
); 
5002                     CalcUnscrolledPosition( 0, 0, &dummy
, &top 
); 
5004                     wxClientDC 
dc( m_gridWin 
); 
5007                     x 
= wxMax( x
, GetColLeft(m_dragRowOrCol
) + 
5008                                   GetColMinimalWidth(m_dragRowOrCol
)); 
5009                     dc
.SetLogicalFunction(wxINVERT
); 
5010                     if ( m_dragLastPos 
>= 0 ) 
5012                         dc
.DrawLine( m_dragLastPos
, top
, m_dragLastPos
, top
+ch 
); 
5014                     dc
.DrawLine( x
, top
, x
, top
+ch 
); 
5019                 case WXGRID_CURSOR_SELECT_COL
: 
5020                     if ( (col 
= XToCol( x 
)) >= 0 ) 
5024                             m_selection
->SelectCol( col
, 
5025                                                     event
.ControlDown(), 
5032                 // default label to suppress warnings about "enumeration value 
5033                 // 'xxx' not handled in switch 
5041     if ( m_isDragging 
&& (event
.Entering() || event
.Leaving()) ) 
5046         if (m_colLabelWin
->HasCapture()) m_colLabelWin
->ReleaseMouse(); 
5047         m_isDragging 
= FALSE
; 
5050     // ------------ Entering or leaving the window 
5052     if ( event
.Entering() || event
.Leaving() ) 
5054         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
); 
5058     // ------------ Left button pressed 
5060     else if ( event
.LeftDown() ) 
5062         // don't send a label click event for a hit on the 
5063         // edge of the col label - this is probably the user 
5064         // wanting to resize the col 
5066         if ( XToEdgeOfCol(x
) < 0 ) 
5070                  !SendEvent( wxEVT_GRID_LABEL_LEFT_CLICK
, -1, col
, event 
) ) 
5072                 if ( !event
.ShiftDown() && !event
.ControlDown() ) 
5076                     if ( event
.ShiftDown() ) 
5078                         m_selection
->SelectBlock( 0, 
5079                                                   m_currentCellCoords
.GetCol(), 
5080                                                   GetNumberRows() - 1, col
, 
5081                                                   event
.ControlDown(), 
5088                         m_selection
->SelectCol( col
, 
5089                                                 event
.ControlDown(), 
5096                 ChangeCursorMode(WXGRID_CURSOR_SELECT_COL
, m_colLabelWin
); 
5101             // starting to drag-resize a col 
5103             if ( CanDragColSize() ) 
5104                 ChangeCursorMode(WXGRID_CURSOR_RESIZE_COL
, m_colLabelWin
); 
5109     // ------------ Left double click 
5111     if ( event
.LeftDClick() ) 
5113         int col 
= XToEdgeOfCol(x
); 
5118                  ! SendEvent( wxEVT_GRID_LABEL_LEFT_DCLICK
, -1, col
, event 
) ) 
5120                 // no default action at the moment 
5125             // adjust column width depending on label text 
5126             AutoSizeColLabelSize( col 
); 
5128             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
); 
5134     // ------------ Left button released 
5136     else if ( event
.LeftUp() ) 
5138         if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_COL 
) 
5140             DoEndDragResizeCol(); 
5142             // Note: we are ending the event *after* doing 
5143             // default processing in this case 
5145             SendEvent( wxEVT_GRID_COL_SIZE
, -1, m_dragRowOrCol
, event 
); 
5148         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
); 
5153     // ------------ Right button down 
5155     else if ( event
.RightDown() ) 
5159              !SendEvent( wxEVT_GRID_LABEL_RIGHT_CLICK
, -1, col
, event 
) ) 
5161             // no default action at the moment 
5166     // ------------ Right double click 
5168     else if ( event
.RightDClick() ) 
5172              !SendEvent( wxEVT_GRID_LABEL_RIGHT_DCLICK
, -1, col
, event 
) ) 
5174             // no default action at the moment 
5179     // ------------ No buttons down and mouse moving 
5181     else if ( event
.Moving() ) 
5183         m_dragRowOrCol 
= XToEdgeOfCol( x 
); 
5184         if ( m_dragRowOrCol 
>= 0 ) 
5186             if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
5188                 // don't capture the cursor yet 
5189                 if ( CanDragColSize() ) 
5190                     ChangeCursorMode(WXGRID_CURSOR_RESIZE_COL
, m_colLabelWin
, FALSE
); 
5193         else if ( m_cursorMode 
!= WXGRID_CURSOR_SELECT_CELL 
) 
5195             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
, FALSE
); 
5201 void wxGrid::ProcessCornerLabelMouseEvent( wxMouseEvent
& event 
) 
5203     if ( event
.LeftDown() ) 
5205         // indicate corner label by having both row and 
5208         if ( !SendEvent( wxEVT_GRID_LABEL_LEFT_CLICK
, -1, -1, event 
) ) 
5214     else if ( event
.LeftDClick() ) 
5216         SendEvent( wxEVT_GRID_LABEL_LEFT_DCLICK
, -1, -1, event 
); 
5219     else if ( event
.RightDown() ) 
5221         if ( !SendEvent( wxEVT_GRID_LABEL_RIGHT_CLICK
, -1, -1, event 
) ) 
5223             // no default action at the moment 
5227     else if ( event
.RightDClick() ) 
5229         if ( !SendEvent( wxEVT_GRID_LABEL_RIGHT_DCLICK
, -1, -1, event 
) ) 
5231             // no default action at the moment 
5236 void wxGrid::ChangeCursorMode(CursorMode mode
, 
5241     static const wxChar 
*cursorModes
[] = 
5250     wxLogTrace(_T("grid"), 
5251                _T("wxGrid cursor mode (mouse capture for %s): %s -> %s"), 
5252                win 
== m_colLabelWin 
? _T("colLabelWin") 
5253                                     : win 
? _T("rowLabelWin") 
5255                cursorModes
[m_cursorMode
], cursorModes
[mode
]); 
5256 #endif // __WXDEBUG__ 
5258     if ( mode 
== m_cursorMode 
&& 
5259          win 
== m_winCapture 
&& 
5260          captureMouse 
== (m_winCapture 
!= NULL
)) 
5265         // by default use the grid itself 
5271         if (m_winCapture
->HasCapture()) m_winCapture
->ReleaseMouse(); 
5272         m_winCapture 
= (wxWindow 
*)NULL
; 
5275     m_cursorMode 
= mode
; 
5277     switch ( m_cursorMode 
) 
5279         case WXGRID_CURSOR_RESIZE_ROW
: 
5280             win
->SetCursor( m_rowResizeCursor 
); 
5283         case WXGRID_CURSOR_RESIZE_COL
: 
5284             win
->SetCursor( m_colResizeCursor 
); 
5288             win
->SetCursor( *wxSTANDARD_CURSOR 
); 
5291     // we need to capture mouse when resizing 
5292     bool resize 
= m_cursorMode 
== WXGRID_CURSOR_RESIZE_ROW 
|| 
5293                   m_cursorMode 
== WXGRID_CURSOR_RESIZE_COL
; 
5295     if ( captureMouse 
&& resize 
) 
5297         win
->CaptureMouse(); 
5302 void wxGrid::ProcessGridCellMouseEvent( wxMouseEvent
& event 
) 
5305     wxPoint 
pos( event
.GetPosition() ); 
5306     CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y 
); 
5308     wxGridCellCoords coords
; 
5309     XYToCell( x
, y
, coords 
); 
5311     int cell_rows
, cell_cols
; 
5312     GetCellSize( coords
.GetRow(), coords
.GetCol(), &cell_rows
, &cell_cols 
); 
5313     if ((cell_rows 
< 0) || (cell_cols 
< 0)) 
5315         coords
.SetRow(coords
.GetRow() + cell_rows
); 
5316         coords
.SetCol(coords
.GetCol() + cell_cols
); 
5319     if ( event
.Dragging() ) 
5321         //wxLogDebug("pos(%d, %d) coords(%d, %d)", pos.x, pos.y, coords.GetRow(), coords.GetCol()); 
5323         // Don't start doing anything until the mouse has been drug at 
5324         // least 3 pixels in any direction... 
5327             if (m_startDragPos 
== wxDefaultPosition
) 
5329                 m_startDragPos 
= pos
; 
5332             if (abs(m_startDragPos
.x 
- pos
.x
) < 4 && abs(m_startDragPos
.y 
- pos
.y
) < 4) 
5336         m_isDragging 
= TRUE
; 
5337         if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
5339             // Hide the edit control, so it 
5340             // won't interfer with drag-shrinking. 
5341             if ( IsCellEditControlShown() ) 
5343                 HideCellEditControl(); 
5344                 SaveEditControlValue(); 
5347             // Have we captured the mouse yet? 
5350                 m_winCapture 
= m_gridWin
; 
5351                 m_winCapture
->CaptureMouse(); 
5354             if ( coords 
!= wxGridNoCellCoords 
) 
5356                 if ( event
.ControlDown() ) 
5358                     if ( m_selectingKeyboard 
== wxGridNoCellCoords
) 
5359                         m_selectingKeyboard 
= coords
; 
5360                     HighlightBlock ( m_selectingKeyboard
, coords 
); 
5364                     if ( !IsSelection() ) 
5366                         HighlightBlock( coords
, coords 
); 
5370                         HighlightBlock( m_currentCellCoords
, coords 
); 
5374                 if (! IsVisible(coords
)) 
5376                     MakeCellVisible(coords
); 
5377                     // TODO: need to introduce a delay or something here.  The 
5378                     // scrolling is way to fast, at least on MSW - also on GTK. 
5382         else if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_ROW 
) 
5384             int cw
, ch
, left
, dummy
; 
5385             m_gridWin
->GetClientSize( &cw
, &ch 
); 
5386             CalcUnscrolledPosition( 0, 0, &left
, &dummy 
); 
5388             wxClientDC 
dc( m_gridWin 
); 
5390             y 
= wxMax( y
, GetRowTop(m_dragRowOrCol
) + 
5391                           GetRowMinimalHeight(m_dragRowOrCol
) ); 
5392             dc
.SetLogicalFunction(wxINVERT
); 
5393             if ( m_dragLastPos 
>= 0 ) 
5395                 dc
.DrawLine( left
, m_dragLastPos
, left
+cw
, m_dragLastPos 
); 
5397             dc
.DrawLine( left
, y
, left
+cw
, y 
); 
5400         else if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_COL 
) 
5402             int cw
, ch
, dummy
, top
; 
5403             m_gridWin
->GetClientSize( &cw
, &ch 
); 
5404             CalcUnscrolledPosition( 0, 0, &dummy
, &top 
); 
5406             wxClientDC 
dc( m_gridWin 
); 
5408             x 
= wxMax( x
, GetColLeft(m_dragRowOrCol
) + 
5409                           GetColMinimalWidth(m_dragRowOrCol
) ); 
5410             dc
.SetLogicalFunction(wxINVERT
); 
5411             if ( m_dragLastPos 
>= 0 ) 
5413                 dc
.DrawLine( m_dragLastPos
, top
, m_dragLastPos
, top
+ch 
); 
5415             dc
.DrawLine( x
, top
, x
, top
+ch 
); 
5422     m_isDragging 
= FALSE
; 
5423     m_startDragPos 
= wxDefaultPosition
; 
5425     // VZ: if we do this, the mode is reset to WXGRID_CURSOR_SELECT_CELL 
5426     //     immediately after it becomes WXGRID_CURSOR_RESIZE_ROW/COL under 
5429     if ( event
.Entering() || event
.Leaving() ) 
5431         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5432         m_gridWin
->SetCursor( *wxSTANDARD_CURSOR 
); 
5437     // ------------ Left button pressed 
5439     if ( event
.LeftDown() && coords 
!= wxGridNoCellCoords 
) 
5441         if ( !SendEvent( wxEVT_GRID_CELL_LEFT_CLICK
, 
5446             if ( !event
.ControlDown() ) 
5448             if ( event
.ShiftDown() ) 
5452                     m_selection
->SelectBlock( m_currentCellCoords
.GetRow(), 
5453                                               m_currentCellCoords
.GetCol(), 
5456                                               event
.ControlDown(), 
5462             else if ( XToEdgeOfCol(x
) < 0  && 
5463                       YToEdgeOfRow(y
) < 0 ) 
5465                 DisableCellEditControl(); 
5466                 MakeCellVisible( coords 
); 
5468                 if ( event
.ControlDown() ) 
5472                         m_selection
->ToggleCellSelection( coords
.GetRow(), 
5474                                                           event
.ControlDown(), 
5479                     m_selectingTopLeft 
= wxGridNoCellCoords
; 
5480                     m_selectingBottomRight 
= wxGridNoCellCoords
; 
5481                     m_selectingKeyboard 
= coords
; 
5485                     m_waitForSlowClick 
= m_currentCellCoords 
== coords 
&& coords 
!= wxGridNoCellCoords
; 
5486                     SetCurrentCell( coords 
); 
5489                         if ( m_selection
->GetSelectionMode() != 
5490                                 wxGrid::wxGridSelectCells 
) 
5492                             HighlightBlock( coords
, coords 
); 
5501     // ------------ Left double click 
5503     else if ( event
.LeftDClick() && coords 
!= wxGridNoCellCoords 
) 
5505         DisableCellEditControl(); 
5507         if ( XToEdgeOfCol(x
) < 0  &&  YToEdgeOfRow(y
) < 0 ) 
5509             SendEvent( wxEVT_GRID_CELL_LEFT_DCLICK
, 
5517     // ------------ Left button released 
5519     else if ( event
.LeftUp() ) 
5521         if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
5525                 if (m_winCapture
->HasCapture()) m_winCapture
->ReleaseMouse(); 
5526                 m_winCapture 
= NULL
; 
5529             if ( coords 
== m_currentCellCoords 
&& m_waitForSlowClick 
&& CanEnableCellControl()) 
5532                 EnableCellEditControl(); 
5534                 wxGridCellAttr
* attr 
= GetCellAttr(coords
); 
5535                 wxGridCellEditor 
*editor 
= attr
->GetEditor(this, coords
.GetRow(), coords
.GetCol()); 
5536                 editor
->StartingClick(); 
5540                 m_waitForSlowClick 
= FALSE
; 
5542             else if ( m_selectingTopLeft 
!= wxGridNoCellCoords 
&& 
5543                  m_selectingBottomRight 
!= wxGridNoCellCoords 
) 
5547                     m_selection
->SelectBlock( m_selectingTopLeft
.GetRow(), 
5548                                               m_selectingTopLeft
.GetCol(), 
5549                                               m_selectingBottomRight
.GetRow(), 
5550                                               m_selectingBottomRight
.GetCol(), 
5551                                               event
.ControlDown(), 
5557                 m_selectingTopLeft 
= wxGridNoCellCoords
; 
5558                 m_selectingBottomRight 
= wxGridNoCellCoords
; 
5560                 // Show the edit control, if it has been hidden for 
5562                 ShowCellEditControl(); 
5565         else if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_ROW 
) 
5567             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5568             DoEndDragResizeRow(); 
5570             // Note: we are ending the event *after* doing 
5571             // default processing in this case 
5573             SendEvent( wxEVT_GRID_ROW_SIZE
, m_dragRowOrCol
, -1, event 
); 
5575         else if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_COL 
) 
5577             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5578             DoEndDragResizeCol(); 
5580             // Note: we are ending the event *after* doing 
5581             // default processing in this case 
5583             SendEvent( wxEVT_GRID_COL_SIZE
, -1, m_dragRowOrCol
, event 
); 
5590     // ------------ Right button down 
5592     else if ( event
.RightDown() && coords 
!= wxGridNoCellCoords 
) 
5594         DisableCellEditControl(); 
5595         if ( !SendEvent( wxEVT_GRID_CELL_RIGHT_CLICK
, 
5600             // no default action at the moment 
5605     // ------------ Right double click 
5607     else if ( event
.RightDClick() && coords 
!= wxGridNoCellCoords 
) 
5609         DisableCellEditControl(); 
5610         if ( !SendEvent( wxEVT_GRID_CELL_RIGHT_DCLICK
, 
5615             // no default action at the moment 
5619     // ------------ Moving and no button action 
5621     else if ( event
.Moving() && !event
.IsButton() ) 
5623         if( coords
.GetRow() < 0 || coords
.GetCol() < 0 ) 
5625             // out of grid cell area 
5626             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5630         int dragRow 
= YToEdgeOfRow( y 
); 
5631         int dragCol 
= XToEdgeOfCol( x 
); 
5633         // Dragging on the corner of a cell to resize in both 
5634         // directions is not implemented yet... 
5636         if ( dragRow 
>= 0  &&  dragCol 
>= 0 ) 
5638             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5644             m_dragRowOrCol 
= dragRow
; 
5646             if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
5648                 if ( CanDragRowSize() && CanDragGridSize() ) 
5649                     ChangeCursorMode(WXGRID_CURSOR_RESIZE_ROW
); 
5654                 m_dragRowOrCol 
= dragCol
; 
5662             m_dragRowOrCol 
= dragCol
; 
5664             if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
5666                 if ( CanDragColSize() && CanDragGridSize() ) 
5667                     ChangeCursorMode(WXGRID_CURSOR_RESIZE_COL
); 
5673         // Neither on a row or col edge 
5675         if ( m_cursorMode 
!= WXGRID_CURSOR_SELECT_CELL 
) 
5677             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5683 void wxGrid::DoEndDragResizeRow() 
5685     if ( m_dragLastPos 
>= 0 ) 
5687         // erase the last line and resize the row 
5689         int cw
, ch
, left
, dummy
; 
5690         m_gridWin
->GetClientSize( &cw
, &ch 
); 
5691         CalcUnscrolledPosition( 0, 0, &left
, &dummy 
); 
5693         wxClientDC 
dc( m_gridWin 
); 
5695         dc
.SetLogicalFunction( wxINVERT 
); 
5696         dc
.DrawLine( left
, m_dragLastPos
, left
+cw
, m_dragLastPos 
); 
5697         HideCellEditControl(); 
5698         SaveEditControlValue(); 
5700         int rowTop 
= GetRowTop(m_dragRowOrCol
); 
5701         SetRowSize( m_dragRowOrCol
, 
5702                     wxMax( m_dragLastPos 
- rowTop
, m_minAcceptableRowHeight 
) ); 
5704         if ( !GetBatchCount() ) 
5706             // Only needed to get the correct rect.y: 
5707             wxRect 
rect ( CellToRect( m_dragRowOrCol
, 0 ) ); 
5709             CalcScrolledPosition(0, rect
.y
, &dummy
, &rect
.y
); 
5710             rect
.width 
= m_rowLabelWidth
; 
5711             rect
.height 
= ch 
- rect
.y
; 
5712             m_rowLabelWin
->Refresh( TRUE
, &rect 
); 
5714             // if there is a multicell block, paint all of it 
5717                 int i
, cell_rows
, cell_cols
, subtract_rows 
= 0; 
5718                 int leftCol 
= XToCol(left
); 
5719                 int rightCol 
= internalXToCol(left
+cw
); 
5722                     for (i
=leftCol
; i
<rightCol
; i
++) 
5724                         GetCellSize(m_dragRowOrCol
, i
, &cell_rows
, &cell_cols
); 
5725                         if (cell_rows 
< subtract_rows
) 
5726                             subtract_rows 
= cell_rows
; 
5728                     rect
.y 
= GetRowTop(m_dragRowOrCol 
+ subtract_rows
); 
5729                     CalcScrolledPosition(0, rect
.y
, &dummy
, &rect
.y
); 
5730                     rect
.height 
= ch 
- rect
.y
; 
5733             m_gridWin
->Refresh( FALSE
, &rect 
); 
5736         ShowCellEditControl(); 
5741 void wxGrid::DoEndDragResizeCol() 
5743     if ( m_dragLastPos 
>= 0 ) 
5745         // erase the last line and resize the col 
5747         int cw
, ch
, dummy
, top
; 
5748         m_gridWin
->GetClientSize( &cw
, &ch 
); 
5749         CalcUnscrolledPosition( 0, 0, &dummy
, &top 
); 
5751         wxClientDC 
dc( m_gridWin 
); 
5753         dc
.SetLogicalFunction( wxINVERT 
); 
5754         dc
.DrawLine( m_dragLastPos
, top
, m_dragLastPos
, top
+ch 
); 
5755         HideCellEditControl(); 
5756         SaveEditControlValue(); 
5758         int colLeft 
= GetColLeft(m_dragRowOrCol
); 
5759         SetColSize( m_dragRowOrCol
, 
5760                     wxMax( m_dragLastPos 
- colLeft
, 
5761                            GetColMinimalWidth(m_dragRowOrCol
) ) ); 
5763         if ( !GetBatchCount() ) 
5765             // Only needed to get the correct rect.x: 
5766             wxRect 
rect ( CellToRect( 0, m_dragRowOrCol 
) ); 
5768             CalcScrolledPosition(rect
.x
, 0, &rect
.x
, &dummy
); 
5769             rect
.width 
= cw 
- rect
.x
; 
5770             rect
.height 
= m_colLabelHeight
; 
5771             m_colLabelWin
->Refresh( TRUE
, &rect 
); 
5773             // if there is a multicell block, paint all of it 
5776                 int i
, cell_rows
, cell_cols
, subtract_cols 
= 0; 
5777                 int topRow 
= YToRow(top
); 
5778                 int bottomRow 
= internalYToRow(top
+cw
); 
5781                     for (i
=topRow
; i
<bottomRow
; i
++) 
5783                         GetCellSize(i
, m_dragRowOrCol
, &cell_rows
, &cell_cols
); 
5784                         if (cell_cols 
< subtract_cols
) 
5785                             subtract_cols 
= cell_cols
; 
5787                     rect
.x 
= GetColLeft(m_dragRowOrCol 
+ subtract_cols
); 
5788                     CalcScrolledPosition(rect
.x
, 0, &rect
.x
, &dummy
); 
5789                     rect
.width 
= cw 
- rect
.x
; 
5792             m_gridWin
->Refresh( FALSE
, &rect 
); 
5795         ShowCellEditControl(); 
5802 // ------ interaction with data model 
5804 bool wxGrid::ProcessTableMessage( wxGridTableMessage
& msg 
) 
5806     switch ( msg
.GetId() ) 
5808         case wxGRIDTABLE_REQUEST_VIEW_GET_VALUES
: 
5809             return GetModelValues(); 
5811         case wxGRIDTABLE_REQUEST_VIEW_SEND_VALUES
: 
5812             return SetModelValues(); 
5814         case wxGRIDTABLE_NOTIFY_ROWS_INSERTED
: 
5815         case wxGRIDTABLE_NOTIFY_ROWS_APPENDED
: 
5816         case wxGRIDTABLE_NOTIFY_ROWS_DELETED
: 
5817         case wxGRIDTABLE_NOTIFY_COLS_INSERTED
: 
5818         case wxGRIDTABLE_NOTIFY_COLS_APPENDED
: 
5819         case wxGRIDTABLE_NOTIFY_COLS_DELETED
: 
5820             return Redimension( msg 
); 
5829 // The behaviour of this function depends on the grid table class 
5830 // Clear() function.  For the default wxGridStringTable class the 
5831 // behavious is to replace all cell contents with wxEmptyString but 
5832 // not to change the number of rows or cols. 
5834 void wxGrid::ClearGrid() 
5838         if (IsCellEditControlEnabled()) 
5839             DisableCellEditControl(); 
5842         if ( !GetBatchCount() ) m_gridWin
->Refresh(); 
5847 bool wxGrid::InsertRows( int pos
, int numRows
, bool WXUNUSED(updateLabels
) ) 
5849     // TODO: something with updateLabels flag 
5853         wxFAIL_MSG( wxT("Called wxGrid::InsertRows() before calling CreateGrid()") ); 
5859         if (IsCellEditControlEnabled()) 
5860             DisableCellEditControl(); 
5862         bool done 
= m_table
->InsertRows( pos
, numRows 
); 
5865         // the table will have sent the results of the insert row 
5866         // operation to this view object as a grid table message 
5872 bool wxGrid::AppendRows( int numRows
, bool WXUNUSED(updateLabels
) ) 
5874     // TODO: something with updateLabels flag 
5878         wxFAIL_MSG( wxT("Called wxGrid::AppendRows() before calling CreateGrid()") ); 
5884         bool done 
= m_table 
&& m_table
->AppendRows( numRows 
); 
5886         // the table will have sent the results of the append row 
5887         // operation to this view object as a grid table message 
5893 bool wxGrid::DeleteRows( int pos
, int numRows
, bool WXUNUSED(updateLabels
) ) 
5895     // TODO: something with updateLabels flag 
5899         wxFAIL_MSG( wxT("Called wxGrid::DeleteRows() before calling CreateGrid()") ); 
5905         if (IsCellEditControlEnabled()) 
5906             DisableCellEditControl(); 
5908         bool done 
= m_table
->DeleteRows( pos
, numRows 
); 
5910         // the table will have sent the results of the delete row 
5911         // operation to this view object as a grid table message 
5917 bool wxGrid::InsertCols( int pos
, int numCols
, bool WXUNUSED(updateLabels
) ) 
5919     // TODO: something with updateLabels flag 
5923         wxFAIL_MSG( wxT("Called wxGrid::InsertCols() before calling CreateGrid()") ); 
5929         if (IsCellEditControlEnabled()) 
5930             DisableCellEditControl(); 
5932         bool done 
= m_table
->InsertCols( pos
, numCols 
); 
5934         // the table will have sent the results of the insert col 
5935         // operation to this view object as a grid table message 
5941 bool wxGrid::AppendCols( int numCols
, bool WXUNUSED(updateLabels
) ) 
5943     // TODO: something with updateLabels flag 
5947         wxFAIL_MSG( wxT("Called wxGrid::AppendCols() before calling CreateGrid()") ); 
5953         bool done 
= m_table
->AppendCols( numCols 
); 
5955         // the table will have sent the results of the append col 
5956         // operation to this view object as a grid table message 
5962 bool wxGrid::DeleteCols( int pos
, int numCols
, bool WXUNUSED(updateLabels
) ) 
5964     // TODO: something with updateLabels flag 
5968         wxFAIL_MSG( wxT("Called wxGrid::DeleteCols() before calling CreateGrid()") ); 
5974         if (IsCellEditControlEnabled()) 
5975             DisableCellEditControl(); 
5977         bool done 
= m_table
->DeleteCols( pos
, numCols 
); 
5979         // the table will have sent the results of the delete col 
5980         // operation to this view object as a grid table message 
5988 // ----- event handlers 
5991 // Generate a grid event based on a mouse event and 
5992 // return the result of ProcessEvent() 
5994 int wxGrid::SendEvent( const wxEventType type
, 
5996                         wxMouseEvent
& mouseEv 
) 
6001    if ( type 
== wxEVT_GRID_ROW_SIZE 
|| type 
== wxEVT_GRID_COL_SIZE 
) 
6003        int rowOrCol 
= (row 
== -1 ? col 
: row
); 
6005        wxGridSizeEvent 
gridEvt( GetId(), 
6009                mouseEv
.GetX() + GetRowLabelSize(), 
6010                mouseEv
.GetY() + GetColLabelSize(), 
6011                mouseEv
.ControlDown(), 
6012                mouseEv
.ShiftDown(), 
6014                mouseEv
.MetaDown() ); 
6016        claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
6017        vetoed 
= !gridEvt
.IsAllowed(); 
6019    else if ( type 
== wxEVT_GRID_RANGE_SELECT 
) 
6021        // Right now, it should _never_ end up here! 
6022        wxGridRangeSelectEvent 
gridEvt( GetId(), 
6026                m_selectingBottomRight
, 
6028                mouseEv
.ControlDown(), 
6029                mouseEv
.ShiftDown(), 
6031                mouseEv
.MetaDown() ); 
6033        claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
6034        vetoed 
= !gridEvt
.IsAllowed(); 
6038        wxGridEvent 
gridEvt( GetId(), 
6042                mouseEv
.GetX() + GetRowLabelSize(), 
6043                mouseEv
.GetY() + GetColLabelSize(), 
6045                mouseEv
.ControlDown(), 
6046                mouseEv
.ShiftDown(), 
6048                mouseEv
.MetaDown() ); 
6049        claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
6050        vetoed 
= !gridEvt
.IsAllowed(); 
6053    // A Veto'd event may not be `claimed' so test this first 
6054    if (vetoed
) return -1; 
6055    return claimed 
? 1 : 0; 
6059 // Generate a grid event of specified type and return the result 
6060 // of ProcessEvent(). 
6062 int wxGrid::SendEvent( const wxEventType type
, 
6068     if ( type 
== wxEVT_GRID_ROW_SIZE 
|| type 
== wxEVT_GRID_COL_SIZE 
) 
6070         int rowOrCol 
= (row 
== -1 ? col 
: row
); 
6072         wxGridSizeEvent 
gridEvt( GetId(), 
6077         claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
6078         vetoed  
= !gridEvt
.IsAllowed(); 
6082         wxGridEvent 
gridEvt( GetId(), 
6087         claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
6088         vetoed  
= !gridEvt
.IsAllowed(); 
6091     // A Veto'd event may not be `claimed' so test this first 
6092     if (vetoed
) return -1; 
6093     return claimed 
? 1 : 0; 
6097 void wxGrid::OnPaint( wxPaintEvent
& WXUNUSED(event
) ) 
6099     wxPaintDC 
dc(this);  // needed to prevent zillions of paint events on MSW 
6102 void wxGrid::Refresh(bool eraseb
, const wxRect
* rect
) 
6104     // Don't do anything if between Begin/EndBatch... 
6105     // EndBatch() will do all this on the last nested one anyway. 
6106     if (! GetBatchCount()) 
6108         // Refresh to get correct scrolled position: 
6109         wxScrolledWindow::Refresh(eraseb
,rect
); 
6113             int rect_x
, rect_y
, rectWidth
, rectHeight
; 
6114             int width_label
, width_cell
, height_label
, height_cell
; 
6117             //Copy rectangle can get scroll offsets.. 
6118             rect_x 
= rect
->GetX(); 
6119             rect_y 
= rect
->GetY(); 
6120             rectWidth 
= rect
->GetWidth(); 
6121             rectHeight 
= rect
->GetHeight(); 
6123             width_label 
= m_rowLabelWidth 
- rect_x
; 
6124             if (width_label 
> rectWidth
) width_label 
= rectWidth
; 
6126             height_label 
= m_colLabelHeight 
- rect_y
; 
6127             if (height_label 
> rectHeight
) height_label 
= rectHeight
; 
6129             if (rect_x 
> m_rowLabelWidth
) 
6131                 x 
= rect_x 
- m_rowLabelWidth
; 
6132                 width_cell 
= rectWidth
; 
6137                 width_cell 
= rectWidth 
- (m_rowLabelWidth 
- rect_x
); 
6140             if (rect_y 
> m_colLabelHeight
) 
6142                 y 
= rect_y 
- m_colLabelHeight
; 
6143                 height_cell 
= rectHeight
; 
6148                 height_cell 
= rectHeight 
- (m_colLabelHeight 
- rect_y
); 
6151             // Paint corner label part intersecting rect. 
6152             if ( width_label 
> 0 && height_label 
> 0 ) 
6154                 wxRect 
anotherrect(rect_x
, rect_y
, width_label
, height_label
); 
6155                 m_cornerLabelWin
->Refresh(eraseb
, &anotherrect
); 
6158             // Paint col labels part intersecting rect. 
6159             if ( width_cell 
> 0 && height_label 
> 0 ) 
6161                 wxRect 
anotherrect(x
, rect_y
, width_cell
, height_label
); 
6162                 m_colLabelWin
->Refresh(eraseb
, &anotherrect
); 
6165             // Paint row labels part intersecting rect. 
6166             if ( width_label 
> 0 && height_cell 
> 0 ) 
6168                 wxRect 
anotherrect(rect_x
, y
, width_label
, height_cell
); 
6169                 m_rowLabelWin
->Refresh(eraseb
, &anotherrect
); 
6172             // Paint cell area part intersecting rect. 
6173             if ( width_cell 
> 0 && height_cell 
> 0 ) 
6175                 wxRect 
anotherrect(x
, y
, width_cell
, height_cell
); 
6176                 m_gridWin
->Refresh(eraseb
, &anotherrect
); 
6181             m_cornerLabelWin
->Refresh(eraseb
, NULL
); 
6182             m_colLabelWin
->Refresh(eraseb
, NULL
); 
6183             m_rowLabelWin
->Refresh(eraseb
, NULL
); 
6184             m_gridWin
->Refresh(eraseb
, NULL
); 
6189 void wxGrid::OnSize( wxSizeEvent
& event 
) 
6191     // position the child windows 
6194     // don't call CalcDimensions() from here, the base class handles the size 
6200 void wxGrid::OnKeyDown( wxKeyEvent
& event 
) 
6202     if ( m_inOnKeyDown 
) 
6204         // shouldn't be here - we are going round in circles... 
6206         wxFAIL_MSG( wxT("wxGrid::OnKeyDown called while already active") ); 
6209     m_inOnKeyDown 
= TRUE
; 
6211     // propagate the event up and see if it gets processed 
6213     wxWindow 
*parent 
= GetParent(); 
6214     wxKeyEvent 
keyEvt( event 
); 
6215     keyEvt
.SetEventObject( parent 
); 
6217     if ( !parent
->GetEventHandler()->ProcessEvent( keyEvt 
) ) 
6220         // try local handlers 
6222         switch ( event
.GetKeyCode() ) 
6225                 if ( event
.ControlDown() ) 
6227                     MoveCursorUpBlock( event
.ShiftDown() ); 
6231                     MoveCursorUp( event
.ShiftDown() ); 
6236                 if ( event
.ControlDown() ) 
6238                     MoveCursorDownBlock( event
.ShiftDown() ); 
6242                     MoveCursorDown( event
.ShiftDown() ); 
6247                 if ( event
.ControlDown() ) 
6249                     MoveCursorLeftBlock( event
.ShiftDown() ); 
6253                     MoveCursorLeft( event
.ShiftDown() ); 
6258                 if ( event
.ControlDown() ) 
6260                     MoveCursorRightBlock( event
.ShiftDown() ); 
6264                     MoveCursorRight( event
.ShiftDown() ); 
6269             case WXK_NUMPAD_ENTER
: 
6270                 if ( event
.ControlDown() ) 
6272                     event
.Skip();  // to let the edit control have the return 
6276                     if ( GetGridCursorRow() < GetNumberRows()-1 ) 
6278                         MoveCursorDown( event
.ShiftDown() ); 
6282                         // at the bottom of a column 
6283                         HideCellEditControl(); 
6284                         SaveEditControlValue(); 
6294                 if (event
.ShiftDown()) 
6296                     if ( GetGridCursorCol() > 0 ) 
6298                         MoveCursorLeft( FALSE 
); 
6303                         HideCellEditControl(); 
6304                         SaveEditControlValue(); 
6309                     if ( GetGridCursorCol() < GetNumberCols()-1 ) 
6311                         MoveCursorRight( FALSE 
); 
6316                         HideCellEditControl(); 
6317                         SaveEditControlValue(); 
6323                 if ( event
.ControlDown() ) 
6325                     MakeCellVisible( 0, 0 ); 
6326                     SetCurrentCell( 0, 0 ); 
6335                 if ( event
.ControlDown() ) 
6337                     MakeCellVisible( m_numRows
-1, m_numCols
-1 ); 
6338                     SetCurrentCell( m_numRows
-1, m_numCols
-1 ); 
6355                 if ( event
.ControlDown() ) 
6359                         m_selection
->ToggleCellSelection( m_currentCellCoords
.GetRow(), 
6360                                                           m_currentCellCoords
.GetCol(), 
6361                                                           event
.ControlDown(), 
6368                 if ( !IsEditable() ) 
6370                     MoveCursorRight( FALSE 
); 
6373                 // Otherwise fall through to default 
6376                 // is it possible to edit the current cell at all? 
6377                 if ( !IsCellEditControlEnabled() && CanEnableCellControl() ) 
6379                     // yes, now check whether the cells editor accepts the key 
6380                     int row 
= m_currentCellCoords
.GetRow(); 
6381                     int col 
= m_currentCellCoords
.GetCol(); 
6382                     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
6383                     wxGridCellEditor 
*editor 
= attr
->GetEditor(this, row
, col
); 
6385                     // <F2> is special and will always start editing, for 
6386                     // other keys - ask the editor itself 
6387                     if ( (event
.GetKeyCode() == WXK_F2 
&& !event
.HasModifiers()) 
6388                          || editor
->IsAcceptedKey(event
) ) 
6390                         // ensure cell is visble 
6391                         MakeCellVisible(row
, col
); 
6392                         EnableCellEditControl(); 
6394                         // a problem can arise if the cell is not completely 
6395                         // visible (even after calling MakeCellVisible the 
6396                         // control is not created and calling StartingKey will 
6398                         if( editor
->IsCreated() && m_cellEditCtrlEnabled 
) editor
->StartingKey(event
); 
6410                     // let others process char events with modifiers or all 
6411                     // char events for readonly cells 
6418     m_inOnKeyDown 
= FALSE
; 
6421 void wxGrid::OnKeyUp( wxKeyEvent
& event 
) 
6423     // try local handlers 
6425     if ( event
.GetKeyCode() == WXK_SHIFT 
) 
6427         if ( m_selectingTopLeft 
!= wxGridNoCellCoords 
&& 
6428              m_selectingBottomRight 
!= wxGridNoCellCoords 
) 
6432                 m_selection
->SelectBlock( m_selectingTopLeft
.GetRow(), 
6433                                           m_selectingTopLeft
.GetCol(), 
6434                                           m_selectingBottomRight
.GetRow(), 
6435                                           m_selectingBottomRight
.GetCol(), 
6436                                           event
.ControlDown(), 
6443         m_selectingTopLeft 
= wxGridNoCellCoords
; 
6444         m_selectingBottomRight 
= wxGridNoCellCoords
; 
6445         m_selectingKeyboard 
= wxGridNoCellCoords
; 
6449 void wxGrid::OnEraseBackground(wxEraseEvent
&) 
6453 void wxGrid::SetCurrentCell( const wxGridCellCoords
& coords 
) 
6455     if ( SendEvent( wxEVT_GRID_SELECT_CELL
, coords
.GetRow(), coords
.GetCol() ) ) 
6457         // the event has been intercepted - do nothing 
6461     wxClientDC 
dc(m_gridWin
); 
6464     if ( m_currentCellCoords 
!= wxGridNoCellCoords 
) 
6466         HideCellEditControl(); 
6467         DisableCellEditControl(); 
6469         if ( IsVisible( m_currentCellCoords
, FALSE 
) ) 
6472             r 
= BlockToDeviceRect(m_currentCellCoords
, m_currentCellCoords
); 
6473             if ( !m_gridLinesEnabled 
) 
6481             wxGridCellCoordsArray cells 
= CalcCellsExposed( r 
); 
6483             // Otherwise refresh redraws the highlight! 
6484             m_currentCellCoords 
= coords
; 
6486             DrawGridCellArea(dc
,cells
); 
6487             DrawAllGridLines( dc
, r 
); 
6491     m_currentCellCoords 
= coords
; 
6493     wxGridCellAttr
* attr 
= GetCellAttr(coords
); 
6494     DrawCellHighlight(dc
, attr
); 
6499 void wxGrid::HighlightBlock( int topRow
, int leftCol
, int bottomRow
, int rightCol 
) 
6502     wxGridCellCoords updateTopLeft
, updateBottomRight
; 
6506         if ( m_selection
->GetSelectionMode() == wxGrid::wxGridSelectRows 
) 
6509             rightCol 
= GetNumberCols() - 1; 
6511         else if ( m_selection
->GetSelectionMode() == wxGrid::wxGridSelectColumns 
) 
6514             bottomRow 
= GetNumberRows() - 1; 
6518     if ( topRow 
> bottomRow 
) 
6525     if ( leftCol 
> rightCol 
) 
6532     updateTopLeft 
= wxGridCellCoords( topRow
, leftCol 
); 
6533     updateBottomRight 
= wxGridCellCoords( bottomRow
, rightCol 
); 
6535     // First the case that we selected a completely new area 
6536     if ( m_selectingTopLeft 
== wxGridNoCellCoords 
|| 
6537          m_selectingBottomRight 
== wxGridNoCellCoords 
) 
6540         rect 
= BlockToDeviceRect( wxGridCellCoords ( topRow
, leftCol 
), 
6541                                   wxGridCellCoords ( bottomRow
, rightCol 
) ); 
6542         m_gridWin
->Refresh( FALSE
, &rect 
); 
6544     // Now handle changing an existing selection area. 
6545     else if ( m_selectingTopLeft 
!= updateTopLeft 
|| 
6546               m_selectingBottomRight 
!= updateBottomRight 
) 
6548         // Compute two optimal update rectangles: 
6549         // Either one rectangle is a real subset of the 
6550         // other, or they are (almost) disjoint! 
6552         bool    need_refresh
[4]; 
6556         need_refresh
[3] = FALSE
; 
6559         // Store intermediate values 
6560         wxCoord oldLeft   
= m_selectingTopLeft
.GetCol(); 
6561         wxCoord oldTop    
= m_selectingTopLeft
.GetRow(); 
6562         wxCoord oldRight  
= m_selectingBottomRight
.GetCol(); 
6563         wxCoord oldBottom 
= m_selectingBottomRight
.GetRow(); 
6565         // Determine the outer/inner coordinates. 
6566         if (oldLeft 
> leftCol
) 
6572         if (oldTop 
> topRow 
) 
6578         if (oldRight 
< rightCol 
) 
6581             oldRight 
= rightCol
; 
6584         if (oldBottom 
< bottomRow
) 
6587             oldBottom 
= bottomRow
; 
6591         // Now, either the stuff marked old is the outer 
6592         // rectangle or we don't have a situation where one 
6593         // is contained in the other. 
6595         if ( oldLeft 
< leftCol 
) 
6597             // Refresh the newly selected or deselected 
6598             // area to the left of the old or new selection. 
6599             need_refresh
[0] = TRUE
; 
6600             rect
[0] = BlockToDeviceRect( wxGridCellCoords ( oldTop
, 
6602                                          wxGridCellCoords ( oldBottom
, 
6606         if ( oldTop  
< topRow 
) 
6608             // Refresh the newly selected or deselected 
6609             // area above the old or new selection. 
6610             need_refresh
[1] = TRUE
; 
6611             rect
[1] = BlockToDeviceRect( wxGridCellCoords ( oldTop
, 
6613                                          wxGridCellCoords ( topRow 
- 1, 
6617         if ( oldRight 
> rightCol 
) 
6619             // Refresh the newly selected or deselected 
6620             // area to the right of the old or new selection. 
6621             need_refresh
[2] = TRUE
; 
6622             rect
[2] = BlockToDeviceRect( wxGridCellCoords ( oldTop
, 
6624                                          wxGridCellCoords ( oldBottom
, 
6628         if ( oldBottom 
> bottomRow 
) 
6630             // Refresh the newly selected or deselected 
6631             // area below the old or new selection. 
6632             need_refresh
[3] = TRUE
; 
6633             rect
[3] = BlockToDeviceRect( wxGridCellCoords ( bottomRow 
+ 1, 
6635                                          wxGridCellCoords ( oldBottom
, 
6639         // various Refresh() calls 
6640         for (i 
= 0; i 
< 4; i
++ ) 
6641             if ( need_refresh
[i
] && rect
[i
] != wxGridNoCellRect 
) 
6642                 m_gridWin
->Refresh( FALSE
, &(rect
[i
]) ); 
6645     m_selectingTopLeft 
= updateTopLeft
; 
6646     m_selectingBottomRight 
= updateBottomRight
; 
6650 // ------ functions to get/send data (see also public functions) 
6653 bool wxGrid::GetModelValues() 
6655     // Hide the editor, so it won't hide a changed value. 
6656     HideCellEditControl(); 
6660         // all we need to do is repaint the grid 
6662         m_gridWin
->Refresh(); 
6670 bool wxGrid::SetModelValues() 
6674     // Disable the editor, so it won't hide a changed value. 
6675     // Do we also want to save the current value of the editor first? 
6677     DisableCellEditControl(); 
6681         for ( row 
= 0;  row 
< m_numRows
;  row
++ ) 
6683             for ( col 
= 0;  col 
< m_numCols
;  col
++ ) 
6685                 m_table
->SetValue( row
, col
, GetCellValue(row
, col
) ); 
6697 // Note - this function only draws cells that are in the list of 
6698 // exposed cells (usually set from the update region by 
6699 // CalcExposedCells) 
6701 void wxGrid::DrawGridCellArea( wxDC
& dc
, const wxGridCellCoordsArray
& cells 
) 
6703     if ( !m_numRows 
|| !m_numCols 
) return; 
6705     int i
, numCells 
= cells
.GetCount(); 
6706     int row
, col
, cell_rows
, cell_cols
; 
6707     wxGridCellCoordsArray redrawCells
; 
6709     for ( i 
= numCells
-1; i 
>= 0;  i
-- ) 
6711         row 
= cells
[i
].GetRow(); 
6712         col 
= cells
[i
].GetCol(); 
6713         GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
6715         // If this cell is part of a multicell block, find owner for repaint 
6716         if ( cell_rows 
<= 0 || cell_cols 
<= 0 ) 
6718             wxGridCellCoords 
cell(row
+cell_rows
, col
+cell_cols
); 
6719             bool marked 
= FALSE
; 
6720             for ( int j 
= 0;  j 
< numCells
;  j
++ ) 
6722                 if ( cell 
== cells
[j
] ) 
6730                 int count 
= redrawCells
.GetCount(); 
6731                 for (int j 
= 0; j 
< count
; j
++) 
6733                     if ( cell 
== redrawCells
[j
] ) 
6739                 if (!marked
) redrawCells
.Add( cell 
); 
6741             continue; // don't bother drawing this cell 
6744         // If this cell is empty, find cell to left that might want to overflow 
6745         if (m_table 
&& m_table
->IsEmptyCell(row
, col
)) 
6747             for ( int l 
= 0; l 
< cell_rows
; l
++ ) 
6749                 // find a cell in this row to left alreay marked for repaint 
6751                 for (int k 
= 0; k 
< int(redrawCells
.GetCount()); k
++) 
6752                     if ((redrawCells
[k
].GetCol() < left
) && 
6753                         (redrawCells
[k
].GetRow() == row
)) 
6754                         left
=redrawCells
[k
].GetCol(); 
6756                 if (left 
== col
) left 
= 0; // oh well 
6758                 for (int j 
= col
-1; j 
>= left
; j
--) 
6760                     if (!m_table
->IsEmptyCell(row
+l
, j
)) 
6762                         if (GetCellOverflow(row
+l
, j
)) 
6764                             wxGridCellCoords 
cell(row
+l
, j
); 
6765                             bool marked 
= FALSE
; 
6767                             for (int k 
= 0; k 
< numCells
; k
++) 
6769                                 if ( cell 
== cells
[k
] ) 
6777                                 int count 
= redrawCells
.GetCount(); 
6778                                 for (int k 
= 0; k 
< count
; k
++) 
6780                                     if ( cell 
== redrawCells
[k
] ) 
6786                                 if (!marked
) redrawCells
.Add( cell 
); 
6794         DrawCell( dc
, cells
[i
] ); 
6797     numCells 
= redrawCells
.GetCount(); 
6799     for ( i 
= numCells 
- 1; i 
>= 0;  i
-- ) 
6801         DrawCell( dc
, redrawCells
[i
] ); 
6806 void wxGrid::DrawGridSpace( wxDC
& dc 
) 
6809   m_gridWin
->GetClientSize( &cw
, &ch 
); 
6812   CalcUnscrolledPosition( cw
, ch
, &right
, &bottom 
); 
6814   int rightCol 
= m_numCols 
> 0 ? GetColRight(m_numCols 
- 1) : 0; 
6815   int bottomRow 
= m_numRows 
> 0 ? GetRowBottom(m_numRows 
- 1) : 0 ; 
6817   if ( right 
> rightCol 
|| bottom 
> bottomRow 
) 
6820       CalcUnscrolledPosition( 0, 0, &left
, &top 
); 
6822       dc
.SetBrush( wxBrush(GetDefaultCellBackgroundColour(), wxSOLID
) ); 
6823       dc
.SetPen( *wxTRANSPARENT_PEN 
); 
6825       if ( right 
> rightCol 
) 
6827           dc
.DrawRectangle( rightCol
, top
, right 
- rightCol
, ch
); 
6830       if ( bottom 
> bottomRow 
) 
6832           dc
.DrawRectangle( left
, bottomRow
, cw
, bottom 
- bottomRow
); 
6838 void wxGrid::DrawCell( wxDC
& dc
, const wxGridCellCoords
& coords 
) 
6840     int row 
= coords
.GetRow(); 
6841     int col 
= coords
.GetCol(); 
6843     if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
6846     // we draw the cell border ourselves 
6847 #if !WXGRID_DRAW_LINES 
6848     if ( m_gridLinesEnabled 
) 
6849         DrawCellBorder( dc
, coords 
); 
6852     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
6854     bool isCurrent 
= coords 
== m_currentCellCoords
; 
6856     wxRect rect 
= CellToRect( row
, col 
); 
6858     // if the editor is shown, we should use it and not the renderer 
6859     // Note: However, only if it is really _shown_, i.e. not hidden! 
6860     if ( isCurrent 
&& IsCellEditControlShown() ) 
6862         wxGridCellEditor 
*editor 
= attr
->GetEditor(this, row
, col
); 
6863         editor
->PaintBackground(rect
, attr
); 
6868         // but all the rest is drawn by the cell renderer and hence may be 
6870         wxGridCellRenderer 
*renderer 
= attr
->GetRenderer(this, row
, col
); 
6871         renderer
->Draw(*this, *attr
, dc
, rect
, row
, col
, IsInSelection(coords
)); 
6878 void wxGrid::DrawCellHighlight( wxDC
& dc
, const wxGridCellAttr 
*attr 
) 
6880     int row 
= m_currentCellCoords
.GetRow(); 
6881     int col 
= m_currentCellCoords
.GetCol(); 
6883     if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
6886     wxRect rect 
= CellToRect(row
, col
); 
6888     // hmmm... what could we do here to show that the cell is disabled? 
6889     // for now, I just draw a thinner border than for the other ones, but 
6890     // it doesn't look really good 
6892     int penWidth 
= attr
->IsReadOnly() ? m_cellHighlightROPenWidth 
: m_cellHighlightPenWidth
; 
6896         // The center of th drawn line is where the position/width/height of 
6897         // the rectangle is actually at, (on wxMSW atr least,) so we will 
6898         // reduce the size of the rectangle to compensate for the thickness of 
6899         // the line.  If this is too strange on non wxMSW platforms then 
6900         // please #ifdef this appropriately. 
6901         rect
.x 
+= penWidth
/2; 
6902         rect
.y 
+= penWidth
/2; 
6903         rect
.width 
-= penWidth
-1; 
6904         rect
.height 
-= penWidth
-1; 
6907         // Now draw the rectangle 
6908         // use the cellHighlightColour if the cell is inside a selection, this 
6909         // will ensure the cell is always visible. 
6910         dc
.SetPen(wxPen(IsInSelection(row
,col
)?m_selectionForeground
:m_cellHighlightColour
, penWidth
, wxSOLID
)); 
6911         dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
6912         dc
.DrawRectangle(rect
); 
6916         // VZ: my experiments with 3d borders... 
6918         // how to properly set colours for arbitrary bg? 
6919         wxCoord x1 
= rect
.x
, 
6921                 x2 
= rect
.x 
+ rect
.width 
-1, 
6922                 y2 
= rect
.y 
+ rect
.height 
-1; 
6924         dc
.SetPen(*wxWHITE_PEN
); 
6925         dc
.DrawLine(x1
, y1
, x2
, y1
); 
6926         dc
.DrawLine(x1
, y1
, x1
, y2
); 
6928         dc
.DrawLine(x1 
+ 1, y2 
- 1, x2 
- 1, y2 
- 1); 
6929         dc
.DrawLine(x2 
- 1, y1 
+ 1, x2 
- 1, y2 
); 
6931         dc
.SetPen(*wxBLACK_PEN
); 
6932         dc
.DrawLine(x1
, y2
, x2
, y2
); 
6933         dc
.DrawLine(x2
, y1
, x2
, y2
+1); 
6938 void wxGrid::DrawCellBorder( wxDC
& dc
, const wxGridCellCoords
& coords 
) 
6940     int row 
= coords
.GetRow(); 
6941     int col 
= coords
.GetCol(); 
6942     if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
6945     dc
.SetPen( wxPen(GetGridLineColour(), 1, wxSOLID
) ); 
6947     wxRect rect 
= CellToRect( row
, col 
); 
6949     // right hand border 
6951     dc
.DrawLine( rect
.x 
+ rect
.width
, rect
.y
, 
6952                  rect
.x 
+ rect
.width
, rect
.y 
+ rect
.height 
+ 1 ); 
6956     dc
.DrawLine( rect
.x
,              rect
.y 
+ rect
.height
, 
6957                  rect
.x 
+ rect
.width
, rect
.y 
+ rect
.height
); 
6960 void wxGrid::DrawHighlight(wxDC
& dc
,const  wxGridCellCoordsArray
& cells
) 
6962     // This if block was previously in wxGrid::OnPaint but that doesn't 
6963     // seem to get called under wxGTK - MB 
6965     if ( m_currentCellCoords 
== wxGridNoCellCoords  
&& 
6966          m_numRows 
&& m_numCols 
) 
6968         m_currentCellCoords
.Set(0, 0); 
6971     if ( IsCellEditControlShown() ) 
6973         // don't show highlight when the edit control is shown 
6977     // if the active cell was repainted, repaint its highlight too because it 
6978     // might have been damaged by the grid lines 
6979     size_t count 
= cells
.GetCount(); 
6980     for ( size_t n 
= 0; n 
< count
; n
++ ) 
6982         if ( cells
[n
] == m_currentCellCoords 
) 
6984             wxGridCellAttr
* attr 
= GetCellAttr(m_currentCellCoords
); 
6985             DrawCellHighlight(dc
, attr
); 
6993 // TODO: remove this ??? 
6994 // This is used to redraw all grid lines e.g. when the grid line colour 
6997 void wxGrid::DrawAllGridLines( wxDC
& dc
, const wxRegion 
& WXUNUSED(reg
) ) 
6999 #if !WXGRID_DRAW_LINES 
7003     if ( !m_gridLinesEnabled 
|| 
7005          !m_numCols 
) return; 
7007     int top
, bottom
, left
, right
; 
7009 #if 0  //#ifndef __WXGTK__ 
7013       m_gridWin
->GetClientSize(&cw
, &ch
); 
7015       // virtual coords of visible area 
7017       CalcUnscrolledPosition( 0, 0, &left
, &top 
); 
7018       CalcUnscrolledPosition( cw
, ch
, &right
, &bottom 
); 
7023       reg
.GetBox(x
, y
, w
, h
); 
7024       CalcUnscrolledPosition( x
, y
, &left
, &top 
); 
7025       CalcUnscrolledPosition( x 
+ w
, y 
+ h
, &right
, &bottom 
); 
7029       m_gridWin
->GetClientSize(&cw
, &ch
); 
7030       CalcUnscrolledPosition( 0, 0, &left
, &top 
); 
7031       CalcUnscrolledPosition( cw
, ch
, &right
, &bottom 
); 
7034     // avoid drawing grid lines past the last row and col 
7036     right 
= wxMin( right
, GetColRight(m_numCols 
- 1) ); 
7037     bottom 
= wxMin( bottom
, GetRowBottom(m_numRows 
- 1) ); 
7039     // no gridlines inside multicells, clip them out 
7040     int leftCol   
= internalXToCol(left
); 
7041     int topRow    
= internalYToRow(top
); 
7042     int rightCol  
= internalXToCol(right
); 
7043     int bottomRow 
= internalYToRow(bottom
); 
7044     wxRegion 
clippedcells(0, 0, cw
, ch
); 
7047     int i
, j
, cell_rows
, cell_cols
; 
7050     for (j
=topRow
; j
<bottomRow
; j
++) 
7052         for (i
=leftCol
; i
<rightCol
; i
++) 
7054             GetCellSize( j
, i
, &cell_rows
, &cell_cols 
); 
7055             if ((cell_rows 
> 1) || (cell_cols 
> 1)) 
7057                 rect 
= CellToRect(j
,i
); 
7058                 CalcScrolledPosition( rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
7059                 clippedcells
.Subtract(rect
); 
7061             else if ((cell_rows 
< 0) || (cell_cols 
< 0)) 
7063                 rect 
= CellToRect(j
+cell_rows
, i
+cell_cols
); 
7064                 CalcScrolledPosition( rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
7065                 clippedcells
.Subtract(rect
); 
7069     dc
.SetClippingRegion( clippedcells 
); 
7071     dc
.SetPen( wxPen(GetGridLineColour(), 1, wxSOLID
) ); 
7073     // horizontal grid lines 
7075     // already declared above - int i; 
7076     for ( i 
= internalYToRow(top
); i 
< m_numRows
; i
++ ) 
7078         int bot 
= GetRowBottom(i
) - 1; 
7087             dc
.DrawLine( left
, bot
, right
, bot 
); 
7092     // vertical grid lines 
7094     for ( i 
= internalXToCol(left
); i 
< m_numCols
; i
++ ) 
7096         int colRight 
= GetColRight(i
) - 1; 
7097         if ( colRight 
> right 
) 
7102         if ( colRight 
>= left 
) 
7104             dc
.DrawLine( colRight
, top
, colRight
, bottom 
); 
7107     dc
.DestroyClippingRegion(); 
7111 void wxGrid::DrawRowLabels( wxDC
& dc 
,const wxArrayInt
& rows
) 
7113     if ( !m_numRows 
) return; 
7116     size_t numLabels 
= rows
.GetCount(); 
7118     for ( i 
= 0;  i 
< numLabels
;  i
++ ) 
7120         DrawRowLabel( dc
, rows
[i
] ); 
7125 void wxGrid::DrawRowLabel( wxDC
& dc
, int row 
) 
7127     if ( GetRowHeight(row
) <= 0 ) 
7130     int rowTop 
= GetRowTop(row
), 
7131         rowBottom 
= GetRowBottom(row
) - 1; 
7133     dc
.SetPen( wxPen(wxSystemSettings::GetColour(wxSYS_COLOUR_3DDKSHADOW
),1, wxSOLID
) ); 
7134     dc
.DrawLine( m_rowLabelWidth
-1, rowTop
, 
7135                  m_rowLabelWidth
-1, rowBottom 
); 
7137     dc
.DrawLine( 0, rowTop
, 0, rowBottom 
); 
7139     dc
.DrawLine( 0, rowBottom
, m_rowLabelWidth
, rowBottom 
); 
7141     dc
.SetPen( *wxWHITE_PEN 
); 
7142     dc
.DrawLine( 1, rowTop
, 1, rowBottom 
); 
7143     dc
.DrawLine( 1, rowTop
, m_rowLabelWidth
-1, rowTop 
); 
7145     dc
.SetBackgroundMode( wxTRANSPARENT 
); 
7146     dc
.SetTextForeground( GetLabelTextColour() ); 
7147     dc
.SetFont( GetLabelFont() ); 
7150     GetRowLabelAlignment( &hAlign
, &vAlign 
); 
7154     rect
.SetY( GetRowTop(row
) + 2 ); 
7155     rect
.SetWidth( m_rowLabelWidth 
- 4 ); 
7156     rect
.SetHeight( GetRowHeight(row
) - 4 ); 
7157     DrawTextRectangle( dc
, GetRowLabelValue( row 
), rect
, hAlign
, vAlign 
); 
7161 void wxGrid::DrawColLabels( wxDC
& dc
,const wxArrayInt
& cols 
) 
7163     if ( !m_numCols 
) return; 
7166     size_t numLabels 
= cols
.GetCount(); 
7168     for ( i 
= 0;  i 
< numLabels
;  i
++ ) 
7170         DrawColLabel( dc
, cols
[i
] ); 
7175 void wxGrid::DrawColLabel( wxDC
& dc
, int col 
) 
7177     if ( GetColWidth(col
) <= 0 ) 
7180     int colLeft 
= GetColLeft(col
), 
7181         colRight 
= GetColRight(col
) - 1; 
7183     dc
.SetPen( wxPen(wxSystemSettings::GetColour(wxSYS_COLOUR_3DDKSHADOW
),1, wxSOLID
) ); 
7184     dc
.DrawLine( colRight
, 0, 
7185                  colRight
, m_colLabelHeight
-1 ); 
7187     dc
.DrawLine( colLeft
, 0, colRight
, 0 ); 
7189     dc
.DrawLine( colLeft
, m_colLabelHeight
-1, 
7190                  colRight
+1, m_colLabelHeight
-1 ); 
7192     dc
.SetPen( *wxWHITE_PEN 
); 
7193     dc
.DrawLine( colLeft
, 1, colLeft
, m_colLabelHeight
-1 ); 
7194     dc
.DrawLine( colLeft
, 1, colRight
, 1 ); 
7196     dc
.SetBackgroundMode( wxTRANSPARENT 
); 
7197     dc
.SetTextForeground( GetLabelTextColour() ); 
7198     dc
.SetFont( GetLabelFont() ); 
7200     int hAlign
, vAlign
, orient
; 
7201     GetColLabelAlignment( &hAlign
, &vAlign 
); 
7202     orient 
= GetColLabelTextOrientation(); 
7205     rect
.SetX( colLeft 
+ 2 ); 
7207     rect
.SetWidth( GetColWidth(col
) - 4 ); 
7208     rect
.SetHeight( m_colLabelHeight 
- 4 ); 
7209     DrawTextRectangle( dc
, GetColLabelValue( col 
), rect
, hAlign
, vAlign
, orient 
); 
7212 void wxGrid::DrawTextRectangle( wxDC
& dc
, 
7213                                 const wxString
& value
, 
7217                                 int textOrientation 
) 
7219     wxArrayString lines
; 
7221     StringToLines( value
, lines 
); 
7224     //Forward to new API. 
7225     DrawTextRectangle(  dc
, 
7234 void wxGrid::DrawTextRectangle( wxDC
& dc
, 
7235                                const wxArrayString
& lines
, 
7239                                int textOrientation 
) 
7241     long textWidth
, textHeight
; 
7242     long lineWidth
, lineHeight
; 
7245     dc
.SetClippingRegion( rect 
); 
7247     nLines 
= lines
.GetCount(); 
7251         float x 
= 0.0, y 
= 0.0; 
7253         if( textOrientation 
== wxHORIZONTAL 
) 
7254             GetTextBoxSize(dc
, lines
, &textWidth
, &textHeight
); 
7256             GetTextBoxSize( dc
, lines
, &textHeight
, &textWidth 
); 
7260         case wxALIGN_BOTTOM
: 
7261             if( textOrientation 
== wxHORIZONTAL 
) 
7262                 y 
= rect
.y 
+ (rect
.height 
- textHeight 
- 1); 
7264                 x 
= rect
.x 
+ rect
.width 
- textWidth
; 
7267         case wxALIGN_CENTRE
: 
7268             if( textOrientation 
== wxHORIZONTAL 
) 
7269                 y 
= rect
.y 
+ ((rect
.height 
- textHeight
)/2); 
7271                 x 
= rect
.x 
+ ((rect
.width 
- textWidth
)/2); 
7276             if( textOrientation 
== wxHORIZONTAL 
) 
7283         // Align each line of a multi-line label 
7284         for( l 
= 0; l 
< nLines
; l
++ ) 
7286             dc
.GetTextExtent(lines
[l
], &lineWidth
, &lineHeight
); 
7288             switch( horizAlign 
) 
7291                 if( textOrientation 
== wxHORIZONTAL 
) 
7292                     x 
= rect
.x 
+ (rect
.width 
- lineWidth 
- 1); 
7294                     y 
= rect
.y 
+ lineWidth 
+ 1; 
7297             case wxALIGN_CENTRE
: 
7298                 if( textOrientation 
== wxHORIZONTAL 
) 
7299                     x 
= rect
.x 
+ ((rect
.width 
- lineWidth
)/2); 
7301                     y 
= rect
.y 
+ rect
.height 
- ((rect
.height 
- lineWidth
)/2); 
7306                 if( textOrientation 
== wxHORIZONTAL 
) 
7309                     y 
= rect
.y 
+ rect
.height 
- 1; 
7313             if( textOrientation 
== wxHORIZONTAL 
) 
7315                 dc
.DrawText( lines
[l
], (int)x
, (int)y 
); 
7320                 dc
.DrawRotatedText( lines
[l
], (int)x
, (int)y
, 90.0 ); 
7325     dc
.DestroyClippingRegion(); 
7329 // Split multi line text up into an array of strings.  Any existing 
7330 // contents of the string array are preserved. 
7332 void wxGrid::StringToLines( const wxString
& value
, wxArrayString
& lines 
) 
7336     wxString eol 
= wxTextFile::GetEOL( wxTextFileType_Unix 
); 
7337     wxString tVal 
= wxTextFile::Translate( value
, wxTextFileType_Unix 
); 
7339     while ( startPos 
< (int)tVal
.Length() ) 
7341         pos 
= tVal
.Mid(startPos
).Find( eol 
); 
7346         else if ( pos 
== 0 ) 
7348             lines
.Add( wxEmptyString 
); 
7352             lines
.Add( value
.Mid(startPos
, pos
) ); 
7356     if ( startPos 
< (int)value
.Length() ) 
7358         lines
.Add( value
.Mid( startPos 
) ); 
7363 void wxGrid::GetTextBoxSize( wxDC
& dc
, 
7364                              const wxArrayString
& lines
, 
7365                              long *width
, long *height 
) 
7372     for ( i 
= 0;  i 
< lines
.GetCount();  i
++ ) 
7374         dc
.GetTextExtent( lines
[i
], &lineW
, &lineH 
); 
7375         w 
= wxMax( w
, lineW 
); 
7384 // ------ Batch processing. 
7386 void wxGrid::EndBatch() 
7388     if ( m_batchCount 
> 0 ) 
7391         if ( !m_batchCount 
) 
7394             m_rowLabelWin
->Refresh(); 
7395             m_colLabelWin
->Refresh(); 
7396             m_cornerLabelWin
->Refresh(); 
7397             m_gridWin
->Refresh(); 
7402 // Use this, rather than wxWindow::Refresh(), to force an immediate 
7403 // repainting of the grid. Has no effect if you are already inside a 
7404 // BeginBatch / EndBatch block. 
7406 void wxGrid::ForceRefresh() 
7414 // ------ Edit control functions 
7418 void wxGrid::EnableEditing( bool edit 
) 
7420     // TODO: improve this ? 
7422     if ( edit 
!= m_editable 
) 
7424         if(!edit
) EnableCellEditControl(edit
); 
7430 void wxGrid::EnableCellEditControl( bool enable 
) 
7435     if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
7436         SetCurrentCell( 0, 0 ); 
7438     if ( enable 
!= m_cellEditCtrlEnabled 
) 
7442             if (SendEvent( wxEVT_GRID_EDITOR_SHOWN
) <0) 
7445             // this should be checked by the caller! 
7446             wxASSERT_MSG( CanEnableCellControl(), 
7447                           _T("can't enable editing for this cell!") ); 
7449             // do it before ShowCellEditControl() 
7450             m_cellEditCtrlEnabled 
= enable
; 
7452             ShowCellEditControl(); 
7456             //FIXME:add veto support 
7457             SendEvent( wxEVT_GRID_EDITOR_HIDDEN
); 
7459             HideCellEditControl(); 
7460             SaveEditControlValue(); 
7462             // do it after HideCellEditControl() 
7463             m_cellEditCtrlEnabled 
= enable
; 
7468 bool wxGrid::IsCurrentCellReadOnly() const 
7471     wxGridCellAttr
* attr 
= ((wxGrid 
*)this)->GetCellAttr(m_currentCellCoords
); 
7472     bool readonly 
= attr
->IsReadOnly(); 
7478 bool wxGrid::CanEnableCellControl() const 
7480     return m_editable 
&& (m_currentCellCoords 
!= wxGridNoCellCoords
) && 
7481         !IsCurrentCellReadOnly(); 
7485 bool wxGrid::IsCellEditControlEnabled() const 
7487     // the cell edit control might be disable for all cells or just for the 
7488     // current one if it's read only 
7489     return m_cellEditCtrlEnabled 
? !IsCurrentCellReadOnly() : FALSE
; 
7492 bool wxGrid::IsCellEditControlShown() const 
7494     bool isShown 
= FALSE
; 
7496     if ( m_cellEditCtrlEnabled 
) 
7498         int row 
= m_currentCellCoords
.GetRow(); 
7499         int col 
= m_currentCellCoords
.GetCol(); 
7500         wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
7501         wxGridCellEditor
* editor 
= attr
->GetEditor((wxGrid
*) this, row
, col
); 
7506             if ( editor
->IsCreated() ) 
7508                 isShown 
= editor
->GetControl()->IsShown(); 
7518 void wxGrid::ShowCellEditControl() 
7520     if ( IsCellEditControlEnabled() ) 
7522         if ( !IsVisible( m_currentCellCoords 
) ) 
7524             m_cellEditCtrlEnabled 
= FALSE
; 
7529             wxRect rect 
= CellToRect( m_currentCellCoords 
); 
7530             int row 
= m_currentCellCoords
.GetRow(); 
7531             int col 
= m_currentCellCoords
.GetCol(); 
7533             // if this is part of a multicell, find owner (topleft) 
7534             int cell_rows
, cell_cols
; 
7535             GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
7536             if ( cell_rows 
<= 0 || cell_cols 
<= 0 ) 
7540                 m_currentCellCoords
.SetRow( row 
); 
7541                 m_currentCellCoords
.SetCol( col 
); 
7544             // convert to scrolled coords 
7546             CalcScrolledPosition( rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
7548             // done in PaintBackground() 
7550             // erase the highlight and the cell contents because the editor 
7551             // might not cover the entire cell 
7552             wxClientDC 
dc( m_gridWin 
); 
7554             dc
.SetBrush(*wxLIGHT_GREY_BRUSH
); //wxBrush(attr->GetBackgroundColour(), wxSOLID)); 
7555             dc
.SetPen(*wxTRANSPARENT_PEN
); 
7556             dc
.DrawRectangle(rect
); 
7559             // cell is shifted by one pixel 
7560             // However, don't allow x or y to become negative 
7561             // since the SetSize() method interprets that as 
7568             wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
7569             wxGridCellEditor
* editor 
= attr
->GetEditor(this, row
, col
); 
7570             if ( !editor
->IsCreated() ) 
7572                 editor
->Create(m_gridWin
, -1, 
7573                                new wxGridCellEditorEvtHandler(this, editor
)); 
7575                 wxGridEditorCreatedEvent 
evt(GetId(), 
7576                                              wxEVT_GRID_EDITOR_CREATED
, 
7580                                              editor
->GetControl()); 
7581                 GetEventHandler()->ProcessEvent(evt
); 
7585             // resize editor to overflow into righthand cells if allowed 
7586             int maxWidth 
= rect
.width
; 
7587             wxString value 
= GetCellValue(row
, col
); 
7588             if ( (value 
!= wxEmptyString
) && (attr
->GetOverflow()) ) 
7591                 GetTextExtent(value
, &maxWidth
, &y
, 
7592                         NULL
, NULL
, &attr
->GetFont()); 
7593                 if (maxWidth 
< rect
.width
) maxWidth 
= rect
.width
; 
7595             int client_right 
= m_gridWin
->GetClientSize().GetWidth(); 
7596             if (rect
.x
+maxWidth 
> client_right
) 
7597                 maxWidth 
= client_right 
- rect
.x
; 
7599             if ((maxWidth 
> rect
.width
) && (col 
< m_numCols
) && m_table
) 
7601                 GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
7602                 // may have changed earlier 
7603                 for (int i 
= col
+cell_cols
; i 
< m_numCols
; i
++) 
7606                     GetCellSize( row
, i
, &c_rows
, &c_cols 
); 
7607                     // looks weird going over a multicell 
7608                     if (m_table
->IsEmptyCell(row
,i
) && 
7609                             (rect
.width 
< maxWidth
) && (c_rows 
== 1)) 
7610                         rect
.width 
+= GetColWidth(i
); 
7614                 if (rect
.GetRight() > client_right
) 
7615                     rect
.SetRight(client_right
-1); 
7618             editor
->SetCellAttr(attr
); 
7619             editor
->SetSize( rect 
); 
7620             editor
->Show( TRUE
, attr 
); 
7622             // recalc dimensions in case we need to 
7623             // expand the scrolled window to account for editor 
7626             editor
->BeginEdit(row
, col
, this); 
7627             editor
->SetCellAttr(NULL
); 
7636 void wxGrid::HideCellEditControl() 
7638     if ( IsCellEditControlEnabled() ) 
7640         int row 
= m_currentCellCoords
.GetRow(); 
7641         int col 
= m_currentCellCoords
.GetCol(); 
7643         wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
7644         wxGridCellEditor 
*editor 
= attr
->GetEditor(this, row
, col
); 
7645         editor
->Show( FALSE 
); 
7648         m_gridWin
->SetFocus(); 
7649         // refresh whole row to the right 
7650         wxRect 
rect( CellToRect(row
, col
) ); 
7651         CalcScrolledPosition(rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
7652         rect
.width 
= m_gridWin
->GetClientSize().GetWidth() - rect
.x
; 
7653         m_gridWin
->Refresh( FALSE
, &rect 
); 
7658 void wxGrid::SaveEditControlValue() 
7660     if ( IsCellEditControlEnabled() ) 
7662         int row 
= m_currentCellCoords
.GetRow(); 
7663         int col 
= m_currentCellCoords
.GetCol(); 
7665         wxString oldval 
= GetCellValue(row
,col
); 
7667         wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
7668         wxGridCellEditor
* editor 
= attr
->GetEditor(this, row
, col
); 
7669         bool changed 
= editor
->EndEdit(row
, col
, this); 
7676             if ( SendEvent( wxEVT_GRID_CELL_CHANGE
, 
7677                        m_currentCellCoords
.GetRow(), 
7678                        m_currentCellCoords
.GetCol() ) < 0 ) { 
7680                 // Event has been vetoed, set the data back. 
7681                 SetCellValue(row
,col
,oldval
); 
7689 // ------ Grid location functions 
7690 //  Note that all of these functions work with the logical coordinates of 
7691 //  grid cells and labels so you will need to convert from device 
7692 //  coordinates for mouse events etc. 
7695 void wxGrid::XYToCell( int x
, int y
, wxGridCellCoords
& coords 
) 
7697     int row 
= YToRow(y
); 
7698     int col 
= XToCol(x
); 
7700     if ( row 
== -1  ||  col 
== -1 ) 
7702         coords 
= wxGridNoCellCoords
; 
7706         coords
.Set( row
, col 
); 
7711 // Internal Helper function for computing row or column from some 
7712 // (unscrolled) coordinate value, using either 
7713 // m_defaultRowHeight/m_defaultColWidth or binary search on array 
7714 // of m_rowBottoms/m_ColRights to speed up the search! 
7716 static int CoordToRowOrCol(int coord
, int defaultDist
, int minDist
, 
7717                            const wxArrayInt
& BorderArray
, int nMax
, 
7722         return clipToMinMax 
&& (nMax 
> 0) ? 0 : -1; 
7728     size_t i_max 
= coord 
/ defaultDist
, 
7731     if (BorderArray
.IsEmpty()) 
7733         if((int) i_max 
< nMax
) 
7735         return clipToMinMax 
? nMax 
- 1 : -1; 
7738     if ( i_max 
>= BorderArray
.GetCount()) 
7739         i_max 
= BorderArray
.GetCount() - 1; 
7742         if ( coord 
>= BorderArray
[i_max
]) 
7746                 i_max 
= coord 
/ minDist
; 
7748                 i_max 
=  BorderArray
.GetCount() - 1; 
7750         if ( i_max 
>= BorderArray
.GetCount()) 
7751             i_max 
= BorderArray
.GetCount() - 1; 
7753     if ( coord 
>= BorderArray
[i_max
]) 
7754         return clipToMinMax 
? (int)i_max 
: -1; 
7755     if ( coord 
< BorderArray
[0] ) 
7758     while ( i_max 
- i_min 
> 0 ) 
7760         wxCHECK_MSG(BorderArray
[i_min
] <= coord 
&& coord 
< BorderArray
[i_max
], 
7761                     0, _T("wxGrid: internal error in CoordToRowOrCol")); 
7762         if (coord 
>=  BorderArray
[ i_max 
- 1]) 
7766         int median 
= i_min 
+ (i_max 
- i_min 
+ 1) / 2; 
7767         if (coord 
< BorderArray
[median
]) 
7775 int wxGrid::YToRow( int y 
) 
7777     return CoordToRowOrCol(y
, m_defaultRowHeight
, 
7778                            m_minAcceptableRowHeight
, m_rowBottoms
, m_numRows
, FALSE
); 
7782 int wxGrid::XToCol( int x 
) 
7784     return CoordToRowOrCol(x
, m_defaultColWidth
, 
7785                            m_minAcceptableColWidth
, m_colRights
, m_numCols
, FALSE
); 
7789 // return the row number that that the y coord is near the edge of, or 
7790 // -1 if not near an edge 
7792 int wxGrid::YToEdgeOfRow( int y 
) 
7795     i 
= internalYToRow(y
); 
7797     if ( GetRowHeight(i
) > WXGRID_LABEL_EDGE_ZONE 
) 
7799         // We know that we are in row i, test whether we are 
7800         // close enough to lower or upper border, respectively. 
7801         if ( abs(GetRowBottom(i
) - y
) < WXGRID_LABEL_EDGE_ZONE 
) 
7803         else if( i 
> 0 && y 
- GetRowTop(i
) < WXGRID_LABEL_EDGE_ZONE 
) 
7811 // return the col number that that the x coord is near the edge of, or 
7812 // -1 if not near an edge 
7814 int wxGrid::XToEdgeOfCol( int x 
) 
7817     i 
= internalXToCol(x
); 
7819     if ( GetColWidth(i
) > WXGRID_LABEL_EDGE_ZONE 
) 
7821         // We know that we are in column i,  test whether we are 
7822         // close enough to right or left border, respectively. 
7823         if ( abs(GetColRight(i
) - x
) < WXGRID_LABEL_EDGE_ZONE 
) 
7825         else if( i 
> 0 && x 
- GetColLeft(i
) < WXGRID_LABEL_EDGE_ZONE 
) 
7833 wxRect 
wxGrid::CellToRect( int row
, int col 
) 
7835     wxRect 
rect( -1, -1, -1, -1 ); 
7837     if ( row 
>= 0  &&  row 
< m_numRows  
&& 
7838          col 
>= 0  &&  col 
< m_numCols 
) 
7840         int i
, cell_rows
, cell_cols
; 
7841         rect
.width 
= rect
.height 
= 0; 
7842         GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
7843         // if negative then find multicell owner 
7844         if (cell_rows 
< 0) row 
+= cell_rows
; 
7845         if (cell_cols 
< 0) col 
+= cell_cols
; 
7846         GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
7848         rect
.x 
= GetColLeft(col
); 
7849         rect
.y 
= GetRowTop(row
); 
7850         for (i
=col
; i
<col
+cell_cols
; i
++) 
7851             rect
.width  
+= GetColWidth(i
); 
7852         for (i
=row
; i
<row
+cell_rows
; i
++) 
7853             rect
.height 
+= GetRowHeight(i
); 
7856     // if grid lines are enabled, then the area of the cell is a bit smaller 
7857     if (m_gridLinesEnabled
) { 
7865 bool wxGrid::IsVisible( int row
, int col
, bool wholeCellVisible 
) 
7867     // get the cell rectangle in logical coords 
7869     wxRect 
r( CellToRect( row
, col 
) ); 
7871     // convert to device coords 
7873     int left
, top
, right
, bottom
; 
7874     CalcScrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top 
); 
7875     CalcScrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom 
); 
7877     // check against the client area of the grid window 
7880     m_gridWin
->GetClientSize( &cw
, &ch 
); 
7882     if ( wholeCellVisible 
) 
7884         // is the cell wholly visible ? 
7886         return ( left 
>= 0  &&  right 
<= cw  
&& 
7887                  top 
>= 0  &&  bottom 
<= ch 
); 
7891         // is the cell partly visible ? 
7893         return ( ((left 
>=0 && left 
< cw
) || (right 
> 0 && right 
<= cw
))  && 
7894                  ((top 
>=0 && top 
< ch
) || (bottom 
> 0 && bottom 
<= ch
)) ); 
7899 // make the specified cell location visible by doing a minimal amount 
7902 void wxGrid::MakeCellVisible( int row
, int col 
) 
7906     int xpos 
= -1, ypos 
= -1; 
7908     if ( row 
>= 0  &&  row 
< m_numRows  
&& 
7909          col 
>= 0  &&  col 
< m_numCols 
) 
7911         // get the cell rectangle in logical coords 
7913         wxRect 
r( CellToRect( row
, col 
) ); 
7915         // convert to device coords 
7917         int left
, top
, right
, bottom
; 
7918         CalcScrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top 
); 
7919         CalcScrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom 
); 
7922         m_gridWin
->GetClientSize( &cw
, &ch 
); 
7928         else if ( bottom 
> ch 
) 
7930             int h 
= r
.GetHeight(); 
7932             for ( i 
= row
-1;  i 
>= 0;  i
-- ) 
7934                 int rowHeight 
= GetRowHeight(i
); 
7935                 if ( h 
+ rowHeight 
> ch 
) 
7942             // we divide it later by GRID_SCROLL_LINE, make sure that we don't 
7943             // have rounding errors (this is important, because if we do, we 
7944             // might not scroll at all and some cells won't be redrawn) 
7946             // Sometimes GRID_SCROLL_LINE/2 is not enough, so just add a full 
7948             ypos 
+= GRID_SCROLL_LINE_Y
; 
7955         else if ( right 
> cw 
) 
7957             // position the view so that the cell is on the right 
7959             CalcUnscrolledPosition(0, 0, &x0
, &y0
); 
7960             xpos 
= x0 
+ (right 
- cw
); 
7962             // see comment for ypos above 
7963             xpos 
+= GRID_SCROLL_LINE_X
; 
7966         if ( xpos 
!= -1  ||  ypos 
!= -1 ) 
7969                 xpos 
/= GRID_SCROLL_LINE_X
; 
7971                 ypos 
/= GRID_SCROLL_LINE_Y
; 
7972             Scroll( xpos
, ypos 
); 
7980 // ------ Grid cursor movement functions 
7983 bool wxGrid::MoveCursorUp( bool expandSelection 
) 
7985     if ( m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
7986          m_currentCellCoords
.GetRow() >= 0 ) 
7988         if ( expandSelection
) 
7990             if ( m_selectingKeyboard 
== wxGridNoCellCoords 
) 
7991                 m_selectingKeyboard 
= m_currentCellCoords
; 
7992             if ( m_selectingKeyboard
.GetRow() > 0 ) 
7994                 m_selectingKeyboard
.SetRow( m_selectingKeyboard
.GetRow() - 1 ); 
7995                 MakeCellVisible( m_selectingKeyboard
.GetRow(), 
7996                                  m_selectingKeyboard
.GetCol() ); 
7997                 HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8000         else if ( m_currentCellCoords
.GetRow() > 0 ) 
8003             MakeCellVisible( m_currentCellCoords
.GetRow() - 1, 
8004                              m_currentCellCoords
.GetCol() ); 
8005             SetCurrentCell( m_currentCellCoords
.GetRow() - 1, 
8006                             m_currentCellCoords
.GetCol() ); 
8017 bool wxGrid::MoveCursorDown( bool expandSelection 
) 
8019     if ( m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
8020          m_currentCellCoords
.GetRow() < m_numRows 
) 
8022         if ( expandSelection 
) 
8024             if ( m_selectingKeyboard 
== wxGridNoCellCoords 
) 
8025                 m_selectingKeyboard 
= m_currentCellCoords
; 
8026             if ( m_selectingKeyboard
.GetRow() < m_numRows
-1 ) 
8028                 m_selectingKeyboard
.SetRow( m_selectingKeyboard
.GetRow() + 1 ); 
8029                 MakeCellVisible( m_selectingKeyboard
.GetRow(), 
8030                         m_selectingKeyboard
.GetCol() ); 
8031                 HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8034         else if ( m_currentCellCoords
.GetRow() < m_numRows 
- 1 ) 
8037             MakeCellVisible( m_currentCellCoords
.GetRow() + 1, 
8038                              m_currentCellCoords
.GetCol() ); 
8039             SetCurrentCell( m_currentCellCoords
.GetRow() + 1, 
8040                             m_currentCellCoords
.GetCol() ); 
8051 bool wxGrid::MoveCursorLeft( bool expandSelection 
) 
8053     if ( m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
8054          m_currentCellCoords
.GetCol() >= 0 ) 
8056         if ( expandSelection 
) 
8058             if ( m_selectingKeyboard 
== wxGridNoCellCoords 
) 
8059                 m_selectingKeyboard 
= m_currentCellCoords
; 
8060             if ( m_selectingKeyboard
.GetCol() > 0 ) 
8062                 m_selectingKeyboard
.SetCol( m_selectingKeyboard
.GetCol() - 1 ); 
8063                 MakeCellVisible( m_selectingKeyboard
.GetRow(), 
8064                         m_selectingKeyboard
.GetCol() ); 
8065                 HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8068         else if ( m_currentCellCoords
.GetCol() > 0 ) 
8071             MakeCellVisible( m_currentCellCoords
.GetRow(), 
8072                              m_currentCellCoords
.GetCol() - 1 ); 
8073             SetCurrentCell( m_currentCellCoords
.GetRow(), 
8074                             m_currentCellCoords
.GetCol() - 1 ); 
8085 bool wxGrid::MoveCursorRight( bool expandSelection 
) 
8087     if ( m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
8088          m_currentCellCoords
.GetCol() < m_numCols 
) 
8090         if ( expandSelection 
) 
8092             if ( m_selectingKeyboard 
== wxGridNoCellCoords 
) 
8093                 m_selectingKeyboard 
= m_currentCellCoords
; 
8094             if ( m_selectingKeyboard
.GetCol() < m_numCols 
- 1 ) 
8096                 m_selectingKeyboard
.SetCol( m_selectingKeyboard
.GetCol() + 1 ); 
8097                 MakeCellVisible( m_selectingKeyboard
.GetRow(), 
8098                         m_selectingKeyboard
.GetCol() ); 
8099                 HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8102         else if ( m_currentCellCoords
.GetCol() < m_numCols 
- 1 ) 
8105             MakeCellVisible( m_currentCellCoords
.GetRow(), 
8106                              m_currentCellCoords
.GetCol() + 1 ); 
8107             SetCurrentCell( m_currentCellCoords
.GetRow(), 
8108                             m_currentCellCoords
.GetCol() + 1 ); 
8119 bool wxGrid::MovePageUp() 
8121     if ( m_currentCellCoords 
== wxGridNoCellCoords 
) return FALSE
; 
8123     int row 
= m_currentCellCoords
.GetRow(); 
8127         m_gridWin
->GetClientSize( &cw
, &ch 
); 
8129         int y 
= GetRowTop(row
); 
8130         int newRow 
= internalYToRow( y 
- ch 
+ 1 ); 
8132         if ( newRow 
== row 
) 
8134             //row > 0 , so newrow can never be less than 0 here. 
8138         MakeCellVisible( newRow
, m_currentCellCoords
.GetCol() ); 
8139         SetCurrentCell( newRow
, m_currentCellCoords
.GetCol() ); 
8147 bool wxGrid::MovePageDown() 
8149     if ( m_currentCellCoords 
== wxGridNoCellCoords 
) return FALSE
; 
8151     int row 
= m_currentCellCoords
.GetRow(); 
8152     if ( (row
+1) < m_numRows 
) 
8155         m_gridWin
->GetClientSize( &cw
, &ch 
); 
8157         int y 
= GetRowTop(row
); 
8158         int newRow 
= internalYToRow( y 
+ ch 
); 
8159         if ( newRow 
== row 
) 
8161             // row < m_numRows , so newrow can't overflow here. 
8165         MakeCellVisible( newRow
, m_currentCellCoords
.GetCol() ); 
8166         SetCurrentCell( newRow
, m_currentCellCoords
.GetCol() ); 
8174 bool wxGrid::MoveCursorUpBlock( bool expandSelection 
) 
8177          m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
8178          m_currentCellCoords
.GetRow() > 0 ) 
8180         int row 
= m_currentCellCoords
.GetRow(); 
8181         int col 
= m_currentCellCoords
.GetCol(); 
8183         if ( m_table
->IsEmptyCell(row
, col
) ) 
8185             // starting in an empty cell: find the next block of 
8191                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8194         else if ( m_table
->IsEmptyCell(row
-1, col
) ) 
8196             // starting at the top of a block: find the next block 
8202                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8207             // starting within a block: find the top of the block 
8212                 if ( m_table
->IsEmptyCell(row
, col
) ) 
8220         MakeCellVisible( row
, col 
); 
8221         if ( expandSelection 
) 
8223             m_selectingKeyboard 
= wxGridCellCoords( row
, col 
); 
8224             HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8229             SetCurrentCell( row
, col 
); 
8237 bool wxGrid::MoveCursorDownBlock( bool expandSelection 
) 
8240          m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
8241          m_currentCellCoords
.GetRow() < m_numRows
-1 ) 
8243         int row 
= m_currentCellCoords
.GetRow(); 
8244         int col 
= m_currentCellCoords
.GetCol(); 
8246         if ( m_table
->IsEmptyCell(row
, col
) ) 
8248             // starting in an empty cell: find the next block of 
8251             while ( row 
< m_numRows
-1 ) 
8254                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8257         else if ( m_table
->IsEmptyCell(row
+1, col
) ) 
8259             // starting at the bottom of a block: find the next block 
8262             while ( row 
< m_numRows
-1 ) 
8265                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8270             // starting within a block: find the bottom of the block 
8272             while ( row 
< m_numRows
-1 ) 
8275                 if ( m_table
->IsEmptyCell(row
, col
) ) 
8283         MakeCellVisible( row
, col 
); 
8284         if ( expandSelection 
) 
8286             m_selectingKeyboard 
= wxGridCellCoords( row
, col 
); 
8287             HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8292             SetCurrentCell( row
, col 
); 
8301 bool wxGrid::MoveCursorLeftBlock( bool expandSelection 
) 
8304          m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
8305          m_currentCellCoords
.GetCol() > 0 ) 
8307         int row 
= m_currentCellCoords
.GetRow(); 
8308         int col 
= m_currentCellCoords
.GetCol(); 
8310         if ( m_table
->IsEmptyCell(row
, col
) ) 
8312             // starting in an empty cell: find the next block of 
8318                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8321         else if ( m_table
->IsEmptyCell(row
, col
-1) ) 
8323             // starting at the left of a block: find the next block 
8329                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8334             // starting within a block: find the left of the block 
8339                 if ( m_table
->IsEmptyCell(row
, col
) ) 
8347         MakeCellVisible( row
, col 
); 
8348         if ( expandSelection 
) 
8350             m_selectingKeyboard 
= wxGridCellCoords( row
, col 
); 
8351             HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8356             SetCurrentCell( row
, col 
); 
8365 bool wxGrid::MoveCursorRightBlock( bool expandSelection 
) 
8368          m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
8369          m_currentCellCoords
.GetCol() < m_numCols
-1 ) 
8371         int row 
= m_currentCellCoords
.GetRow(); 
8372         int col 
= m_currentCellCoords
.GetCol(); 
8374         if ( m_table
->IsEmptyCell(row
, col
) ) 
8376             // starting in an empty cell: find the next block of 
8379             while ( col 
< m_numCols
-1 ) 
8382                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8385         else if ( m_table
->IsEmptyCell(row
, col
+1) ) 
8387             // starting at the right of a block: find the next block 
8390             while ( col 
< m_numCols
-1 ) 
8393                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8398             // starting within a block: find the right of the block 
8400             while ( col 
< m_numCols
-1 ) 
8403                 if ( m_table
->IsEmptyCell(row
, col
) ) 
8411         MakeCellVisible( row
, col 
); 
8412         if ( expandSelection 
) 
8414             m_selectingKeyboard 
= wxGridCellCoords( row
, col 
); 
8415             HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8420             SetCurrentCell( row
, col 
); 
8432 // ------ Label values and formatting 
8435 void wxGrid::GetRowLabelAlignment( int *horiz
, int *vert 
) 
8437     *horiz 
= m_rowLabelHorizAlign
; 
8438     *vert  
= m_rowLabelVertAlign
; 
8441 void wxGrid::GetColLabelAlignment( int *horiz
, int *vert 
) 
8443     *horiz 
= m_colLabelHorizAlign
; 
8444     *vert  
= m_colLabelVertAlign
; 
8447 int wxGrid::GetColLabelTextOrientation() 
8449     return m_colLabelTextOrientation
; 
8452 wxString 
wxGrid::GetRowLabelValue( int row 
) 
8456         return m_table
->GetRowLabelValue( row 
); 
8466 wxString 
wxGrid::GetColLabelValue( int col 
) 
8470         return m_table
->GetColLabelValue( col 
); 
8481 void wxGrid::SetRowLabelSize( int width 
) 
8483     width 
= wxMax( width
, 0 ); 
8484     if ( width 
!= m_rowLabelWidth 
) 
8488             m_rowLabelWin
->Show( FALSE 
); 
8489             m_cornerLabelWin
->Show( FALSE 
); 
8491         else if ( m_rowLabelWidth 
== 0 ) 
8493             m_rowLabelWin
->Show( TRUE 
); 
8494             if ( m_colLabelHeight 
> 0 ) m_cornerLabelWin
->Show( TRUE 
); 
8497         m_rowLabelWidth 
= width
; 
8499         wxScrolledWindow::Refresh( TRUE 
); 
8504 void wxGrid::SetColLabelSize( int height 
) 
8506     height 
= wxMax( height
, 0 ); 
8507     if ( height 
!= m_colLabelHeight 
) 
8511             m_colLabelWin
->Show( FALSE 
); 
8512             m_cornerLabelWin
->Show( FALSE 
); 
8514         else if ( m_colLabelHeight 
== 0 ) 
8516             m_colLabelWin
->Show( TRUE 
); 
8517             if ( m_rowLabelWidth 
> 0 ) m_cornerLabelWin
->Show( TRUE 
); 
8520         m_colLabelHeight 
= height
; 
8522         wxScrolledWindow::Refresh( TRUE 
); 
8527 void wxGrid::SetLabelBackgroundColour( const wxColour
& colour 
) 
8529     if ( m_labelBackgroundColour 
!= colour 
) 
8531         m_labelBackgroundColour 
= colour
; 
8532         m_rowLabelWin
->SetBackgroundColour( colour 
); 
8533         m_colLabelWin
->SetBackgroundColour( colour 
); 
8534         m_cornerLabelWin
->SetBackgroundColour( colour 
); 
8536         if ( !GetBatchCount() ) 
8538             m_rowLabelWin
->Refresh(); 
8539             m_colLabelWin
->Refresh(); 
8540             m_cornerLabelWin
->Refresh(); 
8545 void wxGrid::SetLabelTextColour( const wxColour
& colour 
) 
8547     if ( m_labelTextColour 
!= colour 
) 
8549         m_labelTextColour 
= colour
; 
8550         if ( !GetBatchCount() ) 
8552             m_rowLabelWin
->Refresh(); 
8553             m_colLabelWin
->Refresh(); 
8558 void wxGrid::SetLabelFont( const wxFont
& font 
) 
8561     if ( !GetBatchCount() ) 
8563         m_rowLabelWin
->Refresh(); 
8564         m_colLabelWin
->Refresh(); 
8568 void wxGrid::SetRowLabelAlignment( int horiz
, int vert 
) 
8570     // allow old (incorrect) defs to be used 
8573         case wxLEFT
:   horiz 
= wxALIGN_LEFT
; break; 
8574         case wxRIGHT
:  horiz 
= wxALIGN_RIGHT
; break; 
8575         case wxCENTRE
: horiz 
= wxALIGN_CENTRE
; break; 
8580         case wxTOP
:    vert 
= wxALIGN_TOP
;    break; 
8581         case wxBOTTOM
: vert 
= wxALIGN_BOTTOM
; break; 
8582         case wxCENTRE
: vert 
= wxALIGN_CENTRE
; break; 
8585     if ( horiz 
== wxALIGN_LEFT 
|| horiz 
== wxALIGN_CENTRE 
|| horiz 
== wxALIGN_RIGHT 
) 
8587         m_rowLabelHorizAlign 
= horiz
; 
8590     if ( vert 
== wxALIGN_TOP 
|| vert 
== wxALIGN_CENTRE 
|| vert 
== wxALIGN_BOTTOM 
) 
8592         m_rowLabelVertAlign 
= vert
; 
8595     if ( !GetBatchCount() ) 
8597         m_rowLabelWin
->Refresh(); 
8601 void wxGrid::SetColLabelAlignment( int horiz
, int vert 
) 
8603     // allow old (incorrect) defs to be used 
8606         case wxLEFT
:   horiz 
= wxALIGN_LEFT
; break; 
8607         case wxRIGHT
:  horiz 
= wxALIGN_RIGHT
; break; 
8608         case wxCENTRE
: horiz 
= wxALIGN_CENTRE
; break; 
8613         case wxTOP
:    vert 
= wxALIGN_TOP
;    break; 
8614         case wxBOTTOM
: vert 
= wxALIGN_BOTTOM
; break; 
8615         case wxCENTRE
: vert 
= wxALIGN_CENTRE
; break; 
8618     if ( horiz 
== wxALIGN_LEFT 
|| horiz 
== wxALIGN_CENTRE 
|| horiz 
== wxALIGN_RIGHT 
) 
8620         m_colLabelHorizAlign 
= horiz
; 
8623     if ( vert 
== wxALIGN_TOP 
|| vert 
== wxALIGN_CENTRE 
|| vert 
== wxALIGN_BOTTOM 
) 
8625         m_colLabelVertAlign 
= vert
; 
8628     if ( !GetBatchCount() ) 
8630         m_colLabelWin
->Refresh(); 
8634 // Note:        under MSW, the default column label font must be changed because it 
8635 //                              does not support vertical printing 
8637 // Example: wxFont font(9, wxSWISS, wxNORMAL, wxBOLD); 
8638 //                              pGrid->SetLabelFont(font); 
8639 //              pGrid->SetColLabelTextOrientation(wxVERTICAL); 
8641 void wxGrid::SetColLabelTextOrientation( int textOrientation 
) 
8643     if( textOrientation 
== wxHORIZONTAL 
|| textOrientation 
== wxVERTICAL 
) 
8645         m_colLabelTextOrientation 
= textOrientation
; 
8648     if ( !GetBatchCount() ) 
8650         m_colLabelWin
->Refresh(); 
8654 void wxGrid::SetRowLabelValue( int row
, const wxString
& s 
) 
8658         m_table
->SetRowLabelValue( row
, s 
); 
8659         if ( !GetBatchCount() ) 
8661             wxRect rect 
= CellToRect( row
, 0); 
8662             if ( rect
.height 
> 0 ) 
8664                 CalcScrolledPosition(0, rect
.y
, &rect
.x
, &rect
.y
); 
8666                 rect
.width 
= m_rowLabelWidth
; 
8667                 m_rowLabelWin
->Refresh( TRUE
, &rect 
); 
8673 void wxGrid::SetColLabelValue( int col
, const wxString
& s 
) 
8677         m_table
->SetColLabelValue( col
, s 
); 
8678         if ( !GetBatchCount() ) 
8680             wxRect rect 
= CellToRect( 0, col 
); 
8681             if ( rect
.width 
> 0 ) 
8683                 CalcScrolledPosition(rect
.x
, 0, &rect
.x
, &rect
.y
); 
8685                 rect
.height 
= m_colLabelHeight
; 
8686                 m_colLabelWin
->Refresh( TRUE
, &rect 
); 
8692 void wxGrid::SetGridLineColour( const wxColour
& colour 
) 
8694     if ( m_gridLineColour 
!= colour 
) 
8696         m_gridLineColour 
= colour
; 
8698         wxClientDC 
dc( m_gridWin 
); 
8700         DrawAllGridLines( dc
, wxRegion() ); 
8705 void wxGrid::SetCellHighlightColour( const wxColour
& colour 
) 
8707     if ( m_cellHighlightColour 
!= colour 
) 
8709         m_cellHighlightColour 
= colour
; 
8711         wxClientDC 
dc( m_gridWin 
); 
8713         wxGridCellAttr
* attr 
= GetCellAttr(m_currentCellCoords
); 
8714         DrawCellHighlight(dc
, attr
); 
8719 void wxGrid::SetCellHighlightPenWidth(int width
) 
8721     if (m_cellHighlightPenWidth 
!= width
) { 
8722         m_cellHighlightPenWidth 
= width
; 
8724         // Just redrawing the cell highlight is not enough since that won't 
8725         // make any visible change if the the thickness is getting smaller. 
8726         int row 
= m_currentCellCoords
.GetRow(); 
8727         int col 
= m_currentCellCoords
.GetCol(); 
8728         if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
8730         wxRect rect 
= CellToRect(row
, col
); 
8731         m_gridWin
->Refresh(TRUE
, &rect
); 
8735 void wxGrid::SetCellHighlightROPenWidth(int width
) 
8737     if (m_cellHighlightROPenWidth 
!= width
) { 
8738         m_cellHighlightROPenWidth 
= width
; 
8740         // Just redrawing the cell highlight is not enough since that won't 
8741         // make any visible change if the the thickness is getting smaller. 
8742         int row 
= m_currentCellCoords
.GetRow(); 
8743         int col 
= m_currentCellCoords
.GetCol(); 
8744         if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
8746         wxRect rect 
= CellToRect(row
, col
); 
8747         m_gridWin
->Refresh(TRUE
, &rect
); 
8751 void wxGrid::EnableGridLines( bool enable 
) 
8753     if ( enable 
!= m_gridLinesEnabled 
) 
8755         m_gridLinesEnabled 
= enable
; 
8757         if ( !GetBatchCount() ) 
8761                 wxClientDC 
dc( m_gridWin 
); 
8763                 DrawAllGridLines( dc
, wxRegion() ); 
8767                 m_gridWin
->Refresh(); 
8774 int wxGrid::GetDefaultRowSize() 
8776     return m_defaultRowHeight
; 
8779 int wxGrid::GetRowSize( int row 
) 
8781     wxCHECK_MSG( row 
>= 0 && row 
< m_numRows
, 0, _T("invalid row index") ); 
8783     return GetRowHeight(row
); 
8786 int wxGrid::GetDefaultColSize() 
8788     return m_defaultColWidth
; 
8791 int wxGrid::GetColSize( int col 
) 
8793     wxCHECK_MSG( col 
>= 0 && col 
< m_numCols
, 0, _T("invalid column index") ); 
8795     return GetColWidth(col
); 
8798 // ============================================================================ 
8799 // access to the grid attributes: each of them has a default value in the grid 
8800 // itself and may be overidden on a per-cell basis 
8801 // ============================================================================ 
8803 // ---------------------------------------------------------------------------- 
8804 // setting default attributes 
8805 // ---------------------------------------------------------------------------- 
8807 void wxGrid::SetDefaultCellBackgroundColour( const wxColour
& col 
) 
8809     m_defaultCellAttr
->SetBackgroundColour(col
); 
8811     m_gridWin
->SetBackgroundColour(col
); 
8815 void wxGrid::SetDefaultCellTextColour( const wxColour
& col 
) 
8817     m_defaultCellAttr
->SetTextColour(col
); 
8820 void wxGrid::SetDefaultCellAlignment( int horiz
, int vert 
) 
8822     m_defaultCellAttr
->SetAlignment(horiz
, vert
); 
8825 void wxGrid::SetDefaultCellOverflow( bool allow 
) 
8827     m_defaultCellAttr
->SetOverflow(allow
); 
8830 void wxGrid::SetDefaultCellFont( const wxFont
& font 
) 
8832     m_defaultCellAttr
->SetFont(font
); 
8835 void wxGrid::SetDefaultRenderer(wxGridCellRenderer 
*renderer
) 
8837     m_defaultCellAttr
->SetRenderer(renderer
); 
8840 void wxGrid::SetDefaultEditor(wxGridCellEditor 
*editor
) 
8842     m_defaultCellAttr
->SetEditor(editor
); 
8845 // ---------------------------------------------------------------------------- 
8846 // access to the default attrbiutes 
8847 // ---------------------------------------------------------------------------- 
8849 wxColour 
wxGrid::GetDefaultCellBackgroundColour() 
8851     return m_defaultCellAttr
->GetBackgroundColour(); 
8854 wxColour 
wxGrid::GetDefaultCellTextColour() 
8856     return m_defaultCellAttr
->GetTextColour(); 
8859 wxFont 
wxGrid::GetDefaultCellFont() 
8861     return m_defaultCellAttr
->GetFont(); 
8864 void wxGrid::GetDefaultCellAlignment( int *horiz
, int *vert 
) 
8866     m_defaultCellAttr
->GetAlignment(horiz
, vert
); 
8869 bool wxGrid::GetDefaultCellOverflow() 
8871     return m_defaultCellAttr
->GetOverflow(); 
8874 wxGridCellRenderer 
*wxGrid::GetDefaultRenderer() const 
8876     return m_defaultCellAttr
->GetRenderer(NULL
, 0, 0); 
8879 wxGridCellEditor 
*wxGrid::GetDefaultEditor() const 
8881     return m_defaultCellAttr
->GetEditor(NULL
,0,0); 
8884 // ---------------------------------------------------------------------------- 
8885 // access to cell attributes 
8886 // ---------------------------------------------------------------------------- 
8888 wxColour 
wxGrid::GetCellBackgroundColour(int row
, int col
) 
8890     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8891     wxColour colour 
= attr
->GetBackgroundColour(); 
8896 wxColour 
wxGrid::GetCellTextColour( int row
, int col 
) 
8898     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8899     wxColour colour 
= attr
->GetTextColour(); 
8904 wxFont 
wxGrid::GetCellFont( int row
, int col 
) 
8906     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8907     wxFont font 
= attr
->GetFont(); 
8912 void wxGrid::GetCellAlignment( int row
, int col
, int *horiz
, int *vert 
) 
8914     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8915     attr
->GetAlignment(horiz
, vert
); 
8919 bool wxGrid::GetCellOverflow( int row
, int col 
) 
8921     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8922     bool allow 
= attr
->GetOverflow(); 
8927 void wxGrid::GetCellSize( int row
, int col
, int *num_rows
, int *num_cols 
) 
8929     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8930     attr
->GetSize( num_rows
, num_cols 
); 
8934 wxGridCellRenderer
* wxGrid::GetCellRenderer(int row
, int col
) 
8936     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
8937     wxGridCellRenderer
* renderer 
= attr
->GetRenderer(this, row
, col
); 
8943 wxGridCellEditor
* wxGrid::GetCellEditor(int row
, int col
) 
8945     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
8946     wxGridCellEditor
* editor 
= attr
->GetEditor(this, row
, col
); 
8952 bool wxGrid::IsReadOnly(int row
, int col
) const 
8954     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
8955     bool isReadOnly 
= attr
->IsReadOnly(); 
8960 // ---------------------------------------------------------------------------- 
8961 // attribute support: cache, automatic provider creation, ... 
8962 // ---------------------------------------------------------------------------- 
8964 bool wxGrid::CanHaveAttributes() 
8971     return m_table
->CanHaveAttributes(); 
8974 void wxGrid::ClearAttrCache() 
8976     if ( m_attrCache
.row 
!= -1 ) 
8978         wxSafeDecRef(m_attrCache
.attr
); 
8979         m_attrCache
.attr 
= NULL
; 
8980         m_attrCache
.row 
= -1; 
8984 void wxGrid::CacheAttr(int row
, int col
, wxGridCellAttr 
*attr
) const 
8988         wxGrid 
*self 
= (wxGrid 
*)this;  // const_cast 
8990         self
->ClearAttrCache(); 
8991         self
->m_attrCache
.row 
= row
; 
8992         self
->m_attrCache
.col 
= col
; 
8993         self
->m_attrCache
.attr 
= attr
; 
8998 bool wxGrid::LookupAttr(int row
, int col
, wxGridCellAttr 
**attr
) const 
9000     if ( row 
== m_attrCache
.row 
&& col 
== m_attrCache
.col 
) 
9002         *attr 
= m_attrCache
.attr
; 
9003         wxSafeIncRef(m_attrCache
.attr
); 
9005 #ifdef DEBUG_ATTR_CACHE 
9006         gs_nAttrCacheHits
++; 
9013 #ifdef DEBUG_ATTR_CACHE 
9014         gs_nAttrCacheMisses
++; 
9020 wxGridCellAttr 
*wxGrid::GetCellAttr(int row
, int col
) const 
9022     wxGridCellAttr 
*attr 
= NULL
; 
9023     // Additional test to avoid looking at the cache e.g. for 
9024     // wxNoCellCoords, as this will confuse memory management. 
9027         if ( !LookupAttr(row
, col
, &attr
) ) 
9029             attr 
= m_table 
? m_table
->GetAttr(row
, col 
, wxGridCellAttr::Any
) 
9030                            : (wxGridCellAttr 
*)NULL
; 
9031             CacheAttr(row
, col
, attr
); 
9036         attr
->SetDefAttr(m_defaultCellAttr
); 
9040         attr 
= m_defaultCellAttr
; 
9047 wxGridCellAttr 
*wxGrid::GetOrCreateCellAttr(int row
, int col
) const 
9049     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
9051     wxCHECK_MSG( m_table
, attr
, 
9052                   _T("we may only be called if CanHaveAttributes() returned TRUE and then m_table should be !NULL") ); 
9054     attr 
= m_table
->GetAttr(row
, col
, wxGridCellAttr::Cell
); 
9057         attr 
= new wxGridCellAttr(m_defaultCellAttr
); 
9059         // artificially inc the ref count to match DecRef() in caller 
9061         m_table
->SetAttr(attr
, row
, col
); 
9067 // ---------------------------------------------------------------------------- 
9068 // setting column attributes (wrappers around SetColAttr) 
9069 // ---------------------------------------------------------------------------- 
9071 void wxGrid::SetColFormatBool(int col
) 
9073     SetColFormatCustom(col
, wxGRID_VALUE_BOOL
); 
9076 void wxGrid::SetColFormatNumber(int col
) 
9078     SetColFormatCustom(col
, wxGRID_VALUE_NUMBER
); 
9081 void wxGrid::SetColFormatFloat(int col
, int width
, int precision
) 
9083     wxString typeName 
= wxGRID_VALUE_FLOAT
; 
9084     if ( (width 
!= -1) || (precision 
!= -1) ) 
9086         typeName 
<< _T(':') << width 
<< _T(',') << precision
; 
9089     SetColFormatCustom(col
, typeName
); 
9092 void wxGrid::SetColFormatCustom(int col
, const wxString
& typeName
) 
9094     wxGridCellAttr 
*attr 
= m_table
->GetAttr(-1, col
, wxGridCellAttr::Col 
); 
9096         attr 
= new wxGridCellAttr
; 
9097     wxGridCellRenderer 
*renderer 
= GetDefaultRendererForType(typeName
); 
9098     attr
->SetRenderer(renderer
); 
9100     SetColAttr(col
, attr
); 
9104 // ---------------------------------------------------------------------------- 
9105 // setting cell attributes: this is forwarded to the table 
9106 // ---------------------------------------------------------------------------- 
9108 void wxGrid::SetAttr(int row
, int col
, wxGridCellAttr 
*attr
) 
9110     if ( CanHaveAttributes() ) 
9112         m_table
->SetAttr(attr
, row
, col
); 
9121 void wxGrid::SetRowAttr(int row
, wxGridCellAttr 
*attr
) 
9123     if ( CanHaveAttributes() ) 
9125         m_table
->SetRowAttr(attr
, row
); 
9134 void wxGrid::SetColAttr(int col
, wxGridCellAttr 
*attr
) 
9136     if ( CanHaveAttributes() ) 
9138         m_table
->SetColAttr(attr
, col
); 
9147 void wxGrid::SetCellBackgroundColour( int row
, int col
, const wxColour
& colour 
) 
9149     if ( CanHaveAttributes() ) 
9151         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9152         attr
->SetBackgroundColour(colour
); 
9157 void wxGrid::SetCellTextColour( int row
, int col
, const wxColour
& colour 
) 
9159     if ( CanHaveAttributes() ) 
9161         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9162         attr
->SetTextColour(colour
); 
9167 void wxGrid::SetCellFont( int row
, int col
, const wxFont
& font 
) 
9169     if ( CanHaveAttributes() ) 
9171         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9172         attr
->SetFont(font
); 
9177 void wxGrid::SetCellAlignment( int row
, int col
, int horiz
, int vert 
) 
9179     if ( CanHaveAttributes() ) 
9181         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9182         attr
->SetAlignment(horiz
, vert
); 
9187 void wxGrid::SetCellOverflow( int row
, int col
, bool allow 
) 
9189     if ( CanHaveAttributes() ) 
9191         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9192         attr
->SetOverflow(allow
); 
9197 void wxGrid::SetCellSize( int row
, int col
, int num_rows
, int num_cols 
) 
9199     if ( CanHaveAttributes() ) 
9201         int cell_rows
, cell_cols
; 
9203         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9204         attr
->GetSize(&cell_rows
, &cell_cols
); 
9205         attr
->SetSize(num_rows
, num_cols
); 
9208         // Cannot set the size of a cell to 0 or negative values 
9209         // While it is perfectly legal to do that, this function cannot 
9210         // handle all the possibilies, do it by hand by getting the CellAttr. 
9211         // You can only set the size of a cell to 1,1 or greater with this fn 
9212         wxASSERT_MSG( !((cell_rows 
< 1) || (cell_cols 
< 1)), 
9213                       wxT("wxGrid::SetCellSize setting cell size that is already part of another cell")); 
9214         wxASSERT_MSG( !((num_rows 
< 1) || (num_cols 
< 1)), 
9215                       wxT("wxGrid::SetCellSize setting cell size to < 1")); 
9217         // if this was already a multicell then "turn off" the other cells first 
9218         if ((cell_rows 
> 1) || (cell_rows 
> 1)) 
9221             for (j
=row
; j
<row
+cell_rows
; j
++) 
9223                 for (i
=col
; i
<col
+cell_cols
; i
++) 
9225                     if ((i 
!= col
) || (j 
!= row
)) 
9227                         wxGridCellAttr 
*attr_stub 
= GetOrCreateCellAttr(j
, i
); 
9228                         attr_stub
->SetSize( 1, 1 ); 
9229                         attr_stub
->DecRef(); 
9235         // mark the cells that will be covered by this cell to 
9236         // negative or zero values to point back at this cell 
9237         if (((num_rows 
> 1) || (num_cols 
> 1)) && (num_rows 
>= 1) && (num_cols 
>= 1)) 
9240             for (j
=row
; j
<row
+num_rows
; j
++) 
9242                 for (i
=col
; i
<col
+num_cols
; i
++) 
9244                     if ((i 
!= col
) || (j 
!= row
)) 
9246                         wxGridCellAttr 
*attr_stub 
= GetOrCreateCellAttr(j
, i
); 
9247                         attr_stub
->SetSize( row
-j
, col
-i 
); 
9248                         attr_stub
->DecRef(); 
9256 void wxGrid::SetCellRenderer(int row
, int col
, wxGridCellRenderer 
*renderer
) 
9258     if ( CanHaveAttributes() ) 
9260         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9261         attr
->SetRenderer(renderer
); 
9266 void wxGrid::SetCellEditor(int row
, int col
, wxGridCellEditor
* editor
) 
9268     if ( CanHaveAttributes() ) 
9270         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9271         attr
->SetEditor(editor
); 
9276 void wxGrid::SetReadOnly(int row
, int col
, bool isReadOnly
) 
9278     if ( CanHaveAttributes() ) 
9280         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9281         attr
->SetReadOnly(isReadOnly
); 
9286 // ---------------------------------------------------------------------------- 
9287 // Data type registration 
9288 // ---------------------------------------------------------------------------- 
9290 void wxGrid::RegisterDataType(const wxString
& typeName
, 
9291                               wxGridCellRenderer
* renderer
, 
9292                               wxGridCellEditor
* editor
) 
9294     m_typeRegistry
->RegisterDataType(typeName
, renderer
, editor
); 
9298 wxGridCellEditor
* wxGrid::GetDefaultEditorForCell(int row
, int col
) const 
9300     wxString typeName 
= m_table
->GetTypeName(row
, col
); 
9301     return GetDefaultEditorForType(typeName
); 
9304 wxGridCellRenderer
* wxGrid::GetDefaultRendererForCell(int row
, int col
) const 
9306     wxString typeName 
= m_table
->GetTypeName(row
, col
); 
9307     return GetDefaultRendererForType(typeName
); 
9311 wxGrid::GetDefaultEditorForType(const wxString
& typeName
) const 
9313     int index 
= m_typeRegistry
->FindOrCloneDataType(typeName
); 
9314     if ( index 
== wxNOT_FOUND 
) 
9316         wxFAIL_MSG(wxT("Unknown data type name")); 
9321     return m_typeRegistry
->GetEditor(index
); 
9325 wxGrid::GetDefaultRendererForType(const wxString
& typeName
) const 
9327     int index 
= m_typeRegistry
->FindOrCloneDataType(typeName
); 
9328     if ( index 
== wxNOT_FOUND 
) 
9330         wxFAIL_MSG(wxT("Unknown data type name")); 
9335     return m_typeRegistry
->GetRenderer(index
); 
9339 // ---------------------------------------------------------------------------- 
9341 // ---------------------------------------------------------------------------- 
9343 void wxGrid::EnableDragRowSize( bool enable 
) 
9345     m_canDragRowSize 
= enable
; 
9349 void wxGrid::EnableDragColSize( bool enable 
) 
9351     m_canDragColSize 
= enable
; 
9354 void wxGrid::EnableDragGridSize( bool enable 
) 
9356     m_canDragGridSize 
= enable
; 
9360 void wxGrid::SetDefaultRowSize( int height
, bool resizeExistingRows 
) 
9362     m_defaultRowHeight 
= wxMax( height
, m_minAcceptableRowHeight 
); 
9364     if ( resizeExistingRows 
) 
9366         // since we are resizing all rows to the default row size, 
9367         // we can simply clear the row heights and row bottoms 
9368         // arrays (which also allows us to take advantage of 
9369         // some speed optimisations) 
9370         m_rowHeights
.Empty(); 
9371         m_rowBottoms
.Empty(); 
9372         if ( !GetBatchCount() ) 
9377 void wxGrid::SetRowSize( int row
, int height 
) 
9379     wxCHECK_RET( row 
>= 0 && row 
< m_numRows
, _T("invalid row index") ); 
9381     // See comment in SetColSize 
9382     if ( height 
< GetRowMinimalAcceptableHeight()) { return; } 
9384     if ( m_rowHeights
.IsEmpty() ) 
9386         // need to really create the array 
9390     int h 
= wxMax( 0, height 
); 
9391     int diff 
= h 
- m_rowHeights
[row
]; 
9393     m_rowHeights
[row
] = h
; 
9395     for ( i 
= row
;  i 
< m_numRows
;  i
++ ) 
9397         m_rowBottoms
[i
] += diff
; 
9399     if ( !GetBatchCount() ) 
9403 void wxGrid::SetDefaultColSize( int width
, bool resizeExistingCols 
) 
9405     m_defaultColWidth 
= wxMax( width
, m_minAcceptableColWidth 
); 
9407     if ( resizeExistingCols 
) 
9409         // since we are resizing all columns to the default column size, 
9410         // we can simply clear the col widths and col rights 
9411         // arrays (which also allows us to take advantage of 
9412         // some speed optimisations) 
9413         m_colWidths
.Empty(); 
9414         m_colRights
.Empty(); 
9415         if ( !GetBatchCount() ) 
9420 void wxGrid::SetColSize( int col
, int width 
) 
9422     wxCHECK_RET( col 
>= 0 && col 
< m_numCols
, _T("invalid column index") ); 
9424     // should we check that it's bigger than GetColMinimalWidth(col) here? 
9426     // No, because it is reasonable to assume the library user know's 
9427     // what he is doing. However whe should test against the weaker 
9428     // constariant of minimalAcceptableWidth, as this breaks rendering 
9430     // This test then fixes sf.net bug #645734 
9432     if ( width 
< GetColMinimalAcceptableWidth()) { return; } 
9434     if ( m_colWidths
.IsEmpty() ) 
9436         // need to really create the array 
9440     // if < 0 calc new width from label 
9444       wxArrayString lines
; 
9445       wxClientDC 
dc(m_colLabelWin
); 
9446       dc
.SetFont(GetLabelFont()); 
9447       StringToLines(GetColLabelValue(col
), lines
); 
9448       GetTextBoxSize(dc
, lines
, &w
, &h
); 
9451     int w 
= wxMax( 0, width 
); 
9452     int diff 
= w 
- m_colWidths
[col
]; 
9453     m_colWidths
[col
] = w
; 
9456     for ( i 
= col
;  i 
< m_numCols
;  i
++ ) 
9458         m_colRights
[i
] += diff
; 
9460     if ( !GetBatchCount() ) 
9465 void wxGrid::SetColMinimalWidth( int col
, int width 
) 
9467     if (width 
> GetColMinimalAcceptableWidth()) { 
9468         m_colMinWidths
[col
] = width
; 
9472 void wxGrid::SetRowMinimalHeight( int row
, int width 
) 
9474     if (width 
> GetRowMinimalAcceptableHeight()) { 
9475        m_rowMinHeights
[row
] = width
; 
9479 int wxGrid::GetColMinimalWidth(int col
) const 
9481     wxLongToLongHashMap::const_iterator it 
= m_colMinWidths
.find(col
); 
9482     return it 
!= m_colMinWidths
.end() ? (int)it
->second 
: m_minAcceptableColWidth
; 
9485 int wxGrid::GetRowMinimalHeight(int row
) const 
9487     wxLongToLongHashMap::const_iterator it 
= m_rowMinHeights
.find(row
); 
9488     return it 
!= m_rowMinHeights
.end() ? (int)it
->second 
: m_minAcceptableRowHeight
; 
9491 void wxGrid::SetColMinimalAcceptableWidth( int width 
) 
9493     // We do allow a width of 0 since this gives us 
9494     // an easy way to temporarily hidding columns. 
9497     m_minAcceptableColWidth 
= width
; 
9500 void wxGrid::SetRowMinimalAcceptableHeight( int height 
) 
9502     // We do allow a height of 0 since this gives us 
9503     // an easy way to temporarily hidding rows. 
9506     m_minAcceptableRowHeight 
= height
; 
9509 int  wxGrid::GetColMinimalAcceptableWidth() const 
9511     return m_minAcceptableColWidth
; 
9514 int  wxGrid::GetRowMinimalAcceptableHeight() const 
9516     return m_minAcceptableRowHeight
; 
9519 // ---------------------------------------------------------------------------- 
9521 // ---------------------------------------------------------------------------- 
9523 void wxGrid::AutoSizeColOrRow( int colOrRow
, bool setAsMin
, bool column 
) 
9525     wxClientDC 
dc(m_gridWin
); 
9527     // init both of them to avoid compiler warnings, even if weo nly need one 
9535     wxCoord extent
, extentMax 
= 0; 
9536     int max 
= column 
? m_numRows 
: m_numCols
; 
9537     for ( int rowOrCol 
= 0; rowOrCol 
< max
; rowOrCol
++ ) 
9544         wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
9545         wxGridCellRenderer
* renderer 
= attr
->GetRenderer(this, row
, col
); 
9548             wxSize size 
= renderer
->GetBestSize(*this, *attr
, dc
, row
, col
); 
9549             extent 
= column 
? size
.x 
: size
.y
; 
9550             if ( extent 
> extentMax 
) 
9561     // now also compare with the column label extent 
9563     dc
.SetFont( GetLabelFont() ); 
9567         dc
.GetTextExtent( GetColLabelValue(col
), &w
, &h 
); 
9568         if( GetColLabelTextOrientation() == wxVERTICAL 
) 
9572         dc
.GetTextExtent( GetRowLabelValue(row
), &w
, &h 
); 
9574     extent 
= column 
? w 
: h
; 
9575     if ( extent 
> extentMax 
) 
9582         // empty column - give default extent (notice that if extentMax is less 
9583         // than default extent but != 0, it's ok) 
9584         extentMax 
= column 
? m_defaultColWidth 
: m_defaultRowHeight
; 
9590             // leave some space around text 
9601         SetColSize(col
, extentMax
); 
9602         if ( !GetBatchCount() ) 
9605             m_gridWin
->GetClientSize( &cw
, &ch 
); 
9606             wxRect 
rect ( CellToRect( 0, col 
) ); 
9608             CalcScrolledPosition(rect
.x
, 0, &rect
.x
, &dummy
); 
9609             rect
.width 
= cw 
- rect
.x
; 
9610             rect
.height 
= m_colLabelHeight
; 
9611             m_colLabelWin
->Refresh( TRUE
, &rect 
); 
9616         SetRowSize(row
, extentMax
); 
9617         if ( !GetBatchCount() ) 
9620             m_gridWin
->GetClientSize( &cw
, &ch 
); 
9621             wxRect 
rect ( CellToRect( row
, 0 ) ); 
9623             CalcScrolledPosition(0, rect
.y
, &dummy
, &rect
.y
); 
9624             rect
.width 
= m_rowLabelWidth
; 
9625             rect
.height 
= ch 
- rect
.y
; 
9626             m_rowLabelWin
->Refresh( TRUE
, &rect 
); 
9632             SetColMinimalWidth(col
, extentMax
); 
9634             SetRowMinimalHeight(row
, extentMax
); 
9638 int wxGrid::SetOrCalcColumnSizes(bool calcOnly
, bool setAsMin
) 
9640     int width 
= m_rowLabelWidth
; 
9645     for ( int col 
= 0; col 
< m_numCols
; col
++ ) 
9649             AutoSizeColumn(col
, setAsMin
); 
9652         width 
+= GetColWidth(col
); 
9661 int wxGrid::SetOrCalcRowSizes(bool calcOnly
, bool setAsMin
) 
9663     int height 
= m_colLabelHeight
; 
9668     for ( int row 
= 0; row 
< m_numRows
; row
++ ) 
9672             AutoSizeRow(row
, setAsMin
); 
9675         height 
+= GetRowHeight(row
); 
9684 void wxGrid::AutoSize() 
9688     wxSize 
size(SetOrCalcColumnSizes(FALSE
), SetOrCalcRowSizes(FALSE
)); 
9690     // round up the size to a multiple of scroll step - this ensures that we 
9691     // won't get the scrollbars if we're sized exactly to this width 
9692     // CalcDimension adds m_extraWidth + 1 etc. to calculate the necessary 
9694     wxSize 
sizeFit(GetScrollX(size
.x 
+ m_extraWidth 
+ 1) * GRID_SCROLL_LINE_X
, 
9695                    GetScrollY(size
.y 
+ m_extraHeight 
+ 1) * GRID_SCROLL_LINE_Y
); 
9697     // distribute the extra space between the columns/rows to avoid having 
9698     // extra white space 
9700     // Remove the extra m_extraWidth + 1 added above 
9701     wxCoord diff 
= sizeFit
.x 
- size
.x 
+ (m_extraWidth 
+ 1); 
9702     if ( diff 
&& m_numCols 
) 
9704         // try to resize the columns uniformly 
9705         wxCoord diffPerCol 
= diff 
/ m_numCols
; 
9708             for ( int col 
= 0; col 
< m_numCols
; col
++ ) 
9710                 SetColSize(col
, GetColWidth(col
) + diffPerCol
); 
9714         // add remaining amount to the last columns 
9715         diff 
-= diffPerCol 
* m_numCols
; 
9718             for ( int col 
= m_numCols 
- 1; col 
>= m_numCols 
- diff
; col
-- ) 
9720                 SetColSize(col
, GetColWidth(col
) + 1); 
9726     diff 
= sizeFit
.y 
- size
.y 
- (m_extraHeight 
+ 1); 
9727     if ( diff 
&& m_numRows 
) 
9729         // try to resize the columns uniformly 
9730         wxCoord diffPerRow 
= diff 
/ m_numRows
; 
9733             for ( int row 
= 0; row 
< m_numRows
; row
++ ) 
9735                 SetRowSize(row
, GetRowHeight(row
) + diffPerRow
); 
9739         // add remaining amount to the last rows 
9740         diff 
-= diffPerRow 
* m_numRows
; 
9743             for ( int row 
= m_numRows 
- 1; row 
>= m_numRows 
- diff
; row
-- ) 
9745                 SetRowSize(row
, GetRowHeight(row
) + 1); 
9752     SetClientSize(sizeFit
); 
9755 void wxGrid::AutoSizeRowLabelSize( int row 
) 
9757     wxArrayString lines
; 
9760     // Hide the edit control, so it 
9761     // won't interfer with drag-shrinking. 
9762     if( IsCellEditControlShown() ) 
9764         HideCellEditControl(); 
9765         SaveEditControlValue(); 
9768     // autosize row height depending on label text 
9769     StringToLines( GetRowLabelValue( row 
), lines 
); 
9770     wxClientDC 
dc( m_rowLabelWin 
); 
9771     GetTextBoxSize( dc
, lines
, &w
, &h
); 
9772     if( h 
< m_defaultRowHeight 
) 
9773         h 
= m_defaultRowHeight
; 
9778 void wxGrid::AutoSizeColLabelSize( int col 
) 
9780     wxArrayString lines
; 
9783     // Hide the edit control, so it 
9784     // won't interfer with drag-shrinking. 
9785     if( IsCellEditControlShown() ) 
9787         HideCellEditControl(); 
9788         SaveEditControlValue(); 
9791     // autosize column width depending on label text 
9792     StringToLines( GetColLabelValue( col 
), lines 
); 
9793     wxClientDC 
dc( m_colLabelWin 
); 
9794     if( GetColLabelTextOrientation() == wxHORIZONTAL 
) 
9795         GetTextBoxSize( dc
, lines
, &w
, &h
); 
9797         GetTextBoxSize( dc
, lines
, &h
, &w
); 
9798     if( w 
< m_defaultColWidth 
) 
9799         w 
= m_defaultColWidth
; 
9804 wxSize 
wxGrid::DoGetBestSize() const 
9806     // don't set sizes, only calculate them 
9807     wxGrid 
*self 
= (wxGrid 
*)this;  // const_cast 
9810     width 
= self
->SetOrCalcColumnSizes(TRUE
); 
9811     height 
= self
->SetOrCalcRowSizes(TRUE
); 
9813     int maxwidth
, maxheight
; 
9814     wxDisplaySize( & maxwidth
, & maxheight 
); 
9816     if ( width 
> maxwidth 
) width 
= maxwidth
; 
9817     if ( height 
> maxheight 
) height 
= maxheight
; 
9819     return wxSize( width
, height 
); 
9828 wxPen
& wxGrid::GetDividerPen() const 
9833 // ---------------------------------------------------------------------------- 
9834 // cell value accessor functions 
9835 // ---------------------------------------------------------------------------- 
9837 void wxGrid::SetCellValue( int row
, int col
, const wxString
& s 
) 
9841         m_table
->SetValue( row
, col
, s 
); 
9842         if ( !GetBatchCount() ) 
9845             wxRect 
rect( CellToRect( row
, col 
) ); 
9847             rect
.width 
= m_gridWin
->GetClientSize().GetWidth(); 
9848             CalcScrolledPosition(0, rect
.y
, &dummy
, &rect
.y
); 
9849             m_gridWin
->Refresh( FALSE
, &rect 
); 
9852         if ( m_currentCellCoords
.GetRow() == row 
&& 
9853              m_currentCellCoords
.GetCol() == col 
&& 
9854              IsCellEditControlShown()) 
9855              // Note: If we are using IsCellEditControlEnabled, 
9856              // this interacts badly with calling SetCellValue from 
9857              // an EVT_GRID_CELL_CHANGE handler. 
9859             HideCellEditControl(); 
9860             ShowCellEditControl(); // will reread data from table 
9867 // ------ Block, row and col selection 
9870 void wxGrid::SelectRow( int row
, bool addToSelected 
) 
9872     if ( IsSelection() && !addToSelected 
) 
9876         m_selection
->SelectRow( row
, FALSE
, addToSelected 
); 
9880 void wxGrid::SelectCol( int col
, bool addToSelected 
) 
9882     if ( IsSelection() && !addToSelected 
) 
9886         m_selection
->SelectCol( col
, FALSE
, addToSelected 
); 
9890 void wxGrid::SelectBlock( int topRow
, int leftCol
, int bottomRow
, int rightCol
, 
9891                           bool addToSelected 
) 
9893     if ( IsSelection() && !addToSelected 
) 
9897         m_selection
->SelectBlock( topRow
, leftCol
, bottomRow
, rightCol
, 
9898                                   FALSE
, addToSelected 
); 
9902 void wxGrid::SelectAll() 
9904     if ( m_numRows 
> 0 && m_numCols 
> 0 ) 
9907             m_selection
->SelectBlock( 0, 0, m_numRows
-1, m_numCols
-1 ); 
9912 // ------ Cell, row and col deselection 
9915 void wxGrid::DeselectRow( int row 
) 
9920     if ( m_selection
->GetSelectionMode() == wxGrid::wxGridSelectRows 
) 
9922         if ( m_selection
->IsInSelection(row
, 0 ) ) 
9923             m_selection
->ToggleCellSelection( row
, 0); 
9927         int nCols 
= GetNumberCols(); 
9928         for ( int i 
= 0; i 
< nCols 
; i
++ ) 
9930             if ( m_selection
->IsInSelection(row
, i 
) ) 
9931                 m_selection
->ToggleCellSelection( row
, i
); 
9936 void wxGrid::DeselectCol( int col 
) 
9941     if ( m_selection
->GetSelectionMode() == wxGrid::wxGridSelectColumns 
) 
9943         if ( m_selection
->IsInSelection(0, col 
) ) 
9944             m_selection
->ToggleCellSelection( 0, col
); 
9948         int nRows 
= GetNumberRows(); 
9949         for ( int i 
= 0; i 
< nRows 
; i
++ ) 
9951             if ( m_selection
->IsInSelection(i
, col 
) ) 
9952                 m_selection
->ToggleCellSelection(i
, col
); 
9957 void wxGrid::DeselectCell( int row
, int col 
) 
9959     if ( m_selection 
&& m_selection
->IsInSelection(row
, col
) ) 
9960         m_selection
->ToggleCellSelection(row
, col
); 
9963 bool wxGrid::IsSelection() 
9965     return ( m_selection 
&& (m_selection
->IsSelection() || 
9966              ( m_selectingTopLeft 
!= wxGridNoCellCoords 
&& 
9967                m_selectingBottomRight 
!= wxGridNoCellCoords
) ) ); 
9970 bool wxGrid::IsInSelection( int row
, int col 
) const 
9972     return ( m_selection 
&& (m_selection
->IsInSelection( row
, col 
) || 
9973              ( row 
>= m_selectingTopLeft
.GetRow() && 
9974                col 
>= m_selectingTopLeft
.GetCol() && 
9975                row 
<= m_selectingBottomRight
.GetRow() && 
9976                col 
<= m_selectingBottomRight
.GetCol() )) ); 
9979 wxGridCellCoordsArray 
wxGrid::GetSelectedCells() const 
9981     if (!m_selection
) { wxGridCellCoordsArray a
; return a
; } 
9982     return m_selection
->m_cellSelection
; 
9984 wxGridCellCoordsArray 
wxGrid::GetSelectionBlockTopLeft() const 
9986     if (!m_selection
) { wxGridCellCoordsArray a
; return a
; } 
9987     return m_selection
->m_blockSelectionTopLeft
; 
9989 wxGridCellCoordsArray 
wxGrid::GetSelectionBlockBottomRight() const 
9991     if (!m_selection
) { wxGridCellCoordsArray a
; return a
; } 
9992     return m_selection
->m_blockSelectionBottomRight
; 
9994 wxArrayInt 
wxGrid::GetSelectedRows() const 
9996     if (!m_selection
) { wxArrayInt a
; return a
; } 
9997     return m_selection
->m_rowSelection
; 
9999 wxArrayInt 
wxGrid::GetSelectedCols() const 
10001     if (!m_selection
) { wxArrayInt a
; return a
; } 
10002     return m_selection
->m_colSelection
; 
10006 void wxGrid::ClearSelection() 
10008     m_selectingTopLeft 
= wxGridNoCellCoords
; 
10009     m_selectingBottomRight 
= wxGridNoCellCoords
; 
10011         m_selection
->ClearSelection(); 
10015 // This function returns the rectangle that encloses the given block 
10016 // in device coords clipped to the client size of the grid window. 
10018 wxRect 
wxGrid::BlockToDeviceRect( const wxGridCellCoords 
&topLeft
, 
10019                                   const wxGridCellCoords 
&bottomRight 
) 
10021     wxRect 
rect( wxGridNoCellRect 
); 
10024     cellRect 
= CellToRect( topLeft 
); 
10025     if ( cellRect 
!= wxGridNoCellRect 
) 
10031         rect 
= wxRect( 0, 0, 0, 0 ); 
10034     cellRect 
= CellToRect( bottomRight 
); 
10035     if ( cellRect 
!= wxGridNoCellRect 
) 
10041         return wxGridNoCellRect
; 
10045     int left 
= rect
.GetLeft(); 
10046     int top 
= rect
.GetTop(); 
10047     int right 
= rect
.GetRight(); 
10048     int bottom 
= rect
.GetBottom(); 
10050     int leftCol 
= topLeft
.GetCol(); 
10051     int topRow 
= topLeft
.GetRow(); 
10052     int rightCol 
= bottomRight
.GetCol(); 
10053     int bottomRow 
= bottomRight
.GetRow(); 
10071         topRow 
= bottomRow
; 
10076     for ( j 
= topRow
; j 
<= bottomRow
; j
++ ) 
10078         for ( i 
= leftCol
; i 
<= rightCol
; i
++ ) 
10080             if ((j
==topRow
) || (j
==bottomRow
) || (i
==leftCol
) || (i
==rightCol
)) 
10082                 cellRect 
= CellToRect( j
, i 
); 
10084                 if (cellRect
.x 
< left
) 
10086                 if (cellRect
.y 
< top
) 
10088                 if (cellRect
.x 
+ cellRect
.width 
> right
) 
10089                     right 
= cellRect
.x 
+ cellRect
.width
; 
10090                 if (cellRect
.y 
+ cellRect
.height 
> bottom
) 
10091                     bottom 
= cellRect
.y 
+ cellRect
.height
; 
10093             else i 
= rightCol
; // jump over inner cells. 
10097     // convert to scrolled coords 
10099     CalcScrolledPosition( left
, top
, &left
, &top 
); 
10100     CalcScrolledPosition( right
, bottom
, &right
, &bottom 
); 
10103     m_gridWin
->GetClientSize( &cw
, &ch 
); 
10105     if (right 
< 0 || bottom 
< 0 || left 
> cw 
|| top 
> ch
) 
10106         return wxRect( 0, 0, 0, 0); 
10108     rect
.SetLeft( wxMax(0, left
) ); 
10109     rect
.SetTop( wxMax(0, top
) ); 
10110     rect
.SetRight( wxMin(cw
, right
) ); 
10111     rect
.SetBottom( wxMin(ch
, bottom
) ); 
10117 // ------ Grid event classes 
10120 IMPLEMENT_DYNAMIC_CLASS( wxGridEvent
, wxNotifyEvent 
) 
10122 wxGridEvent::wxGridEvent( int id
, wxEventType type
, wxObject
* obj
, 
10123                           int row
, int col
, int x
, int y
, bool sel
, 
10124                           bool control
, bool shift
, bool alt
, bool meta 
) 
10125         : wxNotifyEvent( type
, id 
) 
10132     m_control 
= control
; 
10137     SetEventObject(obj
); 
10141 IMPLEMENT_DYNAMIC_CLASS( wxGridSizeEvent
, wxNotifyEvent 
) 
10143 wxGridSizeEvent::wxGridSizeEvent( int id
, wxEventType type
, wxObject
* obj
, 
10144                                   int rowOrCol
, int x
, int y
, 
10145                                   bool control
, bool shift
, bool alt
, bool meta 
) 
10146         : wxNotifyEvent( type
, id 
) 
10148     m_rowOrCol 
= rowOrCol
; 
10151     m_control 
= control
; 
10156     SetEventObject(obj
); 
10160 IMPLEMENT_DYNAMIC_CLASS( wxGridRangeSelectEvent
, wxNotifyEvent 
) 
10162 wxGridRangeSelectEvent::wxGridRangeSelectEvent(int id
, wxEventType type
, wxObject
* obj
, 
10163                                                const wxGridCellCoords
& topLeft
, 
10164                                                const wxGridCellCoords
& bottomRight
, 
10165                                                bool sel
, bool control
, 
10166                                                bool shift
, bool alt
, bool meta 
) 
10167         : wxNotifyEvent( type
, id 
) 
10169     m_topLeft     
= topLeft
; 
10170     m_bottomRight 
= bottomRight
; 
10172     m_control     
= control
; 
10177     SetEventObject(obj
); 
10181 IMPLEMENT_DYNAMIC_CLASS(wxGridEditorCreatedEvent
, wxCommandEvent
) 
10183 wxGridEditorCreatedEvent::wxGridEditorCreatedEvent(int id
, wxEventType type
, 
10184                                                    wxObject
* obj
, int row
, 
10185                                                    int col
, wxControl
* ctrl
) 
10186     : wxCommandEvent(type
, id
) 
10188     SetEventObject(obj
); 
10194 #endif // wxUSE_GRID