1 /////////////////////////////////////////////////////////////////////////// 
   2 // Name:        generic/grid.cpp 
   3 // Purpose:     wxGrid and related classes 
   4 // Author:      Michael Bedward (based on code by Julian Smart, Robin Dunn) 
   5 // Modified by: Robin Dunn, Vadim Zeitlin 
   8 // Copyright:   (c) Michael Bedward (mbedward@ozemail.com.au) 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  21     #pragma implementation "grid.h" 
  24 // For compilers that support precompilatixon, includes "wx/wx.h". 
  25 #include "wx/wxprec.h" 
  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" 
  46 #include "wx/textfile.h" 
  47 #include "wx/spinctrl.h" 
  48 #include "wx/tokenzr.h" 
  51 #include "wx/generic/gridsel.h" 
  53 #if defined(__WXMOTIF__) 
  54     #define WXUNUSED_MOTIF(identifier)  WXUNUSED(identifier) 
  56     #define WXUNUSED_MOTIF(identifier)  identifier 
  59 #if defined(__WXGTK__) 
  60     #define WXUNUSED_GTK(identifier)    WXUNUSED(identifier) 
  62     #define WXUNUSED_GTK(identifier)    identifier 
  65 // Required for wxIs... functions 
  68 // ---------------------------------------------------------------------------- 
  70 // ---------------------------------------------------------------------------- 
  72 WX_DEFINE_EXPORTED_ARRAY(wxGridCellAttr 
*, wxArrayAttrs
); 
  74 struct wxGridCellWithAttr
 
  76     wxGridCellWithAttr(int row
, int col
, wxGridCellAttr 
*attr_
) 
  77         : coords(row
, col
), attr(attr_
) 
  86     wxGridCellCoords coords
; 
  90 //  DECLARE_NO_COPY_CLASS(wxGridCellWithAttr) 
  91 // without rewriting the macros, which require a public copy constructor. 
  94 WX_DECLARE_EXPORTED_OBJARRAY(wxGridCellWithAttr
, wxGridCellWithAttrArray
); 
  96 #include "wx/arrimpl.cpp" 
  98 WX_DEFINE_OBJARRAY(wxGridCellCoordsArray
) 
  99 WX_DEFINE_OBJARRAY(wxGridCellWithAttrArray
) 
 101 // ---------------------------------------------------------------------------- 
 103 // ---------------------------------------------------------------------------- 
 105 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_LEFT_CLICK
) 
 106 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_RIGHT_CLICK
) 
 107 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_LEFT_DCLICK
) 
 108 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_RIGHT_DCLICK
) 
 109 DEFINE_EVENT_TYPE(wxEVT_GRID_LABEL_LEFT_CLICK
) 
 110 DEFINE_EVENT_TYPE(wxEVT_GRID_LABEL_RIGHT_CLICK
) 
 111 DEFINE_EVENT_TYPE(wxEVT_GRID_LABEL_LEFT_DCLICK
) 
 112 DEFINE_EVENT_TYPE(wxEVT_GRID_LABEL_RIGHT_DCLICK
) 
 113 DEFINE_EVENT_TYPE(wxEVT_GRID_ROW_SIZE
) 
 114 DEFINE_EVENT_TYPE(wxEVT_GRID_COL_SIZE
) 
 115 DEFINE_EVENT_TYPE(wxEVT_GRID_RANGE_SELECT
) 
 116 DEFINE_EVENT_TYPE(wxEVT_GRID_CELL_CHANGE
) 
 117 DEFINE_EVENT_TYPE(wxEVT_GRID_SELECT_CELL
) 
 118 DEFINE_EVENT_TYPE(wxEVT_GRID_EDITOR_SHOWN
) 
 119 DEFINE_EVENT_TYPE(wxEVT_GRID_EDITOR_HIDDEN
) 
 120 DEFINE_EVENT_TYPE(wxEVT_GRID_EDITOR_CREATED
) 
 122 // ---------------------------------------------------------------------------- 
 124 // ---------------------------------------------------------------------------- 
 126 class WXDLLEXPORT wxGridRowLabelWindow 
: public wxWindow
 
 129     wxGridRowLabelWindow() { m_owner 
= (wxGrid 
*)NULL
; } 
 130     wxGridRowLabelWindow( wxGrid 
*parent
, wxWindowID id
, 
 131                           const wxPoint 
&pos
, const wxSize 
&size 
); 
 136     void OnPaint( wxPaintEvent
& event 
); 
 137     void OnMouseEvent( wxMouseEvent
& event 
); 
 138     void OnMouseWheel( wxMouseEvent
& event 
); 
 139     void OnKeyDown( wxKeyEvent
& event 
); 
 140     void OnKeyUp( wxKeyEvent
& ); 
 142     DECLARE_DYNAMIC_CLASS(wxGridRowLabelWindow
) 
 143     DECLARE_EVENT_TABLE() 
 144     DECLARE_NO_COPY_CLASS(wxGridRowLabelWindow
) 
 148 class WXDLLEXPORT wxGridColLabelWindow 
: public wxWindow
 
 151     wxGridColLabelWindow() { m_owner 
= (wxGrid 
*)NULL
; } 
 152     wxGridColLabelWindow( wxGrid 
*parent
, wxWindowID id
, 
 153                           const wxPoint 
&pos
, const wxSize 
&size 
); 
 158     void OnPaint( wxPaintEvent 
&event 
); 
 159     void OnMouseEvent( wxMouseEvent
& event 
); 
 160     void OnMouseWheel( wxMouseEvent
& event 
); 
 161     void OnKeyDown( wxKeyEvent
& event 
); 
 162     void OnKeyUp( wxKeyEvent
& ); 
 164     DECLARE_DYNAMIC_CLASS(wxGridColLabelWindow
) 
 165     DECLARE_EVENT_TABLE() 
 166     DECLARE_NO_COPY_CLASS(wxGridColLabelWindow
) 
 170 class WXDLLEXPORT wxGridCornerLabelWindow 
: public wxWindow
 
 173     wxGridCornerLabelWindow() { m_owner 
= (wxGrid 
*)NULL
; } 
 174     wxGridCornerLabelWindow( wxGrid 
*parent
, wxWindowID id
, 
 175                              const wxPoint 
&pos
, const wxSize 
&size 
); 
 180     void OnMouseEvent( wxMouseEvent
& event 
); 
 181     void OnMouseWheel( wxMouseEvent
& event 
); 
 182     void OnKeyDown( wxKeyEvent
& event 
); 
 183     void OnKeyUp( wxKeyEvent
& ); 
 184     void OnPaint( wxPaintEvent
& event 
); 
 186     DECLARE_DYNAMIC_CLASS(wxGridCornerLabelWindow
) 
 187     DECLARE_EVENT_TABLE() 
 188     DECLARE_NO_COPY_CLASS(wxGridCornerLabelWindow
) 
 191 class WXDLLEXPORT wxGridWindow 
: public wxWindow
 
 196         m_owner 
= (wxGrid 
*)NULL
; 
 197         m_rowLabelWin 
= (wxGridRowLabelWindow 
*)NULL
; 
 198         m_colLabelWin 
= (wxGridColLabelWindow 
*)NULL
; 
 201     wxGridWindow( wxGrid 
*parent
, 
 202                   wxGridRowLabelWindow 
*rowLblWin
, 
 203                   wxGridColLabelWindow 
*colLblWin
, 
 204                   wxWindowID id
, const wxPoint 
&pos
, const wxSize 
&size 
); 
 207     void ScrollWindow( int dx
, int dy
, const wxRect 
*rect 
); 
 209     wxGrid
* GetOwner() { return m_owner
; } 
 213     wxGridRowLabelWindow     
*m_rowLabelWin
; 
 214     wxGridColLabelWindow     
*m_colLabelWin
; 
 216     void OnPaint( wxPaintEvent 
&event 
); 
 217     void OnMouseWheel( wxMouseEvent
& event 
); 
 218     void OnMouseEvent( wxMouseEvent
& event 
); 
 219     void OnKeyDown( wxKeyEvent
& ); 
 220     void OnKeyUp( wxKeyEvent
& ); 
 221     void OnEraseBackground( wxEraseEvent
& ); 
 224     DECLARE_DYNAMIC_CLASS(wxGridWindow
) 
 225     DECLARE_EVENT_TABLE() 
 226     DECLARE_NO_COPY_CLASS(wxGridWindow
) 
 231 class wxGridCellEditorEvtHandler 
: public wxEvtHandler
 
 234     wxGridCellEditorEvtHandler() 
 235         : m_grid(0), m_editor(0) 
 237     wxGridCellEditorEvtHandler(wxGrid
* grid
, wxGridCellEditor
* editor
) 
 238         : m_grid(grid
), m_editor(editor
) 
 241     void OnKeyDown(wxKeyEvent
& event
); 
 242     void OnChar(wxKeyEvent
& event
); 
 246     wxGridCellEditor
*   m_editor
; 
 247     DECLARE_DYNAMIC_CLASS(wxGridCellEditorEvtHandler
) 
 248     DECLARE_EVENT_TABLE() 
 249     DECLARE_NO_COPY_CLASS(wxGridCellEditorEvtHandler
) 
 253 IMPLEMENT_DYNAMIC_CLASS( wxGridCellEditorEvtHandler
, wxEvtHandler 
) 
 254 BEGIN_EVENT_TABLE( wxGridCellEditorEvtHandler
, wxEvtHandler 
) 
 255     EVT_KEY_DOWN( wxGridCellEditorEvtHandler::OnKeyDown 
) 
 256     EVT_CHAR( wxGridCellEditorEvtHandler::OnChar 
) 
 261 // ---------------------------------------------------------------------------- 
 262 // the internal data representation used by wxGridCellAttrProvider 
 263 // ---------------------------------------------------------------------------- 
 265 // this class stores attributes set for cells 
 266 class WXDLLEXPORT wxGridCellAttrData
 
 269     void SetAttr(wxGridCellAttr 
*attr
, int row
, int col
); 
 270     wxGridCellAttr 
*GetAttr(int row
, int col
) const; 
 271     void UpdateAttrRows( size_t pos
, int numRows 
); 
 272     void UpdateAttrCols( size_t pos
, int numCols 
); 
 275     // searches for the attr for given cell, returns wxNOT_FOUND if not found 
 276     int FindIndex(int row
, int col
) const; 
 278     wxGridCellWithAttrArray m_attrs
; 
 281 // this class stores attributes set for rows or columns 
 282 class WXDLLEXPORT wxGridRowOrColAttrData
 
 285     // empty ctor to suppress warnings 
 286     wxGridRowOrColAttrData() { } 
 287     ~wxGridRowOrColAttrData(); 
 289     void SetAttr(wxGridCellAttr 
*attr
, int rowOrCol
); 
 290     wxGridCellAttr 
*GetAttr(int rowOrCol
) const; 
 291     void UpdateAttrRowsOrCols( size_t pos
, int numRowsOrCols 
); 
 294     wxArrayInt m_rowsOrCols
; 
 295     wxArrayAttrs m_attrs
; 
 298 // NB: this is just a wrapper around 3 objects: one which stores cell 
 299 //     attributes, and 2 others for row/col ones 
 300 class WXDLLEXPORT wxGridCellAttrProviderData
 
 303     wxGridCellAttrData m_cellAttrs
; 
 304     wxGridRowOrColAttrData m_rowAttrs
, 
 309 // ---------------------------------------------------------------------------- 
 310 // data structures used for the data type registry 
 311 // ---------------------------------------------------------------------------- 
 313 struct wxGridDataTypeInfo
 
 315     wxGridDataTypeInfo(const wxString
& typeName
, 
 316                        wxGridCellRenderer
* renderer
, 
 317                        wxGridCellEditor
* editor
) 
 318         : m_typeName(typeName
), m_renderer(renderer
), m_editor(editor
) 
 321     ~wxGridDataTypeInfo() 
 323         wxSafeDecRef(m_renderer
); 
 324         wxSafeDecRef(m_editor
); 
 328     wxGridCellRenderer
* m_renderer
; 
 329     wxGridCellEditor
*   m_editor
; 
 331     DECLARE_NO_COPY_CLASS(wxGridDataTypeInfo
) 
 335 WX_DEFINE_EXPORTED_ARRAY(wxGridDataTypeInfo
*, wxGridDataTypeInfoArray
); 
 338 class WXDLLEXPORT wxGridTypeRegistry
 
 341   wxGridTypeRegistry() {} 
 342     ~wxGridTypeRegistry(); 
 344     void RegisterDataType(const wxString
& typeName
, 
 345                      wxGridCellRenderer
* renderer
, 
 346                      wxGridCellEditor
* editor
); 
 348     // find one of already registered data types 
 349     int FindRegisteredDataType(const wxString
& typeName
); 
 351     // try to FindRegisteredDataType(), if this fails and typeName is one of 
 352     // standard typenames, register it and return its index 
 353     int FindDataType(const wxString
& typeName
); 
 355     // try to FindDataType(), if it fails see if it is not one of already 
 356     // registered data types with some params in which case clone the 
 357     // registered data type and set params for it 
 358     int FindOrCloneDataType(const wxString
& typeName
); 
 360     wxGridCellRenderer
* GetRenderer(int index
); 
 361     wxGridCellEditor
*   GetEditor(int index
); 
 364     wxGridDataTypeInfoArray m_typeinfo
; 
 367 // ---------------------------------------------------------------------------- 
 368 // conditional compilation 
 369 // ---------------------------------------------------------------------------- 
 371 #ifndef WXGRID_DRAW_LINES 
 372 #define WXGRID_DRAW_LINES 1 
 375 // ---------------------------------------------------------------------------- 
 377 // ---------------------------------------------------------------------------- 
 379 //#define DEBUG_ATTR_CACHE 
 380 #ifdef DEBUG_ATTR_CACHE 
 381     static size_t gs_nAttrCacheHits 
= 0; 
 382     static size_t gs_nAttrCacheMisses 
= 0; 
 383 #endif // DEBUG_ATTR_CACHE 
 385 // ---------------------------------------------------------------------------- 
 387 // ---------------------------------------------------------------------------- 
 389 wxGridCellCoords 
wxGridNoCellCoords( -1, -1 ); 
 390 wxRect           
wxGridNoCellRect( -1, -1, -1, -1 ); 
 393 // TODO: this doesn't work at all, grid cells have different sizes and approx 
 394 //       calculations don't work as because of the size mismatch scrollbars 
 395 //       sometimes fail to be shown when they should be or vice versa 
 397 //       The scroll bars may be a little flakey once in a while, but that is 
 398 //       surely much less horrible than having scroll lines of only 1!!! 
 401 //       Well, it's still seriously broken so it might be better but needs 
 404 static const size_t GRID_SCROLL_LINE_X 
= 15;  // 1; 
 405 static const size_t GRID_SCROLL_LINE_Y 
= GRID_SCROLL_LINE_X
; 
 407 // the size of hash tables used a bit everywhere (the max number of elements 
 408 // in these hash tables is the number of rows/columns) 
 409 static const int GRID_HASH_SIZE 
= 100; 
 411 // ---------------------------------------------------------------------------- 
 413 // ---------------------------------------------------------------------------- 
 415 static inline int GetScrollX(int x
) 
 417     return (x 
+ GRID_SCROLL_LINE_X 
- 1) / GRID_SCROLL_LINE_X
; 
 420 static inline int GetScrollY(int y
) 
 422     return (y 
+ GRID_SCROLL_LINE_Y 
- 1) / GRID_SCROLL_LINE_Y
; 
 425 // ============================================================================ 
 427 // ============================================================================ 
 429 // ---------------------------------------------------------------------------- 
 431 // ---------------------------------------------------------------------------- 
 433 wxGridCellEditor::wxGridCellEditor() 
 440 wxGridCellEditor::~wxGridCellEditor() 
 445 void wxGridCellEditor::Create(wxWindow
* WXUNUSED(parent
), 
 446                               wxWindowID 
WXUNUSED(id
), 
 447                               wxEvtHandler
* evtHandler
) 
 450         m_control
->PushEventHandler(evtHandler
); 
 453 void wxGridCellEditor::PaintBackground(const wxRect
& rectCell
, 
 454                                        wxGridCellAttr 
*attr
) 
 456     // erase the background because we might not fill the cell 
 457     wxClientDC 
dc(m_control
->GetParent()); 
 458     wxGridWindow
* gridWindow 
= wxDynamicCast(m_control
->GetParent(), wxGridWindow
); 
 460         gridWindow
->GetOwner()->PrepareDC(dc
); 
 462     dc
.SetPen(*wxTRANSPARENT_PEN
); 
 463     dc
.SetBrush(wxBrush(attr
->GetBackgroundColour(), wxSOLID
)); 
 464     dc
.DrawRectangle(rectCell
); 
 466     // redraw the control we just painted over 
 467     m_control
->Refresh(); 
 470 void wxGridCellEditor::Destroy() 
 474         m_control
->PopEventHandler(TRUE 
/* delete it*/); 
 476         m_control
->Destroy(); 
 481 void wxGridCellEditor::Show(bool show
, wxGridCellAttr 
*attr
) 
 483     wxASSERT_MSG(m_control
, 
 484                  wxT("The wxGridCellEditor must be Created first!")); 
 485     m_control
->Show(show
); 
 489         // set the colours/fonts if we have any 
 492             m_colFgOld 
= m_control
->GetForegroundColour(); 
 493             m_control
->SetForegroundColour(attr
->GetTextColour()); 
 495             m_colBgOld 
= m_control
->GetBackgroundColour(); 
 496             m_control
->SetBackgroundColour(attr
->GetBackgroundColour()); 
 498             m_fontOld 
= m_control
->GetFont(); 
 499             m_control
->SetFont(attr
->GetFont()); 
 501             // can't do anything more in the base class version, the other 
 502             // attributes may only be used by the derived classes 
 507         // restore the standard colours fonts 
 508         if ( m_colFgOld
.Ok() ) 
 510             m_control
->SetForegroundColour(m_colFgOld
); 
 511             m_colFgOld 
= wxNullColour
; 
 514         if ( m_colBgOld
.Ok() ) 
 516             m_control
->SetBackgroundColour(m_colBgOld
); 
 517             m_colBgOld 
= wxNullColour
; 
 520         if ( m_fontOld
.Ok() ) 
 522             m_control
->SetFont(m_fontOld
); 
 523             m_fontOld 
= wxNullFont
; 
 528 void wxGridCellEditor::SetSize(const wxRect
& rect
) 
 530     wxASSERT_MSG(m_control
, 
 531                  wxT("The wxGridCellEditor must be Created first!")); 
 532     m_control
->SetSize(rect
, wxSIZE_ALLOW_MINUS_ONE
); 
 535 void wxGridCellEditor::HandleReturn(wxKeyEvent
& event
) 
 540 bool wxGridCellEditor::IsAcceptedKey(wxKeyEvent
& event
) 
 542     // accept the simple key presses, not anything with Ctrl/Alt/Meta 
 543     return !(event
.ControlDown() || event
.AltDown()); 
 546 void wxGridCellEditor::StartingKey(wxKeyEvent
& event
) 
 551 void wxGridCellEditor::StartingClick() 
 557 // ---------------------------------------------------------------------------- 
 558 // wxGridCellTextEditor 
 559 // ---------------------------------------------------------------------------- 
 561 wxGridCellTextEditor::wxGridCellTextEditor() 
 566 void wxGridCellTextEditor::Create(wxWindow
* parent
, 
 568                                   wxEvtHandler
* evtHandler
) 
 570     m_control 
= new wxTextCtrl(parent
, id
, wxEmptyString
, 
 571                                wxDefaultPosition
, wxDefaultSize
 
 572 #if defined(__WXMSW__) 
 573                                , wxTE_PROCESS_TAB 
| wxTE_MULTILINE 
| 
 574                                  wxTE_NO_VSCROLL 
| wxTE_AUTO_SCROLL
 
 578     // set max length allowed in the textctrl, if the parameter was set 
 581         ((wxTextCtrl
*)m_control
)->SetMaxLength(m_maxChars
); 
 584     wxGridCellEditor::Create(parent
, id
, evtHandler
); 
 587 void wxGridCellTextEditor::PaintBackground(const wxRect
& WXUNUSED(rectCell
), 
 588                                            wxGridCellAttr 
* WXUNUSED(attr
)) 
 590     // as we fill the entire client area, don't do anything here to minimize 
 594 void wxGridCellTextEditor::SetSize(const wxRect
& rectOrig
) 
 596     wxRect 
rect(rectOrig
); 
 598     // Make the edit control large enough to allow for internal 
 601     // TODO: remove this if the text ctrl sizing is improved esp. for 
 604 #if defined(__WXGTK__) 
 613     int extra_x 
= ( rect
.x 
> 2 )? 2 : 1; 
 615 // MB: treat MSW separately here otherwise the caret doesn't show 
 616 // when the editor is in the first row. 
 617 #if defined(__WXMSW__) 
 620     int extra_y 
= ( rect
.y 
> 2 )? 2 : 1; 
 623 #if defined(__WXMOTIF__) 
 627     rect
.SetLeft( wxMax(0, rect
.x 
- extra_x
) ); 
 628     rect
.SetTop( wxMax(0, rect
.y 
- extra_y
) ); 
 629     rect
.SetRight( rect
.GetRight() + 2*extra_x 
); 
 630     rect
.SetBottom( rect
.GetBottom() + 2*extra_y 
); 
 633     wxGridCellEditor::SetSize(rect
); 
 636 void wxGridCellTextEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
 638     wxASSERT_MSG(m_control
, 
 639                  wxT("The wxGridCellEditor must be Created first!")); 
 641     m_startValue 
= grid
->GetTable()->GetValue(row
, col
); 
 643     DoBeginEdit(m_startValue
); 
 646 void wxGridCellTextEditor::DoBeginEdit(const wxString
& startValue
) 
 648     Text()->SetValue(startValue
); 
 649     Text()->SetInsertionPointEnd(); 
 650     Text()->SetSelection(-1,-1); 
 654 bool wxGridCellTextEditor::EndEdit(int row
, int col
, 
 657     wxASSERT_MSG(m_control
, 
 658                  wxT("The wxGridCellEditor must be Created first!")); 
 660     bool changed 
= FALSE
; 
 661     wxString value 
= Text()->GetValue(); 
 662     if (value 
!= m_startValue
) 
 666         grid
->GetTable()->SetValue(row
, col
, value
); 
 668     m_startValue 
= wxEmptyString
; 
 669     Text()->SetValue(m_startValue
); 
 675 void wxGridCellTextEditor::Reset() 
 677     wxASSERT_MSG(m_control
, 
 678                  wxT("The wxGridCellEditor must be Created first!")); 
 680     DoReset(m_startValue
); 
 683 void wxGridCellTextEditor::DoReset(const wxString
& startValue
) 
 685     Text()->SetValue(startValue
); 
 686     Text()->SetInsertionPointEnd(); 
 689 bool wxGridCellTextEditor::IsAcceptedKey(wxKeyEvent
& event
) 
 691     if ( wxGridCellEditor::IsAcceptedKey(event
) ) 
 693         int keycode 
= event
.GetKeyCode(); 
 707             case WXK_NUMPAD_MULTIPLY
: 
 711             case WXK_NUMPAD_SUBTRACT
: 
 713             case WXK_NUMPAD_DECIMAL
: 
 715             case WXK_NUMPAD_DIVIDE
: 
 719                 // accept 8 bit chars too if isprint() agrees 
 720                 if ( (keycode 
< 255) && (isprint(keycode
)) ) 
 728 void wxGridCellTextEditor::StartingKey(wxKeyEvent
& event
) 
 730     if ( !Text()->EmulateKeyPress(event
) ) 
 736 void wxGridCellTextEditor::HandleReturn( wxKeyEvent
& 
 737                                          WXUNUSED_GTK(WXUNUSED_MOTIF(event
)) ) 
 739 #if defined(__WXMOTIF__) || defined(__WXGTK__) 
 740     // wxMotif needs a little extra help... 
 741     size_t pos 
= (size_t)( Text()->GetInsertionPoint() ); 
 742     wxString 
s( Text()->GetValue() ); 
 743     s 
= s
.Left(pos
) + wxT("\n") + s
.Mid(pos
); 
 745     Text()->SetInsertionPoint( pos 
); 
 747     // the other ports can handle a Return key press 
 753 void wxGridCellTextEditor::SetParameters(const wxString
& params
) 
 763         if ( !params
.ToLong(&tmp
) ) 
 765             wxLogDebug(_T("Invalid wxGridCellTextEditor parameter string '%s' ignored"), params
.c_str()); 
 769             m_maxChars 
= (size_t)tmp
; 
 774 // return the value in the text control 
 775 wxString 
wxGridCellTextEditor::GetValue() const 
 777   return Text()->GetValue(); 
 780 // ---------------------------------------------------------------------------- 
 781 // wxGridCellNumberEditor 
 782 // ---------------------------------------------------------------------------- 
 784 wxGridCellNumberEditor::wxGridCellNumberEditor(int min
, int max
) 
 790 void wxGridCellNumberEditor::Create(wxWindow
* parent
, 
 792                                     wxEvtHandler
* evtHandler
) 
 796         // create a spin ctrl 
 797         m_control 
= new wxSpinCtrl(parent
, -1, wxEmptyString
, 
 798                                    wxDefaultPosition
, wxDefaultSize
, 
 802         wxGridCellEditor::Create(parent
, id
, evtHandler
); 
 806         // just a text control 
 807         wxGridCellTextEditor::Create(parent
, id
, evtHandler
); 
 810         Text()->SetValidator(wxTextValidator(wxFILTER_NUMERIC
)); 
 811 #endif // wxUSE_VALIDATORS 
 815 void wxGridCellNumberEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
 817     // first get the value 
 818     wxGridTableBase 
*table 
= grid
->GetTable(); 
 819     if ( table
->CanGetValueAs(row
, col
, wxGRID_VALUE_NUMBER
) ) 
 821         m_valueOld 
= table
->GetValueAsLong(row
, col
); 
 826         wxString sValue 
= table
->GetValue(row
, col
); 
 827         if (! sValue
.ToLong(&m_valueOld
) && ! sValue
.IsEmpty()) 
 829             wxFAIL_MSG( _T("this cell doesn't have numeric value") ); 
 836         Spin()->SetValue((int)m_valueOld
); 
 841         DoBeginEdit(GetString()); 
 845 bool wxGridCellNumberEditor::EndEdit(int row
, int col
, 
 854         value 
= Spin()->GetValue(); 
 855         changed 
= value 
!= m_valueOld
; 
 857             text 
= wxString::Format(wxT("%ld"), value
); 
 861         text 
= Text()->GetValue(); 
 862         changed 
= (text
.IsEmpty() || text
.ToLong(&value
)) && (value 
!= m_valueOld
); 
 867         if (grid
->GetTable()->CanSetValueAs(row
, col
, wxGRID_VALUE_NUMBER
)) 
 868             grid
->GetTable()->SetValueAsLong(row
, col
, value
); 
 870             grid
->GetTable()->SetValue(row
, col
, text
); 
 876 void wxGridCellNumberEditor::Reset() 
 880         Spin()->SetValue((int)m_valueOld
); 
 884         DoReset(GetString()); 
 888 bool wxGridCellNumberEditor::IsAcceptedKey(wxKeyEvent
& event
) 
 890     if ( wxGridCellEditor::IsAcceptedKey(event
) ) 
 892         int keycode 
= event
.GetKeyCode(); 
 908             case WXK_NUMPAD_SUBTRACT
: 
 914                 if ( (keycode 
< 128) && isdigit(keycode
) ) 
 922 void wxGridCellNumberEditor::StartingKey(wxKeyEvent
& event
) 
 926         int keycode 
= event
.GetKeyCode(); 
 927         if ( isdigit(keycode
) || keycode 
== '+' || keycode 
== '-' 
 928             || keycode 
==  WXK_NUMPAD0
 
 929             || keycode 
==  WXK_NUMPAD1
 
 930             || keycode 
==  WXK_NUMPAD2
 
 931             || keycode 
==  WXK_NUMPAD3
 
 932             || keycode 
==  WXK_NUMPAD4
 
 933             || keycode 
==  WXK_NUMPAD5
 
 934             || keycode 
==  WXK_NUMPAD6
 
 935             || keycode 
==  WXK_NUMPAD7
 
 936             || keycode 
==  WXK_NUMPAD8
 
 937             || keycode 
==  WXK_NUMPAD9
 
 938             || keycode 
==  WXK_ADD
 
 939             || keycode 
==  WXK_NUMPAD_ADD
 
 940             || keycode 
==  WXK_SUBTRACT
 
 941             || keycode 
==  WXK_NUMPAD_SUBTRACT
) 
 943             wxGridCellTextEditor::StartingKey(event
); 
 953 void wxGridCellNumberEditor::SetParameters(const wxString
& params
) 
 964         if ( params
.BeforeFirst(_T(',')).ToLong(&tmp
) ) 
 968             if ( params
.AfterFirst(_T(',')).ToLong(&tmp
) ) 
 972                 // skip the error message below 
 977         wxLogDebug(_T("Invalid wxGridCellNumberEditor parameter string '%s' ignored"), params
.c_str()); 
 981 // return the value in the spin control if it is there (the text control otherwise) 
 982 wxString 
wxGridCellNumberEditor::GetValue() const 
 988     long value 
= Spin()->GetValue(); 
 989     s
.Printf(wxT("%ld"), value
); 
 993     s 
= Text()->GetValue(); 
 998 // ---------------------------------------------------------------------------- 
 999 // wxGridCellFloatEditor 
1000 // ---------------------------------------------------------------------------- 
1002 wxGridCellFloatEditor::wxGridCellFloatEditor(int width
, int precision
) 
1005     m_precision 
= precision
; 
1008 void wxGridCellFloatEditor::Create(wxWindow
* parent
, 
1010                                    wxEvtHandler
* evtHandler
) 
1012     wxGridCellTextEditor::Create(parent
, id
, evtHandler
); 
1014 #if wxUSE_VALIDATORS 
1015     Text()->SetValidator(wxTextValidator(wxFILTER_NUMERIC
)); 
1016 #endif // wxUSE_VALIDATORS 
1019 void wxGridCellFloatEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
1021     // first get the value 
1022     wxGridTableBase 
*table 
= grid
->GetTable(); 
1023     if ( table
->CanGetValueAs(row
, col
, wxGRID_VALUE_FLOAT
) ) 
1025         m_valueOld 
= table
->GetValueAsDouble(row
, col
); 
1030         wxString sValue 
= table
->GetValue(row
, col
); 
1031         if (! sValue
.ToDouble(&m_valueOld
) && ! sValue
.IsEmpty()) 
1033             wxFAIL_MSG( _T("this cell doesn't have float value") ); 
1038     DoBeginEdit(GetString()); 
1041 bool wxGridCellFloatEditor::EndEdit(int row
, int col
, 
1045     wxString 
text(Text()->GetValue()); 
1047     if ( (text
.IsEmpty() || text
.ToDouble(&value
)) && (value 
!= m_valueOld
) ) 
1049         if (grid
->GetTable()->CanSetValueAs(row
, col
, wxGRID_VALUE_FLOAT
)) 
1050             grid
->GetTable()->SetValueAsDouble(row
, col
, value
); 
1052             grid
->GetTable()->SetValue(row
, col
, text
); 
1059 void wxGridCellFloatEditor::Reset() 
1061     DoReset(GetString()); 
1064 void wxGridCellFloatEditor::StartingKey(wxKeyEvent
& event
) 
1066     int keycode 
= event
.GetKeyCode(); 
1067         if ( isdigit(keycode
) || keycode 
== '+' || keycode 
== '-' || keycode 
== '.' 
1068             || keycode 
==  WXK_NUMPAD0
 
1069             || keycode 
==  WXK_NUMPAD1
 
1070             || keycode 
==  WXK_NUMPAD2
 
1071             || keycode 
==  WXK_NUMPAD3
 
1072             || keycode 
==  WXK_NUMPAD4
 
1073             || keycode 
==  WXK_NUMPAD5
 
1074             || keycode 
==  WXK_NUMPAD6
 
1075             || keycode 
==  WXK_NUMPAD7
 
1076             || keycode 
==  WXK_NUMPAD8
 
1077             || keycode 
==  WXK_NUMPAD9
 
1078             || keycode 
==  WXK_ADD
 
1079             || keycode 
==  WXK_NUMPAD_ADD
 
1080             || keycode 
==  WXK_SUBTRACT
 
1081             || keycode 
==  WXK_NUMPAD_SUBTRACT
) 
1083         wxGridCellTextEditor::StartingKey(event
); 
1085         // skip Skip() below 
1092 void wxGridCellFloatEditor::SetParameters(const wxString
& params
) 
1103         if ( params
.BeforeFirst(_T(',')).ToLong(&tmp
) ) 
1107             if ( params
.AfterFirst(_T(',')).ToLong(&tmp
) ) 
1109                 m_precision 
= (int)tmp
; 
1111                 // skip the error message below 
1116         wxLogDebug(_T("Invalid wxGridCellFloatEditor parameter string '%s' ignored"), params
.c_str()); 
1120 wxString 
wxGridCellFloatEditor::GetString() const 
1123     if ( m_width 
== -1 ) 
1125         // default width/precision 
1128     else if ( m_precision 
== -1 ) 
1130         // default precision 
1131         fmt
.Printf(_T("%%%d.f"), m_width
); 
1135         fmt
.Printf(_T("%%%d.%df"), m_width
, m_precision
); 
1138     return wxString::Format(fmt
, m_valueOld
); 
1141 bool wxGridCellFloatEditor::IsAcceptedKey(wxKeyEvent
& event
) 
1143     if ( wxGridCellEditor::IsAcceptedKey(event
) ) 
1145         int keycode 
= event
.GetKeyCode(); 
1159             case WXK_NUMPAD_ADD
: 
1161             case WXK_NUMPAD_SUBTRACT
: 
1163             case WXK_NUMPAD_DECIMAL
: 
1167                 // additionally accept 'e' as in '1e+6' 
1168                 if ( (keycode 
< 128) && 
1169                      (isdigit(keycode
) || tolower(keycode
) == 'e') ) 
1177 #endif // wxUSE_TEXTCTRL 
1181 // ---------------------------------------------------------------------------- 
1182 // wxGridCellBoolEditor 
1183 // ---------------------------------------------------------------------------- 
1185 void wxGridCellBoolEditor::Create(wxWindow
* parent
, 
1187                                   wxEvtHandler
* evtHandler
) 
1189     m_control 
= new wxCheckBox(parent
, id
, wxEmptyString
, 
1190                                wxDefaultPosition
, wxDefaultSize
, 
1193     wxGridCellEditor::Create(parent
, id
, evtHandler
); 
1196 void wxGridCellBoolEditor::SetSize(const wxRect
& r
) 
1198     bool resize 
= FALSE
; 
1199     wxSize size 
= m_control
->GetSize(); 
1200     wxCoord minSize 
= wxMin(r
.width
, r
.height
); 
1202     // check if the checkbox is not too big/small for this cell 
1203     wxSize sizeBest 
= m_control
->GetBestSize(); 
1204     if ( !(size 
== sizeBest
) ) 
1206         // reset to default size if it had been made smaller 
1212     if ( size
.x 
>= minSize 
|| size
.y 
>= minSize 
) 
1214         // leave 1 pixel margin 
1215         size
.x 
= size
.y 
= minSize 
- 2; 
1222         m_control
->SetSize(size
); 
1225     // position it in the centre of the rectangle (TODO: support alignment?) 
1227 #if defined(__WXGTK__) || defined (__WXMOTIF__) 
1228     // the checkbox without label still has some space to the right in wxGTK, 
1229     // so shift it to the right 
1231 #elif defined(__WXMSW__) 
1232     // here too, but in other way 
1237     int hAlign 
= wxALIGN_CENTRE
; 
1238     int vAlign 
= wxALIGN_CENTRE
; 
1240         GetCellAttr()->GetAlignment(& hAlign
, & vAlign
); 
1243     if (hAlign 
== wxALIGN_LEFT
) 
1249         y 
= r
.y 
+ r
.height
/2 - size
.y
/2; 
1251     else if (hAlign 
== wxALIGN_RIGHT
) 
1253         x 
= r
.x 
+ r
.width 
- size
.x 
- 2; 
1254         y 
= r
.y 
+ r
.height
/2 - size
.y
/2; 
1256     else if (hAlign 
== wxALIGN_CENTRE
) 
1258         x 
= r
.x 
+ r
.width
/2 - size
.x
/2; 
1259         y 
= r
.y 
+ r
.height
/2 - size
.y
/2; 
1262     m_control
->Move(x
, y
); 
1265 void wxGridCellBoolEditor::Show(bool show
, wxGridCellAttr 
*attr
) 
1267     m_control
->Show(show
); 
1271         wxColour colBg 
= attr 
? attr
->GetBackgroundColour() : *wxLIGHT_GREY
; 
1272         CBox()->SetBackgroundColour(colBg
); 
1276 void wxGridCellBoolEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
1278     wxASSERT_MSG(m_control
, 
1279                  wxT("The wxGridCellEditor must be Created first!")); 
1281     if (grid
->GetTable()->CanGetValueAs(row
, col
, wxGRID_VALUE_BOOL
)) 
1282         m_startValue 
= grid
->GetTable()->GetValueAsBool(row
, col
); 
1285         wxString 
cellval( grid
->GetTable()->GetValue(row
, col
) ); 
1286         m_startValue 
= !( !cellval 
|| (cellval 
== wxT("0")) ); 
1288     CBox()->SetValue(m_startValue
); 
1292 bool wxGridCellBoolEditor::EndEdit(int row
, int col
, 
1295     wxASSERT_MSG(m_control
, 
1296                  wxT("The wxGridCellEditor must be Created first!")); 
1298     bool changed 
= FALSE
; 
1299     bool value 
= CBox()->GetValue(); 
1300     if ( value 
!= m_startValue 
) 
1305         if (grid
->GetTable()->CanGetValueAs(row
, col
, wxGRID_VALUE_BOOL
)) 
1306             grid
->GetTable()->SetValueAsBool(row
, col
, value
); 
1308             grid
->GetTable()->SetValue(row
, col
, value 
? _T("1") : wxEmptyString
); 
1314 void wxGridCellBoolEditor::Reset() 
1316     wxASSERT_MSG(m_control
, 
1317                  wxT("The wxGridCellEditor must be Created first!")); 
1319     CBox()->SetValue(m_startValue
); 
1322 void wxGridCellBoolEditor::StartingClick() 
1324     CBox()->SetValue(!CBox()->GetValue()); 
1327 bool wxGridCellBoolEditor::IsAcceptedKey(wxKeyEvent
& event
) 
1329     if ( wxGridCellEditor::IsAcceptedKey(event
) ) 
1331         int keycode 
= event
.GetKeyCode(); 
1335             case WXK_NUMPAD_MULTIPLY
: 
1337             case WXK_NUMPAD_ADD
: 
1339             case WXK_NUMPAD_SUBTRACT
: 
1350 // return the value as "1" for true and the empty string for false 
1351 wxString 
wxGridCellBoolEditor::GetValue() const 
1353   bool bSet 
= CBox()->GetValue(); 
1354   return bSet 
? _T("1") : wxEmptyString
; 
1357 #endif // wxUSE_CHECKBOX 
1361 // ---------------------------------------------------------------------------- 
1362 // wxGridCellChoiceEditor 
1363 // ---------------------------------------------------------------------------- 
1365 wxGridCellChoiceEditor::wxGridCellChoiceEditor(size_t count
, 
1366                                                const wxString choices
[], 
1368                       : m_allowOthers(allowOthers
) 
1372         m_choices
.Alloc(count
); 
1373         for ( size_t n 
= 0; n 
< count
; n
++ ) 
1375             m_choices
.Add(choices
[n
]); 
1380 wxGridCellEditor 
*wxGridCellChoiceEditor::Clone() const 
1382     wxGridCellChoiceEditor 
*editor 
= new wxGridCellChoiceEditor
; 
1383     editor
->m_allowOthers 
= m_allowOthers
; 
1384     editor
->m_choices 
= m_choices
; 
1389 void wxGridCellChoiceEditor::Create(wxWindow
* parent
, 
1391                                     wxEvtHandler
* evtHandler
) 
1393     size_t count 
= m_choices
.GetCount(); 
1394     wxString 
*choices 
= new wxString
[count
]; 
1395     for ( size_t n 
= 0; n 
< count
; n
++ ) 
1397         choices
[n
] = m_choices
[n
]; 
1400     m_control 
= new wxComboBox(parent
, id
, wxEmptyString
, 
1401                                wxDefaultPosition
, wxDefaultSize
, 
1403                                m_allowOthers 
? 0 : wxCB_READONLY
); 
1407     wxGridCellEditor::Create(parent
, id
, evtHandler
); 
1410 void wxGridCellChoiceEditor::PaintBackground(const wxRect
& rectCell
, 
1411                                              wxGridCellAttr 
* attr
) 
1413     // as we fill the entire client area, don't do anything here to minimize 
1416     // TODO: It doesn't actually fill the client area since the height of a 
1417     // combo always defaults to the standard...  Until someone has time to 
1418     // figure out the right rectangle to paint, just do it the normal way... 
1419     wxGridCellEditor::PaintBackground(rectCell
, attr
); 
1422 void wxGridCellChoiceEditor::BeginEdit(int row
, int col
, wxGrid
* grid
) 
1424     wxASSERT_MSG(m_control
, 
1425                  wxT("The wxGridCellEditor must be Created first!")); 
1427     m_startValue 
= grid
->GetTable()->GetValue(row
, col
); 
1430         Combo()->SetValue(m_startValue
); 
1433         // find the right position, or default to the first if not found 
1434         int pos 
= Combo()->FindString(m_startValue
); 
1437         Combo()->SetSelection(pos
); 
1439     Combo()->SetInsertionPointEnd(); 
1440     Combo()->SetFocus(); 
1443 bool wxGridCellChoiceEditor::EndEdit(int row
, int col
, 
1446     wxString value 
= Combo()->GetValue(); 
1447     bool changed 
= value 
!= m_startValue
; 
1450         grid
->GetTable()->SetValue(row
, col
, value
); 
1452     m_startValue 
= wxEmptyString
; 
1454         Combo()->SetValue(m_startValue
); 
1456         Combo()->SetSelection(0); 
1461 void wxGridCellChoiceEditor::Reset() 
1463     Combo()->SetValue(m_startValue
); 
1464     Combo()->SetInsertionPointEnd(); 
1467 void wxGridCellChoiceEditor::SetParameters(const wxString
& params
) 
1477     wxStringTokenizer 
tk(params
, _T(',')); 
1478     while ( tk
.HasMoreTokens() ) 
1480         m_choices
.Add(tk
.GetNextToken()); 
1484 // return the value in the text control 
1485 wxString 
wxGridCellChoiceEditor::GetValue() const 
1487   return Combo()->GetValue(); 
1490 #endif // wxUSE_COMBOBOX 
1492 // ---------------------------------------------------------------------------- 
1493 // wxGridCellEditorEvtHandler 
1494 // ---------------------------------------------------------------------------- 
1496 void wxGridCellEditorEvtHandler::OnKeyDown(wxKeyEvent
& event
) 
1498     switch ( event
.GetKeyCode() ) 
1502             m_grid
->DisableCellEditControl(); 
1506             m_grid
->GetEventHandler()->ProcessEvent( event 
); 
1510         case WXK_NUMPAD_ENTER
: 
1511             if (!m_grid
->GetEventHandler()->ProcessEvent(event
)) 
1512                 m_editor
->HandleReturn(event
); 
1521 void wxGridCellEditorEvtHandler::OnChar(wxKeyEvent
& event
) 
1523     switch ( event
.GetKeyCode() ) 
1528         case WXK_NUMPAD_ENTER
: 
1536 // ---------------------------------------------------------------------------- 
1537 // wxGridCellWorker is an (almost) empty common base class for 
1538 // wxGridCellRenderer and wxGridCellEditor managing ref counting 
1539 // ---------------------------------------------------------------------------- 
1541 void wxGridCellWorker::SetParameters(const wxString
& WXUNUSED(params
)) 
1546 wxGridCellWorker::~wxGridCellWorker() 
1550 // ============================================================================ 
1552 // ============================================================================ 
1554 // ---------------------------------------------------------------------------- 
1555 // wxGridCellRenderer 
1556 // ---------------------------------------------------------------------------- 
1558 void wxGridCellRenderer::Draw(wxGrid
& grid
, 
1559                               wxGridCellAttr
& attr
, 
1562                               int WXUNUSED(row
), int WXUNUSED(col
), 
1565     dc
.SetBackgroundMode( wxSOLID 
); 
1567     // grey out fields if the grid is disabled 
1568     if( grid
.IsEnabled() ) 
1572           dc
.SetBrush( wxBrush(grid
.GetSelectionBackground(), wxSOLID
) ); 
1576           dc
.SetBrush( wxBrush(attr
.GetBackgroundColour(), wxSOLID
) ); 
1581       dc
.SetBrush(wxBrush(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_BTNFACE
), wxSOLID
)); 
1584     dc
.SetPen( *wxTRANSPARENT_PEN 
); 
1585     dc
.DrawRectangle(rect
); 
1588 // ---------------------------------------------------------------------------- 
1589 // wxGridCellStringRenderer 
1590 // ---------------------------------------------------------------------------- 
1592 void wxGridCellStringRenderer::SetTextColoursAndFont(wxGrid
& grid
, 
1593                                                      wxGridCellAttr
& attr
, 
1597     dc
.SetBackgroundMode( wxTRANSPARENT 
); 
1599     // TODO some special colours for attr.IsReadOnly() case? 
1601     // different coloured text when the grid is disabled 
1602     if( grid
.IsEnabled() ) 
1606           dc
.SetTextBackground( grid
.GetSelectionBackground() ); 
1607           dc
.SetTextForeground( grid
.GetSelectionForeground() ); 
1611           dc
.SetTextBackground( attr
.GetBackgroundColour() ); 
1612           dc
.SetTextForeground( attr
.GetTextColour() ); 
1617       dc
.SetTextBackground(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_BTNFACE
)); 
1618       dc
.SetTextForeground(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_GRAYTEXT
)); 
1621     dc
.SetFont( attr
.GetFont() ); 
1624 wxSize 
wxGridCellStringRenderer::DoGetBestSize(wxGridCellAttr
& attr
, 
1626                                                const wxString
& text
) 
1628     wxCoord x 
= 0, y 
= 0, max_x 
= 0; 
1629     dc
.SetFont(attr
.GetFont()); 
1630     wxStringTokenizer 
tk(text
, _T('\n')); 
1631     while ( tk
.HasMoreTokens() ) 
1633         dc
.GetTextExtent(tk
.GetNextToken(), &x
, &y
); 
1634         max_x 
= wxMax(max_x
, x
); 
1637     y 
*= 1 + text
.Freq(wxT('\n')); // multiply by the number of lines. 
1639     return wxSize(max_x
, y
); 
1642 wxSize 
wxGridCellStringRenderer::GetBestSize(wxGrid
& grid
, 
1643                                              wxGridCellAttr
& attr
, 
1647     return DoGetBestSize(attr
, dc
, grid
.GetCellValue(row
, col
)); 
1650 void wxGridCellStringRenderer::Draw(wxGrid
& grid
, 
1651                                     wxGridCellAttr
& attr
, 
1653                                     const wxRect
& rectCell
, 
1657     wxRect rect 
= rectCell
; 
1660     // erase only this cells background, overflow cells should have been erased 
1661     wxGridCellRenderer::Draw(grid
, attr
, dc
, rectCell
, row
, col
, isSelected
); 
1664     attr
.GetAlignment(&hAlign
, &vAlign
); 
1666     int overflowCols 
= 0; 
1668     if (attr
.GetOverflow()) 
1670         int cols 
= grid
.GetNumberCols(); 
1671         int best_width 
= GetBestSize(grid
,attr
,dc
,row
,col
).GetWidth(); 
1672         int cell_rows
, cell_cols
; 
1673         attr
.GetSize( &cell_rows
, &cell_cols 
); // shouldn't get here if <=0 
1674         if ((best_width 
> rectCell
.width
) && (col 
< cols
) && grid
.GetTable()) 
1676             int i
, c_cols
, c_rows
; 
1677             for (i 
= col
+cell_cols
; i 
< cols
; i
++) 
1679                 bool is_empty 
= TRUE
; 
1680                 for (int j
=row
; j
<row
+cell_rows
; j
++) 
1682                     // check w/ anchor cell for multicell block 
1683                     grid
.GetCellSize(j
, i
, &c_rows
, &c_cols
); 
1684                     if (c_rows 
> 0) c_rows 
= 0; 
1685                     if (!grid
.GetTable()->IsEmptyCell(j
+c_rows
, i
)) 
1692                     rect
.width 
+= grid
.GetColSize(i
); 
1698                 if (rect
.width 
>= best_width
) break; 
1700             overflowCols 
= i 
- col 
- cell_cols 
+ 1; 
1701             if (overflowCols 
>= cols
) overflowCols 
= cols 
- 1; 
1704         if (overflowCols 
> 0) // redraw overflow cells w/ proper hilight 
1706             hAlign 
= wxALIGN_LEFT
; // if oveflowed then it's left aligned 
1708             clip
.x 
+= rectCell
.width
; 
1709             // draw each overflow cell individually 
1710             int col_end 
= col
+cell_cols
+overflowCols
; 
1711             if (col_end 
>= grid
.GetNumberCols()) 
1712                 col_end 
= grid
.GetNumberCols() - 1; 
1713             for (int i 
= col
+cell_cols
; i 
<= col_end
; i
++) 
1715                 clip
.width 
= grid
.GetColSize(i
) - 1; 
1716                 dc
.DestroyClippingRegion(); 
1717                 dc
.SetClippingRegion(clip
); 
1719                 SetTextColoursAndFont(grid
, attr
, dc
, 
1720                         grid
.IsInSelection(row
,i
)); 
1722                 grid
.DrawTextRectangle(dc
, grid
.GetCellValue(row
, col
), 
1723                         rect
, hAlign
, vAlign
); 
1724                 clip
.x 
+= grid
.GetColSize(i
) - 1; 
1730             dc
.DestroyClippingRegion(); 
1734     // now we only have to draw the text 
1735     SetTextColoursAndFont(grid
, attr
, dc
, isSelected
); 
1737     grid
.DrawTextRectangle(dc
, grid
.GetCellValue(row
, col
), 
1738                            rect
, hAlign
, vAlign
); 
1741 // ---------------------------------------------------------------------------- 
1742 // wxGridCellNumberRenderer 
1743 // ---------------------------------------------------------------------------- 
1745 wxString 
wxGridCellNumberRenderer::GetString(wxGrid
& grid
, int row
, int col
) 
1747     wxGridTableBase 
*table 
= grid
.GetTable(); 
1749     if ( table
->CanGetValueAs(row
, col
, wxGRID_VALUE_NUMBER
) ) 
1751         text
.Printf(_T("%ld"), table
->GetValueAsLong(row
, col
)); 
1755         text 
= table
->GetValue(row
, col
); 
1761 void wxGridCellNumberRenderer::Draw(wxGrid
& grid
, 
1762                                     wxGridCellAttr
& attr
, 
1764                                     const wxRect
& rectCell
, 
1768     wxGridCellRenderer::Draw(grid
, attr
, dc
, rectCell
, row
, col
, isSelected
); 
1770     SetTextColoursAndFont(grid
, attr
, dc
, isSelected
); 
1772     // draw the text right aligned by default 
1774     attr
.GetAlignment(&hAlign
, &vAlign
); 
1775     hAlign 
= wxALIGN_RIGHT
; 
1777     wxRect rect 
= rectCell
; 
1780     grid
.DrawTextRectangle(dc
, GetString(grid
, row
, col
), rect
, hAlign
, vAlign
); 
1783 wxSize 
wxGridCellNumberRenderer::GetBestSize(wxGrid
& grid
, 
1784                                              wxGridCellAttr
& attr
, 
1788     return DoGetBestSize(attr
, dc
, GetString(grid
, row
, col
)); 
1791 // ---------------------------------------------------------------------------- 
1792 // wxGridCellFloatRenderer 
1793 // ---------------------------------------------------------------------------- 
1795 wxGridCellFloatRenderer::wxGridCellFloatRenderer(int width
, int precision
) 
1798     SetPrecision(precision
); 
1801 wxGridCellRenderer 
*wxGridCellFloatRenderer::Clone() const 
1803     wxGridCellFloatRenderer 
*renderer 
= new wxGridCellFloatRenderer
; 
1804     renderer
->m_width 
= m_width
; 
1805     renderer
->m_precision 
= m_precision
; 
1806     renderer
->m_format 
= m_format
; 
1811 wxString 
wxGridCellFloatRenderer::GetString(wxGrid
& grid
, int row
, int col
) 
1813     wxGridTableBase 
*table 
= grid
.GetTable(); 
1818     if ( table
->CanGetValueAs(row
, col
, wxGRID_VALUE_FLOAT
) ) 
1820         val 
= table
->GetValueAsDouble(row
, col
); 
1825         text 
= table
->GetValue(row
, col
); 
1826         hasDouble 
= text
.ToDouble(&val
); 
1833             if ( m_width 
== -1 ) 
1835                 if ( m_precision 
== -1 ) 
1837                     // default width/precision 
1838                     m_format 
= _T("%f"); 
1842                     m_format
.Printf(_T("%%.%df"), m_precision
); 
1845             else if ( m_precision 
== -1 ) 
1847                 // default precision 
1848                 m_format
.Printf(_T("%%%d.f"), m_width
); 
1852                 m_format
.Printf(_T("%%%d.%df"), m_width
, m_precision
); 
1856         text
.Printf(m_format
, val
); 
1859     //else: text already contains the string 
1864 void wxGridCellFloatRenderer::Draw(wxGrid
& grid
, 
1865                                    wxGridCellAttr
& attr
, 
1867                                    const wxRect
& rectCell
, 
1871     wxGridCellRenderer::Draw(grid
, attr
, dc
, rectCell
, row
, col
, isSelected
); 
1873     SetTextColoursAndFont(grid
, attr
, dc
, isSelected
); 
1875     // draw the text right aligned by default 
1877     attr
.GetAlignment(&hAlign
, &vAlign
); 
1878     hAlign 
= wxALIGN_RIGHT
; 
1880     wxRect rect 
= rectCell
; 
1883     grid
.DrawTextRectangle(dc
, GetString(grid
, row
, col
), rect
, hAlign
, vAlign
); 
1886 wxSize 
wxGridCellFloatRenderer::GetBestSize(wxGrid
& grid
, 
1887                                             wxGridCellAttr
& attr
, 
1891     return DoGetBestSize(attr
, dc
, GetString(grid
, row
, col
)); 
1894 void wxGridCellFloatRenderer::SetParameters(const wxString
& params
) 
1898         // reset to defaults 
1904         wxString tmp 
= params
.BeforeFirst(_T(',')); 
1908             if ( tmp
.ToLong(&width
) ) 
1910                 SetWidth((int)width
); 
1914                 wxLogDebug(_T("Invalid wxGridCellFloatRenderer width parameter string '%s ignored"), params
.c_str()); 
1918                 tmp 
= params
.AfterFirst(_T(',')); 
1922             if ( tmp
.ToLong(&precision
) ) 
1924                 SetPrecision((int)precision
); 
1928                 wxLogDebug(_T("Invalid wxGridCellFloatRenderer precision parameter string '%s ignored"), params
.c_str()); 
1936 // ---------------------------------------------------------------------------- 
1937 // wxGridCellBoolRenderer 
1938 // ---------------------------------------------------------------------------- 
1940 wxSize 
wxGridCellBoolRenderer::ms_sizeCheckMark
; 
1942 // FIXME these checkbox size calculations are really ugly... 
1944 // between checkmark and box 
1945 static const wxCoord wxGRID_CHECKMARK_MARGIN 
= 2; 
1947 wxSize 
wxGridCellBoolRenderer::GetBestSize(wxGrid
& grid
, 
1948                                            wxGridCellAttr
& WXUNUSED(attr
), 
1953     // compute it only once (no locks for MT safeness in GUI thread...) 
1954     if ( !ms_sizeCheckMark
.x 
) 
1956         // get checkbox size 
1957         wxCoord checkSize 
= 0; 
1958         wxCheckBox 
*checkbox 
= new wxCheckBox(&grid
, -1, wxEmptyString
); 
1959         wxSize size 
= checkbox
->GetBestSize(); 
1960         checkSize 
= size
.y 
+ 2*wxGRID_CHECKMARK_MARGIN
; 
1962         // FIXME wxGTK::wxCheckBox::GetBestSize() gives "wrong" result 
1963 #if defined(__WXGTK__) || defined(__WXMOTIF__) 
1964         checkSize 
-= size
.y 
/ 2; 
1969         ms_sizeCheckMark
.x 
= ms_sizeCheckMark
.y 
= checkSize
; 
1972     return ms_sizeCheckMark
; 
1975 void wxGridCellBoolRenderer::Draw(wxGrid
& grid
, 
1976                                   wxGridCellAttr
& attr
, 
1982     wxGridCellRenderer::Draw(grid
, attr
, dc
, rect
, row
, col
, isSelected
); 
1984     // draw a check mark in the centre (ignoring alignment - TODO) 
1985     wxSize size 
= GetBestSize(grid
, attr
, dc
, row
, col
); 
1987     // don't draw outside the cell 
1988     wxCoord minSize 
= wxMin(rect
.width
, rect
.height
); 
1989     if ( size
.x 
>= minSize 
|| size
.y 
>= minSize 
) 
1991         // and even leave (at least) 1 pixel margin 
1992         size
.x 
= size
.y 
= minSize 
- 2; 
1995     // draw a border around checkmark 
1997     attr
.GetAlignment(& hAlign
, &vAlign
); 
2000     if (hAlign 
== wxALIGN_CENTRE
) 
2002         rectBorder
.x 
= rect
.x 
+ rect
.width
/2 - size
.x
/2; 
2003         rectBorder
.y 
= rect
.y 
+ rect
.height
/2 - size
.y
/2; 
2004         rectBorder
.width 
= size
.x
; 
2005         rectBorder
.height 
= size
.y
; 
2007     else if (hAlign 
== wxALIGN_LEFT
) 
2009         rectBorder
.x 
= rect
.x 
+ 2; 
2010         rectBorder
.y 
= rect
.y 
+ rect
.height
/2 - size
.y
/2; 
2011         rectBorder
.width 
= size
.x
; 
2012         rectBorder
.height 
= size
.y
; 
2014     else if (hAlign 
== wxALIGN_RIGHT
) 
2016         rectBorder
.x 
= rect
.x 
+ rect
.width 
- size
.x 
- 2; 
2017         rectBorder
.y 
= rect
.y 
+ rect
.height
/2 - size
.y
/2; 
2018         rectBorder
.width 
= size
.x
; 
2019         rectBorder
.height 
= size
.y
; 
2023     if ( grid
.GetTable()->CanGetValueAs(row
, col
, wxGRID_VALUE_BOOL
) ) 
2024         value 
= grid
.GetTable()->GetValueAsBool(row
, col
); 
2027         wxString 
cellval( grid
.GetTable()->GetValue(row
, col
) ); 
2028         value 
= !( !cellval 
|| (cellval 
== wxT("0")) ); 
2033         wxRect rectMark 
= rectBorder
; 
2035         // MSW DrawCheckMark() is weird (and should probably be changed...) 
2036         rectMark
.Inflate(-wxGRID_CHECKMARK_MARGIN
/2); 
2040         rectMark
.Inflate(-wxGRID_CHECKMARK_MARGIN
); 
2043         dc
.SetTextForeground(attr
.GetTextColour()); 
2044         dc
.DrawCheckMark(rectMark
); 
2047     dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
2048     dc
.SetPen(wxPen(attr
.GetTextColour(), 1, wxSOLID
)); 
2049     dc
.DrawRectangle(rectBorder
); 
2052 // ---------------------------------------------------------------------------- 
2054 // ---------------------------------------------------------------------------- 
2056 void wxGridCellAttr::Init(wxGridCellAttr 
*attrDefault
) 
2060     m_isReadOnly 
= Unset
; 
2065     m_attrkind 
= wxGridCellAttr::Cell
; 
2067     m_sizeRows 
= m_sizeCols 
= 1; 
2068     m_overflow 
= UnsetOverflow
; 
2070     SetDefAttr(attrDefault
); 
2073 wxGridCellAttr 
*wxGridCellAttr::Clone() const 
2075     wxGridCellAttr 
*attr 
= new wxGridCellAttr(m_defGridAttr
); 
2077     if ( HasTextColour() ) 
2078         attr
->SetTextColour(GetTextColour()); 
2079     if ( HasBackgroundColour() ) 
2080         attr
->SetBackgroundColour(GetBackgroundColour()); 
2082         attr
->SetFont(GetFont()); 
2083     if ( HasAlignment() ) 
2084         attr
->SetAlignment(m_hAlign
, m_vAlign
); 
2086     attr
->SetSize( m_sizeRows
, m_sizeCols 
); 
2090         attr
->SetRenderer(m_renderer
); 
2091         m_renderer
->IncRef(); 
2095         attr
->SetEditor(m_editor
); 
2100         attr
->SetReadOnly(); 
2102     attr
->SetKind( m_attrkind 
); 
2107 void wxGridCellAttr::MergeWith(wxGridCellAttr 
*mergefrom
) 
2109     if ( !HasTextColour() && mergefrom
->HasTextColour() ) 
2110         SetTextColour(mergefrom
->GetTextColour()); 
2111     if ( !HasBackgroundColour() && mergefrom
->HasBackgroundColour() ) 
2112         SetBackgroundColour(mergefrom
->GetBackgroundColour()); 
2113     if ( !HasFont() && mergefrom
->HasFont() ) 
2114         SetFont(mergefrom
->GetFont()); 
2115     if ( !HasAlignment() && mergefrom
->HasAlignment() ){ 
2117         mergefrom
->GetAlignment( &hAlign
, &vAlign
); 
2118         SetAlignment(hAlign
, vAlign
); 
2121     mergefrom
->GetSize( &m_sizeRows
, &m_sizeCols 
); 
2123     // Directly access member functions as GetRender/Editor don't just return 
2124     // m_renderer/m_editor 
2126     // Maybe add support for merge of Render and Editor? 
2127     if (!HasRenderer() && mergefrom
->HasRenderer() ) 
2129         m_renderer 
= mergefrom
->m_renderer
; 
2130         m_renderer
->IncRef(); 
2132     if ( !HasEditor() && mergefrom
->HasEditor() ) 
2134         m_editor 
=  mergefrom
->m_editor
; 
2137     if ( !HasReadWriteMode()  && mergefrom
->HasReadWriteMode() ) 
2138         SetReadOnly(mergefrom
->IsReadOnly()); 
2140     if (!HasOverflowMode()  && mergefrom
->HasOverflowMode() ) 
2141         SetOverflow(mergefrom
->GetOverflow()); 
2143     SetDefAttr(mergefrom
->m_defGridAttr
); 
2146 void wxGridCellAttr::SetSize(int num_rows
, int num_cols
) 
2148     // The size of a cell is normally 1,1 
2150     // If this cell is larger (2,2) then this is the top left cell 
2151     // the other cells that will be covered (lower right cells) must be 
2152     // set to negative or zero values such that 
2153     // row + num_rows of the covered cell points to the larger cell (this cell) 
2154     // same goes for the col + num_cols. 
2156     // Size of 0,0 is NOT valid, neither is <=0 and any positive value 
2158     wxASSERT_MSG( (!((num_rows
>0)&&(num_cols
<=0)) || 
2159                   !((num_rows
<=0)&&(num_cols
>0)) || 
2160                   !((num_rows
==0)&&(num_cols
==0))), 
2161                   wxT("wxGridCellAttr::SetSize only takes two postive values or negative/zero values")); 
2163     m_sizeRows 
= num_rows
; 
2164     m_sizeCols 
= num_cols
; 
2167 const wxColour
& wxGridCellAttr::GetTextColour() const 
2169     if (HasTextColour()) 
2173     else if (m_defGridAttr 
&& m_defGridAttr 
!= this) 
2175         return m_defGridAttr
->GetTextColour(); 
2179         wxFAIL_MSG(wxT("Missing default cell attribute")); 
2180         return wxNullColour
; 
2185 const wxColour
& wxGridCellAttr::GetBackgroundColour() const 
2187     if (HasBackgroundColour()) 
2189     else if (m_defGridAttr 
&& m_defGridAttr 
!= this) 
2190         return m_defGridAttr
->GetBackgroundColour(); 
2193         wxFAIL_MSG(wxT("Missing default cell attribute")); 
2194         return wxNullColour
; 
2199 const wxFont
& wxGridCellAttr::GetFont() const 
2203     else if (m_defGridAttr 
&& m_defGridAttr 
!= this) 
2204         return m_defGridAttr
->GetFont(); 
2207         wxFAIL_MSG(wxT("Missing default cell attribute")); 
2213 void wxGridCellAttr::GetAlignment(int *hAlign
, int *vAlign
) const 
2217         if ( hAlign 
) *hAlign 
= m_hAlign
; 
2218         if ( vAlign 
) *vAlign 
= m_vAlign
; 
2220     else if (m_defGridAttr 
&& m_defGridAttr 
!= this) 
2221         m_defGridAttr
->GetAlignment(hAlign
, vAlign
); 
2224         wxFAIL_MSG(wxT("Missing default cell attribute")); 
2228 void wxGridCellAttr::GetSize( int *num_rows
, int *num_cols 
) const 
2230     if ( num_rows 
) *num_rows 
= m_sizeRows
; 
2231     if ( num_cols 
) *num_cols 
= m_sizeCols
; 
2234 // GetRenderer and GetEditor use a slightly different decision path about 
2235 // which attribute to use.  If a non-default attr object has one then it is 
2236 // used, otherwise the default editor or renderer is fetched from the grid and 
2237 // used.  It should be the default for the data type of the cell.  If it is 
2238 // NULL (because the table has a type that the grid does not have in its 
2239 // registry,) then the grid's default editor or renderer is used. 
2241 wxGridCellRenderer
* wxGridCellAttr::GetRenderer(wxGrid
* grid
, int row
, int col
) const 
2243     wxGridCellRenderer 
*renderer
; 
2245     if ( m_renderer 
&& this != m_defGridAttr 
) 
2247         // use the cells renderer if it has one 
2248         renderer 
= m_renderer
; 
2251     else // no non default cell renderer 
2253         // get default renderer for the data type 
2256             // GetDefaultRendererForCell() will do IncRef() for us 
2257             renderer 
= grid
->GetDefaultRendererForCell(row
, col
); 
2266             if (m_defGridAttr 
&&  this != m_defGridAttr 
) 
2268                 // if we still don't have one then use the grid default 
2269                 // (no need for IncRef() here neither) 
2270                 renderer 
= m_defGridAttr
->GetRenderer(NULL
, 0, 0); 
2272             else // default grid attr 
2274                 // use m_renderer which we had decided not to use initially 
2275                 renderer 
= m_renderer
; 
2282     // we're supposed to always find something 
2283     wxASSERT_MSG(renderer
, wxT("Missing default cell renderer")); 
2288 // same as above, except for s/renderer/editor/g 
2289 wxGridCellEditor
* wxGridCellAttr::GetEditor(wxGrid
* grid
, int row
, int col
) const 
2291     wxGridCellEditor 
*editor
; 
2293     if ( m_editor 
&& this != m_defGridAttr 
) 
2295         // use the cells editor if it has one 
2299     else // no non default cell editor 
2301         // get default editor for the data type 
2304             // GetDefaultEditorForCell() will do IncRef() for us 
2305             editor 
= grid
->GetDefaultEditorForCell(row
, col
); 
2314             if ( m_defGridAttr 
&& this != m_defGridAttr 
) 
2316                 // if we still don't have one then use the grid default 
2317                 // (no need for IncRef() here neither) 
2318                 editor 
= m_defGridAttr
->GetEditor(NULL
, 0, 0); 
2320             else // default grid attr 
2322                 // use m_editor which we had decided not to use initially 
2330     // we're supposed to always find something 
2331     wxASSERT_MSG(editor
, wxT("Missing default cell editor")); 
2336 // ---------------------------------------------------------------------------- 
2337 // wxGridCellAttrData 
2338 // ---------------------------------------------------------------------------- 
2340 void wxGridCellAttrData::SetAttr(wxGridCellAttr 
*attr
, int row
, int col
) 
2342     int n 
= FindIndex(row
, col
); 
2343     if ( n 
== wxNOT_FOUND 
) 
2345         // add the attribute 
2346         m_attrs
.Add(new wxGridCellWithAttr(row
, col
, attr
)); 
2350         // free the old attribute 
2351         m_attrs
[(size_t)n
].attr
->DecRef(); 
2355             // change the attribute 
2356             m_attrs
[(size_t)n
].attr 
= attr
; 
2360             // remove this attribute 
2361             m_attrs
.RemoveAt((size_t)n
); 
2366 wxGridCellAttr 
*wxGridCellAttrData::GetAttr(int row
, int col
) const 
2368     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
2370     int n 
= FindIndex(row
, col
); 
2371     if ( n 
!= wxNOT_FOUND 
) 
2373         attr 
= m_attrs
[(size_t)n
].attr
; 
2380 void wxGridCellAttrData::UpdateAttrRows( size_t pos
, int numRows 
) 
2382     size_t count 
= m_attrs
.GetCount(); 
2383     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2385         wxGridCellCoords
& coords 
= m_attrs
[n
].coords
; 
2386         wxCoord row 
= coords
.GetRow(); 
2387         if ((size_t)row 
>= pos
) 
2391                 // If rows inserted, include row counter where necessary 
2392                 coords
.SetRow(row 
+ numRows
); 
2394             else if (numRows 
< 0) 
2396                 // If rows deleted ... 
2397                 if ((size_t)row 
>= pos 
- numRows
) 
2399                     // ...either decrement row counter (if row still exists)... 
2400                     coords
.SetRow(row 
+ numRows
); 
2404                     // ...or remove the attribute 
2405                     m_attrs
.RemoveAt((size_t)n
); 
2413 void wxGridCellAttrData::UpdateAttrCols( size_t pos
, int numCols 
) 
2415     size_t count 
= m_attrs
.GetCount(); 
2416     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2418         wxGridCellCoords
& coords 
= m_attrs
[n
].coords
; 
2419         wxCoord col 
= coords
.GetCol(); 
2420         if ( (size_t)col 
>= pos 
) 
2424                 // If rows inserted, include row counter where necessary 
2425                 coords
.SetCol(col 
+ numCols
); 
2427             else if (numCols 
< 0) 
2429                 // If rows deleted ... 
2430                 if ((size_t)col 
>= pos 
- numCols
) 
2432                     // ...either decrement row counter (if row still exists)... 
2433                     coords
.SetCol(col 
+ numCols
); 
2437                     // ...or remove the attribute 
2438                     m_attrs
.RemoveAt((size_t)n
); 
2446 int wxGridCellAttrData::FindIndex(int row
, int col
) const 
2448     size_t count 
= m_attrs
.GetCount(); 
2449     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2451         const wxGridCellCoords
& coords 
= m_attrs
[n
].coords
; 
2452         if ( (coords
.GetRow() == row
) && (coords
.GetCol() == col
) ) 
2461 // ---------------------------------------------------------------------------- 
2462 // wxGridRowOrColAttrData 
2463 // ---------------------------------------------------------------------------- 
2465 wxGridRowOrColAttrData::~wxGridRowOrColAttrData() 
2467     size_t count 
= m_attrs
.Count(); 
2468     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2470         m_attrs
[n
]->DecRef(); 
2474 wxGridCellAttr 
*wxGridRowOrColAttrData::GetAttr(int rowOrCol
) const 
2476     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
2478     int n 
= m_rowsOrCols
.Index(rowOrCol
); 
2479     if ( n 
!= wxNOT_FOUND 
) 
2481         attr 
= m_attrs
[(size_t)n
]; 
2488 void wxGridRowOrColAttrData::SetAttr(wxGridCellAttr 
*attr
, int rowOrCol
) 
2490     int i 
= m_rowsOrCols
.Index(rowOrCol
); 
2491     if ( i 
== wxNOT_FOUND 
) 
2493         // add the attribute 
2494         m_rowsOrCols
.Add(rowOrCol
); 
2499         size_t n 
= (size_t)i
; 
2502             // change the attribute 
2503             m_attrs
[n
]->DecRef(); 
2508             // remove this attribute 
2509             m_attrs
[n
]->DecRef(); 
2510             m_rowsOrCols
.RemoveAt(n
); 
2511             m_attrs
.RemoveAt(n
); 
2516 void wxGridRowOrColAttrData::UpdateAttrRowsOrCols( size_t pos
, int numRowsOrCols 
) 
2518     size_t count 
= m_attrs
.GetCount(); 
2519     for ( size_t n 
= 0; n 
< count
; n
++ ) 
2521         int & rowOrCol 
= m_rowsOrCols
[n
]; 
2522         if ( (size_t)rowOrCol 
>= pos 
) 
2524             if ( numRowsOrCols 
> 0 ) 
2526                 // If rows inserted, include row counter where necessary 
2527                 rowOrCol 
+= numRowsOrCols
; 
2529             else if ( numRowsOrCols 
< 0) 
2531                 // If rows deleted, either decrement row counter (if row still exists) 
2532                 if ((size_t)rowOrCol 
>= pos 
- numRowsOrCols
) 
2533                     rowOrCol 
+= numRowsOrCols
; 
2536                     m_rowsOrCols
.RemoveAt((size_t)n
); 
2537                     m_attrs
.RemoveAt((size_t)n
); 
2545 // ---------------------------------------------------------------------------- 
2546 // wxGridCellAttrProvider 
2547 // ---------------------------------------------------------------------------- 
2549 wxGridCellAttrProvider::wxGridCellAttrProvider() 
2551     m_data 
= (wxGridCellAttrProviderData 
*)NULL
; 
2554 wxGridCellAttrProvider::~wxGridCellAttrProvider() 
2559 void wxGridCellAttrProvider::InitData() 
2561     m_data 
= new wxGridCellAttrProviderData
; 
2564 wxGridCellAttr 
*wxGridCellAttrProvider::GetAttr(int row
, int col
, 
2565                                                 wxGridCellAttr::wxAttrKind  kind 
) const 
2567     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
2572             case (wxGridCellAttr::Any
): 
2573                 //Get cached merge attributes. 
2574                 // Currenlty not used as no cache implemented as not mutiable 
2575                 // attr = m_data->m_mergeAttr.GetAttr(row, col); 
2578                     //Basicaly implement old version. 
2579                     //Also check merge cache, so we don't have to re-merge every time.. 
2580                     wxGridCellAttr 
*attrcell 
= (wxGridCellAttr 
*)NULL
, 
2581                                    *attrrow 
= (wxGridCellAttr 
*)NULL
, 
2582                                    *attrcol 
= (wxGridCellAttr 
*)NULL
; 
2584                     attrcell 
= m_data
->m_cellAttrs
.GetAttr(row
, col
); 
2585                     attrcol 
= m_data
->m_colAttrs
.GetAttr(col
); 
2586                     attrrow 
= m_data
->m_rowAttrs
.GetAttr(row
); 
2588                     if((attrcell 
!= attrrow
) && (attrrow 
!=attrcol
) && (attrcell 
!= attrcol
)){ 
2589                         // Two or move are non NULL 
2590                         attr 
= new wxGridCellAttr
; 
2591                         attr
->SetKind(wxGridCellAttr::Merged
); 
2595                             attr
->MergeWith(attrcell
); 
2599                             attr
->MergeWith(attrcol
); 
2603                             attr
->MergeWith(attrrow
); 
2606                         //store merge attr if cache implemented 
2608                         //m_data->m_mergeAttr.SetAttr(attr, row, col); 
2612                         // one or none is non null return it or null. 
2613                         if(attrrow
) attr 
= attrrow
; 
2614                         if(attrcol
) attr 
= attrcol
; 
2615                         if(attrcell
) attr 
= attrcell
; 
2619             case (wxGridCellAttr::Cell
): 
2620                 attr 
= m_data
->m_cellAttrs
.GetAttr(row
, col
); 
2622             case (wxGridCellAttr::Col
): 
2623                  attr 
= m_data
->m_colAttrs
.GetAttr(col
); 
2625             case (wxGridCellAttr::Row
): 
2626             attr 
= m_data
->m_rowAttrs
.GetAttr(row
); 
2630                 // (wxGridCellAttr::Default): 
2631                 // (wxGridCellAttr::Merged): 
2638 void wxGridCellAttrProvider::SetAttr(wxGridCellAttr 
*attr
, 
2644     m_data
->m_cellAttrs
.SetAttr(attr
, row
, col
); 
2647 void wxGridCellAttrProvider::SetRowAttr(wxGridCellAttr 
*attr
, int row
) 
2652     m_data
->m_rowAttrs
.SetAttr(attr
, row
); 
2655 void wxGridCellAttrProvider::SetColAttr(wxGridCellAttr 
*attr
, int col
) 
2660     m_data
->m_colAttrs
.SetAttr(attr
, col
); 
2663 void wxGridCellAttrProvider::UpdateAttrRows( size_t pos
, int numRows 
) 
2667         m_data
->m_cellAttrs
.UpdateAttrRows( pos
, numRows 
); 
2669         m_data
->m_rowAttrs
.UpdateAttrRowsOrCols( pos
, numRows 
); 
2673 void wxGridCellAttrProvider::UpdateAttrCols( size_t pos
, int numCols 
) 
2677         m_data
->m_cellAttrs
.UpdateAttrCols( pos
, numCols 
); 
2679         m_data
->m_colAttrs
.UpdateAttrRowsOrCols( pos
, numCols 
); 
2683 // ---------------------------------------------------------------------------- 
2684 // wxGridTypeRegistry 
2685 // ---------------------------------------------------------------------------- 
2687 wxGridTypeRegistry::~wxGridTypeRegistry() 
2689     size_t count 
= m_typeinfo
.Count(); 
2690     for ( size_t i 
= 0; i 
< count
; i
++ ) 
2691         delete m_typeinfo
[i
]; 
2695 void wxGridTypeRegistry::RegisterDataType(const wxString
& typeName
, 
2696                                           wxGridCellRenderer
* renderer
, 
2697                                           wxGridCellEditor
* editor
) 
2699     wxGridDataTypeInfo
* info 
= new wxGridDataTypeInfo(typeName
, renderer
, editor
); 
2701     // is it already registered? 
2702     int loc 
= FindRegisteredDataType(typeName
); 
2703     if ( loc 
!= wxNOT_FOUND 
) 
2705         delete m_typeinfo
[loc
]; 
2706         m_typeinfo
[loc
] = info
; 
2710         m_typeinfo
.Add(info
); 
2714 int wxGridTypeRegistry::FindRegisteredDataType(const wxString
& typeName
) 
2716     size_t count 
= m_typeinfo
.GetCount(); 
2717     for ( size_t i 
= 0; i 
< count
; i
++ ) 
2719         if ( typeName 
== m_typeinfo
[i
]->m_typeName 
) 
2728 int wxGridTypeRegistry::FindDataType(const wxString
& typeName
) 
2730     int index 
= FindRegisteredDataType(typeName
); 
2731     if ( index 
== wxNOT_FOUND 
) 
2733         // check whether this is one of the standard ones, in which case 
2734         // register it "on the fly" 
2736         if ( typeName 
== wxGRID_VALUE_STRING 
) 
2738             RegisterDataType(wxGRID_VALUE_STRING
, 
2739                              new wxGridCellStringRenderer
, 
2740                              new wxGridCellTextEditor
); 
2742 #endif // wxUSE_TEXTCTRL 
2744         if ( typeName 
== wxGRID_VALUE_BOOL 
) 
2746             RegisterDataType(wxGRID_VALUE_BOOL
, 
2747                              new wxGridCellBoolRenderer
, 
2748                              new wxGridCellBoolEditor
); 
2750 #endif // wxUSE_CHECKBOX 
2752         if ( typeName 
== wxGRID_VALUE_NUMBER 
) 
2754             RegisterDataType(wxGRID_VALUE_NUMBER
, 
2755                              new wxGridCellNumberRenderer
, 
2756                              new wxGridCellNumberEditor
); 
2758         else if ( typeName 
== wxGRID_VALUE_FLOAT 
) 
2760             RegisterDataType(wxGRID_VALUE_FLOAT
, 
2761                              new wxGridCellFloatRenderer
, 
2762                              new wxGridCellFloatEditor
); 
2764 #endif // wxUSE_TEXTCTRL 
2766         if ( typeName 
== wxGRID_VALUE_CHOICE 
) 
2768             RegisterDataType(wxGRID_VALUE_CHOICE
, 
2769                              new wxGridCellStringRenderer
, 
2770                              new wxGridCellChoiceEditor
); 
2772 #endif // wxUSE_COMBOBOX 
2777         // we get here only if just added the entry for this type, so return 
2779         index 
= m_typeinfo
.GetCount() - 1; 
2785 int wxGridTypeRegistry::FindOrCloneDataType(const wxString
& typeName
) 
2787     int index 
= FindDataType(typeName
); 
2788     if ( index 
== wxNOT_FOUND 
) 
2790         // the first part of the typename is the "real" type, anything after ':' 
2791         // are the parameters for the renderer 
2792         index 
= FindDataType(typeName
.BeforeFirst(_T(':'))); 
2793         if ( index 
== wxNOT_FOUND 
) 
2798         wxGridCellRenderer 
*renderer 
= GetRenderer(index
); 
2799         wxGridCellRenderer 
*rendererOld 
= renderer
; 
2800         renderer 
= renderer
->Clone(); 
2801         rendererOld
->DecRef(); 
2803         wxGridCellEditor 
*editor 
= GetEditor(index
); 
2804         wxGridCellEditor 
*editorOld 
= editor
; 
2805         editor 
= editor
->Clone(); 
2806         editorOld
->DecRef(); 
2808         // do it even if there are no parameters to reset them to defaults 
2809         wxString params 
= typeName
.AfterFirst(_T(':')); 
2810         renderer
->SetParameters(params
); 
2811         editor
->SetParameters(params
); 
2813         // register the new typename 
2814         RegisterDataType(typeName
, renderer
, editor
); 
2816         // we just registered it, it's the last one 
2817         index 
= m_typeinfo
.GetCount() - 1; 
2823 wxGridCellRenderer
* wxGridTypeRegistry::GetRenderer(int index
) 
2825     wxGridCellRenderer
* renderer 
= m_typeinfo
[index
]->m_renderer
; 
2831 wxGridCellEditor
* wxGridTypeRegistry::GetEditor(int index
) 
2833     wxGridCellEditor
* editor 
= m_typeinfo
[index
]->m_editor
; 
2839 // ---------------------------------------------------------------------------- 
2841 // ---------------------------------------------------------------------------- 
2843 IMPLEMENT_ABSTRACT_CLASS( wxGridTableBase
, wxObject 
) 
2846 wxGridTableBase::wxGridTableBase() 
2848     m_view 
= (wxGrid 
*) NULL
; 
2849     m_attrProvider 
= (wxGridCellAttrProvider 
*) NULL
; 
2852 wxGridTableBase::~wxGridTableBase() 
2854     delete m_attrProvider
; 
2857 void wxGridTableBase::SetAttrProvider(wxGridCellAttrProvider 
*attrProvider
) 
2859     delete m_attrProvider
; 
2860     m_attrProvider 
= attrProvider
; 
2863 bool wxGridTableBase::CanHaveAttributes() 
2865     if ( ! GetAttrProvider() ) 
2867         // use the default attr provider by default 
2868         SetAttrProvider(new wxGridCellAttrProvider
); 
2873 wxGridCellAttr 
*wxGridTableBase::GetAttr(int row
, int col
, wxGridCellAttr::wxAttrKind  kind
) 
2875     if ( m_attrProvider 
) 
2876         return m_attrProvider
->GetAttr(row
, col
, kind
); 
2878         return (wxGridCellAttr 
*)NULL
; 
2881 void wxGridTableBase::SetAttr(wxGridCellAttr
* attr
, int row
, int col
) 
2883     if ( m_attrProvider 
) 
2885         attr
->SetKind(wxGridCellAttr::Cell
); 
2886         m_attrProvider
->SetAttr(attr
, row
, col
); 
2890         // as we take ownership of the pointer and don't store it, we must 
2896 void wxGridTableBase::SetRowAttr(wxGridCellAttr 
*attr
, int row
) 
2898     if ( m_attrProvider 
) 
2900         attr
->SetKind(wxGridCellAttr::Row
); 
2901         m_attrProvider
->SetRowAttr(attr
, row
); 
2905         // as we take ownership of the pointer and don't store it, we must 
2911 void wxGridTableBase::SetColAttr(wxGridCellAttr 
*attr
, int col
) 
2913     if ( m_attrProvider 
) 
2915         attr
->SetKind(wxGridCellAttr::Col
); 
2916         m_attrProvider
->SetColAttr(attr
, col
); 
2920         // as we take ownership of the pointer and don't store it, we must 
2926 bool wxGridTableBase::InsertRows( size_t WXUNUSED(pos
), 
2927                                   size_t WXUNUSED(numRows
) ) 
2929     wxFAIL_MSG( wxT("Called grid table class function InsertRows\nbut your derived table class does not override this function") ); 
2934 bool wxGridTableBase::AppendRows( size_t WXUNUSED(numRows
) ) 
2936     wxFAIL_MSG( wxT("Called grid table class function AppendRows\nbut your derived table class does not override this function")); 
2941 bool wxGridTableBase::DeleteRows( size_t WXUNUSED(pos
), 
2942                                   size_t WXUNUSED(numRows
) ) 
2944     wxFAIL_MSG( wxT("Called grid table class function DeleteRows\nbut your derived table class does not override this function")); 
2949 bool wxGridTableBase::InsertCols( size_t WXUNUSED(pos
), 
2950                                   size_t WXUNUSED(numCols
) ) 
2952     wxFAIL_MSG( wxT("Called grid table class function InsertCols\nbut your derived table class does not override this function")); 
2957 bool wxGridTableBase::AppendCols( size_t WXUNUSED(numCols
) ) 
2959     wxFAIL_MSG(wxT("Called grid table class function AppendCols\nbut your derived table class does not override this function")); 
2964 bool wxGridTableBase::DeleteCols( size_t WXUNUSED(pos
), 
2965                                   size_t WXUNUSED(numCols
) ) 
2967     wxFAIL_MSG( wxT("Called grid table class function DeleteCols\nbut your derived table class does not override this function")); 
2973 wxString 
wxGridTableBase::GetRowLabelValue( int row 
) 
2976     s 
<< row 
+ 1; // RD: Starting the rows at zero confuses users, no matter 
2977                   //     how much it makes sense to us geeks. 
2981 wxString 
wxGridTableBase::GetColLabelValue( int col 
) 
2983     // default col labels are: 
2984     //   cols 0 to 25   : A-Z 
2985     //   cols 26 to 675 : AA-ZZ 
2990     for ( n 
= 1; ; n
++ ) 
2992         s 
+= (_T('A') + (wxChar
)( col%26 
)); 
2994         if ( col 
< 0 ) break; 
2997     // reverse the string... 
2999     for ( i 
= 0;  i 
< n
;  i
++ ) 
3008 wxString 
wxGridTableBase::GetTypeName( int WXUNUSED(row
), int WXUNUSED(col
) ) 
3010     return wxGRID_VALUE_STRING
; 
3013 bool wxGridTableBase::CanGetValueAs( int WXUNUSED(row
), int WXUNUSED(col
), 
3014                                      const wxString
& typeName 
) 
3016     return typeName 
== wxGRID_VALUE_STRING
; 
3019 bool wxGridTableBase::CanSetValueAs( int row
, int col
, const wxString
& typeName 
) 
3021     return CanGetValueAs(row
, col
, typeName
); 
3024 long wxGridTableBase::GetValueAsLong( int WXUNUSED(row
), int WXUNUSED(col
) ) 
3029 double wxGridTableBase::GetValueAsDouble( int WXUNUSED(row
), int WXUNUSED(col
) ) 
3034 bool wxGridTableBase::GetValueAsBool( int WXUNUSED(row
), int WXUNUSED(col
) ) 
3039 void wxGridTableBase::SetValueAsLong( int WXUNUSED(row
), int WXUNUSED(col
), 
3040                                       long WXUNUSED(value
) ) 
3044 void wxGridTableBase::SetValueAsDouble( int WXUNUSED(row
), int WXUNUSED(col
), 
3045                                         double WXUNUSED(value
) ) 
3049 void wxGridTableBase::SetValueAsBool( int WXUNUSED(row
), int WXUNUSED(col
), 
3050                                       bool WXUNUSED(value
) ) 
3055 void* wxGridTableBase::GetValueAsCustom( int WXUNUSED(row
), int WXUNUSED(col
), 
3056                                          const wxString
& WXUNUSED(typeName
) ) 
3061 void  wxGridTableBase::SetValueAsCustom( int WXUNUSED(row
), int WXUNUSED(col
), 
3062                                          const wxString
& WXUNUSED(typeName
), 
3063                                          void* WXUNUSED(value
) ) 
3067 ////////////////////////////////////////////////////////////////////// 
3069 // Message class for the grid table to send requests and notifications 
3073 wxGridTableMessage::wxGridTableMessage() 
3075     m_table 
= (wxGridTableBase 
*) NULL
; 
3081 wxGridTableMessage::wxGridTableMessage( wxGridTableBase 
*table
, int id
, 
3082                                         int commandInt1
, int commandInt2 
) 
3086     m_comInt1 
= commandInt1
; 
3087     m_comInt2 
= commandInt2
; 
3092 ////////////////////////////////////////////////////////////////////// 
3094 // A basic grid table for string data. An object of this class will 
3095 // created by wxGrid if you don't specify an alternative table class. 
3098 WX_DEFINE_OBJARRAY(wxGridStringArray
) 
3100 IMPLEMENT_DYNAMIC_CLASS( wxGridStringTable
, wxGridTableBase 
) 
3102 wxGridStringTable::wxGridStringTable() 
3107 wxGridStringTable::wxGridStringTable( int numRows
, int numCols 
) 
3110     m_data
.Alloc( numRows 
); 
3113     sa
.Alloc( numCols 
); 
3114     sa
.Add( wxEmptyString
, numCols 
); 
3116     m_data
.Add( sa
, numRows 
); 
3119 wxGridStringTable::~wxGridStringTable() 
3123 int wxGridStringTable::GetNumberRows() 
3125     return m_data
.GetCount(); 
3128 int wxGridStringTable::GetNumberCols() 
3130     if ( m_data
.GetCount() > 0 ) 
3131         return m_data
[0].GetCount(); 
3136 wxString 
wxGridStringTable::GetValue( int row
, int col 
) 
3138     wxASSERT_MSG( (row 
< GetNumberRows()) && (col 
< GetNumberCols()), 
3139                   _T("invalid row or column index in wxGridStringTable") ); 
3141     return m_data
[row
][col
]; 
3144 void wxGridStringTable::SetValue( int row
, int col
, const wxString
& value 
) 
3146     wxASSERT_MSG( (row 
< GetNumberRows()) && (col 
< GetNumberCols()), 
3147                   _T("invalid row or column index in wxGridStringTable") ); 
3149     m_data
[row
][col
] = value
; 
3152 bool wxGridStringTable::IsEmptyCell( int row
, int col 
) 
3154     wxASSERT_MSG( (row 
< GetNumberRows()) && (col 
< GetNumberCols()), 
3155                   _T("invalid row or column index in wxGridStringTable") ); 
3157     return (m_data
[row
][col
] == wxEmptyString
); 
3160 void wxGridStringTable::Clear() 
3163     int numRows
, numCols
; 
3165     numRows 
= m_data
.GetCount(); 
3168         numCols 
= m_data
[0].GetCount(); 
3170         for ( row 
= 0;  row 
< numRows
;  row
++ ) 
3172             for ( col 
= 0;  col 
< numCols
;  col
++ ) 
3174                 m_data
[row
][col
] = wxEmptyString
; 
3181 bool wxGridStringTable::InsertRows( size_t pos
, size_t numRows 
) 
3183     size_t curNumRows 
= m_data
.GetCount(); 
3184     size_t curNumCols 
= ( curNumRows 
> 0 ? m_data
[0].GetCount() : 
3185                           ( GetView() ? GetView()->GetNumberCols() : 0 ) ); 
3187     if ( pos 
>= curNumRows 
) 
3189         return AppendRows( numRows 
); 
3193     sa
.Alloc( curNumCols 
); 
3194     sa
.Add( wxEmptyString
, curNumCols 
); 
3195     m_data
.Insert( sa
, pos
, numRows 
); 
3198         wxGridTableMessage 
msg( this, 
3199                                 wxGRIDTABLE_NOTIFY_ROWS_INSERTED
, 
3203         GetView()->ProcessTableMessage( msg 
); 
3209 bool wxGridStringTable::AppendRows( size_t numRows 
) 
3211     size_t curNumRows 
= m_data
.GetCount(); 
3212     size_t curNumCols 
= ( curNumRows 
> 0 ? m_data
[0].GetCount() : 
3213                           ( GetView() ? GetView()->GetNumberCols() : 0 ) ); 
3216     if ( curNumCols 
> 0 ) 
3218         sa
.Alloc( curNumCols 
); 
3219         sa
.Add( wxEmptyString
, curNumCols 
); 
3222     m_data
.Add( sa
, numRows 
); 
3226         wxGridTableMessage 
msg( this, 
3227                                 wxGRIDTABLE_NOTIFY_ROWS_APPENDED
, 
3230         GetView()->ProcessTableMessage( msg 
); 
3236 bool wxGridStringTable::DeleteRows( size_t pos
, size_t numRows 
) 
3238     size_t curNumRows 
= m_data
.GetCount(); 
3240     if ( pos 
>= curNumRows 
) 
3242         wxFAIL_MSG( wxString::Format
 
3244                         wxT("Called wxGridStringTable::DeleteRows(pos=%lu, N=%lu)\nPos value is invalid for present table with %lu rows"), 
3246                         (unsigned long)numRows
, 
3247                         (unsigned long)curNumRows
 
3253     if ( numRows 
> curNumRows 
- pos 
) 
3255         numRows 
= curNumRows 
- pos
; 
3258     if ( numRows 
>= curNumRows 
) 
3264         m_data
.RemoveAt( pos
, numRows 
); 
3268         wxGridTableMessage 
msg( this, 
3269                                 wxGRIDTABLE_NOTIFY_ROWS_DELETED
, 
3273         GetView()->ProcessTableMessage( msg 
); 
3279 bool wxGridStringTable::InsertCols( size_t pos
, size_t numCols 
) 
3283     size_t curNumRows 
= m_data
.GetCount(); 
3284     size_t curNumCols 
= ( curNumRows 
> 0 ? m_data
[0].GetCount() : 
3285                           ( GetView() ? GetView()->GetNumberCols() : 0 ) ); 
3287     if ( pos 
>= curNumCols 
) 
3289         return AppendCols( numCols 
); 
3292     for ( row 
= 0;  row 
< curNumRows
;  row
++ ) 
3294         for ( col 
= pos
;  col 
< pos 
+ numCols
;  col
++ ) 
3296             m_data
[row
].Insert( wxEmptyString
, col 
); 
3301         wxGridTableMessage 
msg( this, 
3302                                 wxGRIDTABLE_NOTIFY_COLS_INSERTED
, 
3306         GetView()->ProcessTableMessage( msg 
); 
3312 bool wxGridStringTable::AppendCols( size_t numCols 
) 
3316     size_t curNumRows 
= m_data
.GetCount(); 
3320         // TODO: something better than this ? 
3322         wxFAIL_MSG( wxT("Unable to append cols to a grid table with no rows.\nCall AppendRows() first") ); 
3327     for ( row 
= 0;  row 
< curNumRows
;  row
++ ) 
3329         m_data
[row
].Add( wxEmptyString
, numCols 
); 
3334         wxGridTableMessage 
msg( this, 
3335                                 wxGRIDTABLE_NOTIFY_COLS_APPENDED
, 
3338         GetView()->ProcessTableMessage( msg 
); 
3344 bool wxGridStringTable::DeleteCols( size_t pos
, size_t numCols 
) 
3348     size_t curNumRows 
= m_data
.GetCount(); 
3349     size_t curNumCols 
= ( curNumRows 
> 0 ? m_data
[0].GetCount() : 
3350                           ( GetView() ? GetView()->GetNumberCols() : 0 ) ); 
3352     if ( pos 
>= curNumCols 
) 
3354         wxFAIL_MSG( wxString::Format
 
3356                         wxT("Called wxGridStringTable::DeleteCols(pos=%lu, N=%lu)\nPos value is invalid for present table with %lu cols"), 
3358                         (unsigned long)numCols
, 
3359                         (unsigned long)curNumCols
 
3364     if ( numCols 
> curNumCols 
- pos 
) 
3366         numCols 
= curNumCols 
- pos
; 
3369     for ( row 
= 0;  row 
< curNumRows
;  row
++ ) 
3371         if ( numCols 
>= curNumCols 
) 
3373             m_data
[row
].Clear(); 
3377             m_data
[row
].RemoveAt( pos
, numCols 
); 
3382         wxGridTableMessage 
msg( this, 
3383                                 wxGRIDTABLE_NOTIFY_COLS_DELETED
, 
3387         GetView()->ProcessTableMessage( msg 
); 
3393 wxString 
wxGridStringTable::GetRowLabelValue( int row 
) 
3395     if ( row 
> (int)(m_rowLabels
.GetCount()) - 1 ) 
3397         // using default label 
3399         return wxGridTableBase::GetRowLabelValue( row 
); 
3403         return m_rowLabels
[ row 
]; 
3407 wxString 
wxGridStringTable::GetColLabelValue( int col 
) 
3409     if ( col 
> (int)(m_colLabels
.GetCount()) - 1 ) 
3411         // using default label 
3413         return wxGridTableBase::GetColLabelValue( col 
); 
3417         return m_colLabels
[ col 
]; 
3421 void wxGridStringTable::SetRowLabelValue( int row
, const wxString
& value 
) 
3423     if ( row 
> (int)(m_rowLabels
.GetCount()) - 1 ) 
3425         int n 
= m_rowLabels
.GetCount(); 
3427         for ( i 
= n
;  i 
<= row
;  i
++ ) 
3429             m_rowLabels
.Add( wxGridTableBase::GetRowLabelValue(i
) ); 
3433     m_rowLabels
[row
] = value
; 
3436 void wxGridStringTable::SetColLabelValue( int col
, const wxString
& value 
) 
3438     if ( col 
> (int)(m_colLabels
.GetCount()) - 1 ) 
3440         int n 
= m_colLabels
.GetCount(); 
3442         for ( i 
= n
;  i 
<= col
;  i
++ ) 
3444             m_colLabels
.Add( wxGridTableBase::GetColLabelValue(i
) ); 
3448     m_colLabels
[col
] = value
; 
3453 ////////////////////////////////////////////////////////////////////// 
3454 ////////////////////////////////////////////////////////////////////// 
3456 IMPLEMENT_DYNAMIC_CLASS( wxGridRowLabelWindow
, wxWindow 
) 
3458 BEGIN_EVENT_TABLE( wxGridRowLabelWindow
, wxWindow 
) 
3459     EVT_PAINT( wxGridRowLabelWindow::OnPaint 
) 
3460     EVT_MOUSEWHEEL( wxGridRowLabelWindow::OnMouseWheel
) 
3461     EVT_MOUSE_EVENTS( wxGridRowLabelWindow::OnMouseEvent 
) 
3462     EVT_KEY_DOWN( wxGridRowLabelWindow::OnKeyDown 
) 
3463     EVT_KEY_UP( wxGridRowLabelWindow::OnKeyUp 
) 
3466 wxGridRowLabelWindow::wxGridRowLabelWindow( wxGrid 
*parent
, 
3468                                             const wxPoint 
&pos
, const wxSize 
&size 
) 
3469   : wxWindow( parent
, id
, pos
, size
, wxWANTS_CHARS
|wxBORDER_NONE 
) 
3474 void wxGridRowLabelWindow::OnPaint( wxPaintEvent
& WXUNUSED(event
) ) 
3478     // NO - don't do this because it will set both the x and y origin 
3479     // coords to match the parent scrolled window and we just want to 
3480     // set the y coord  - MB 
3482     // m_owner->PrepareDC( dc ); 
3485     m_owner
->CalcUnscrolledPosition( 0, 0, &x
, &y 
); 
3486     dc
.SetDeviceOrigin( 0, -y 
); 
3488     wxArrayInt rows 
= m_owner
->CalcRowLabelsExposed( GetUpdateRegion() ); 
3489     m_owner
->DrawRowLabels( dc 
, rows 
); 
3493 void wxGridRowLabelWindow::OnMouseEvent( wxMouseEvent
& event 
) 
3495     m_owner
->ProcessRowLabelMouseEvent( event 
); 
3499 void wxGridRowLabelWindow::OnMouseWheel( wxMouseEvent
& event 
) 
3501     m_owner
->GetEventHandler()->ProcessEvent(event
); 
3505 // This seems to be required for wxMotif otherwise the mouse 
3506 // cursor must be in the cell edit control to get key events 
3508 void wxGridRowLabelWindow::OnKeyDown( wxKeyEvent
& event 
) 
3510     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3513 void wxGridRowLabelWindow::OnKeyUp( wxKeyEvent
& event 
) 
3515     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3520 ////////////////////////////////////////////////////////////////////// 
3522 IMPLEMENT_DYNAMIC_CLASS( wxGridColLabelWindow
, wxWindow 
) 
3524 BEGIN_EVENT_TABLE( wxGridColLabelWindow
, wxWindow 
) 
3525     EVT_PAINT( wxGridColLabelWindow::OnPaint 
) 
3526     EVT_MOUSEWHEEL( wxGridColLabelWindow::OnMouseWheel
) 
3527     EVT_MOUSE_EVENTS( wxGridColLabelWindow::OnMouseEvent 
) 
3528     EVT_KEY_DOWN( wxGridColLabelWindow::OnKeyDown 
) 
3529     EVT_KEY_UP( wxGridColLabelWindow::OnKeyUp 
) 
3532 wxGridColLabelWindow::wxGridColLabelWindow( wxGrid 
*parent
, 
3534                                             const wxPoint 
&pos
, const wxSize 
&size 
) 
3535   : wxWindow( parent
, id
, pos
, size
, wxWANTS_CHARS
|wxBORDER_NONE 
) 
3540 void wxGridColLabelWindow::OnPaint( wxPaintEvent
& WXUNUSED(event
) ) 
3544     // NO - don't do this because it will set both the x and y origin 
3545     // coords to match the parent scrolled window and we just want to 
3546     // set the x coord  - MB 
3548     // m_owner->PrepareDC( dc ); 
3551     m_owner
->CalcUnscrolledPosition( 0, 0, &x
, &y 
); 
3552     dc
.SetDeviceOrigin( -x
, 0 ); 
3554     wxArrayInt cols 
= m_owner
->CalcColLabelsExposed( GetUpdateRegion() ); 
3555     m_owner
->DrawColLabels( dc 
, cols 
); 
3559 void wxGridColLabelWindow::OnMouseEvent( wxMouseEvent
& event 
) 
3561     m_owner
->ProcessColLabelMouseEvent( event 
); 
3564 void wxGridColLabelWindow::OnMouseWheel( wxMouseEvent
& event 
) 
3566     m_owner
->GetEventHandler()->ProcessEvent(event
); 
3570 // This seems to be required for wxMotif otherwise the mouse 
3571 // cursor must be in the cell edit control to get key events 
3573 void wxGridColLabelWindow::OnKeyDown( wxKeyEvent
& event 
) 
3575     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3578 void wxGridColLabelWindow::OnKeyUp( wxKeyEvent
& event 
) 
3580     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3585 ////////////////////////////////////////////////////////////////////// 
3587 IMPLEMENT_DYNAMIC_CLASS( wxGridCornerLabelWindow
, wxWindow 
) 
3589 BEGIN_EVENT_TABLE( wxGridCornerLabelWindow
, wxWindow 
) 
3590     EVT_MOUSEWHEEL( wxGridCornerLabelWindow::OnMouseWheel
) 
3591     EVT_MOUSE_EVENTS( wxGridCornerLabelWindow::OnMouseEvent 
) 
3592     EVT_PAINT( wxGridCornerLabelWindow::OnPaint
) 
3593     EVT_KEY_DOWN( wxGridCornerLabelWindow::OnKeyDown 
) 
3594     EVT_KEY_UP( wxGridCornerLabelWindow::OnKeyUp 
) 
3597 wxGridCornerLabelWindow::wxGridCornerLabelWindow( wxGrid 
*parent
, 
3599                                                   const wxPoint 
&pos
, const wxSize 
&size 
) 
3600   : wxWindow( parent
, id
, pos
, size
, wxWANTS_CHARS
|wxBORDER_NONE 
) 
3605 void wxGridCornerLabelWindow::OnPaint( wxPaintEvent
& WXUNUSED(event
) ) 
3609     int client_height 
= 0; 
3610     int client_width 
= 0; 
3611     GetClientSize( &client_width
, &client_height 
); 
3613     dc
.SetPen( wxPen(wxSystemSettings::GetColour(wxSYS_COLOUR_3DDKSHADOW
),1, wxSOLID
) ); 
3614     dc
.DrawLine( client_width
-1, client_height
-1, client_width
-1, 0 ); 
3615     dc
.DrawLine( client_width
-1, client_height
-1, 0, client_height
-1 ); 
3616     dc
.DrawLine( 0, 0, client_width
, 0 ); 
3617     dc
.DrawLine( 0, 0, 0, client_height 
); 
3619     dc
.SetPen( *wxWHITE_PEN 
); 
3620     dc
.DrawLine( 1, 1, client_width
-1, 1 ); 
3621     dc
.DrawLine( 1, 1, 1, client_height
-1 ); 
3625 void wxGridCornerLabelWindow::OnMouseEvent( wxMouseEvent
& event 
) 
3627     m_owner
->ProcessCornerLabelMouseEvent( event 
); 
3631 void wxGridCornerLabelWindow::OnMouseWheel( wxMouseEvent
& event 
) 
3633     m_owner
->GetEventHandler()->ProcessEvent(event
); 
3636 // This seems to be required for wxMotif otherwise the mouse 
3637 // cursor must be in the cell edit control to get key events 
3639 void wxGridCornerLabelWindow::OnKeyDown( wxKeyEvent
& event 
) 
3641     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3644 void wxGridCornerLabelWindow::OnKeyUp( wxKeyEvent
& event 
) 
3646     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3651 ////////////////////////////////////////////////////////////////////// 
3653 IMPLEMENT_DYNAMIC_CLASS( wxGridWindow
, wxWindow 
) 
3655 BEGIN_EVENT_TABLE( wxGridWindow
, wxWindow 
) 
3656     EVT_PAINT( wxGridWindow::OnPaint 
) 
3657     EVT_MOUSEWHEEL( wxGridWindow::OnMouseWheel
) 
3658     EVT_MOUSE_EVENTS( wxGridWindow::OnMouseEvent 
) 
3659     EVT_KEY_DOWN( wxGridWindow::OnKeyDown 
) 
3660     EVT_KEY_UP( wxGridWindow::OnKeyUp 
) 
3661     EVT_ERASE_BACKGROUND( wxGridWindow::OnEraseBackground 
) 
3664 wxGridWindow::wxGridWindow( wxGrid 
*parent
, 
3665                             wxGridRowLabelWindow 
*rowLblWin
, 
3666                             wxGridColLabelWindow 
*colLblWin
, 
3669                             const wxSize 
&size 
) 
3670             : wxWindow( parent
, id
, pos
, size
, wxWANTS_CHARS 
| wxBORDER_NONE 
| wxCLIP_CHILDREN
, 
3671                         wxT("grid window") ) 
3675     m_rowLabelWin 
= rowLblWin
; 
3676     m_colLabelWin 
= colLblWin
; 
3677     SetBackgroundColour(_T("WHITE")); 
3681 wxGridWindow::~wxGridWindow() 
3686 void wxGridWindow::OnPaint( wxPaintEvent 
&WXUNUSED(event
) ) 
3688     wxPaintDC 
dc( this ); 
3689     m_owner
->PrepareDC( dc 
); 
3690     wxRegion reg 
= GetUpdateRegion(); 
3691     wxGridCellCoordsArray DirtyCells 
= m_owner
->CalcCellsExposed( reg 
); 
3692     m_owner
->DrawGridCellArea( dc 
, DirtyCells
); 
3693 #if WXGRID_DRAW_LINES 
3694     m_owner
->DrawAllGridLines( dc
, reg 
); 
3696     m_owner
->DrawGridSpace( dc 
); 
3697     m_owner
->DrawHighlight( dc 
, DirtyCells 
); 
3701 void wxGridWindow::ScrollWindow( int dx
, int dy
, const wxRect 
*rect 
) 
3703     wxWindow::ScrollWindow( dx
, dy
, rect 
); 
3704     m_rowLabelWin
->ScrollWindow( 0, dy
, rect 
); 
3705     m_colLabelWin
->ScrollWindow( dx
, 0, rect 
); 
3709 void wxGridWindow::OnMouseEvent( wxMouseEvent
& event 
) 
3711     m_owner
->ProcessGridCellMouseEvent( event 
); 
3714 void wxGridWindow::OnMouseWheel( wxMouseEvent
& event 
) 
3716     m_owner
->GetEventHandler()->ProcessEvent(event
); 
3719 // This seems to be required for wxMotif/wxGTK otherwise the mouse 
3720 // cursor must be in the cell edit control to get key events 
3722 void wxGridWindow::OnKeyDown( wxKeyEvent
& event 
) 
3724     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3727 void wxGridWindow::OnKeyUp( wxKeyEvent
& event 
) 
3729     if ( !m_owner
->GetEventHandler()->ProcessEvent( event 
) ) event
.Skip(); 
3732 void wxGridWindow::OnEraseBackground( wxEraseEvent
& WXUNUSED(event
) ) 
3737 ////////////////////////////////////////////////////////////////////// 
3739 // Internal Helper function for computing row or column from some 
3740 // (unscrolled) coordinate value, using either 
3741 // m_defaultRowHeight/m_defaultColWidth or binary search on array 
3742 // of m_rowBottoms/m_ColRights to speed up the search! 
3744 // Internal helper macros for simpler use of that function 
3746 static int CoordToRowOrCol(int coord
, int defaultDist
, int minDist
, 
3747                            const wxArrayInt
& BorderArray
, int nMax
, 
3750 #define internalXToCol(x) CoordToRowOrCol(x, m_defaultColWidth, \ 
3751                                           m_minAcceptableColWidth, \ 
3752                                           m_colRights, m_numCols, TRUE) 
3753 #define internalYToRow(y) CoordToRowOrCol(y, m_defaultRowHeight, \ 
3754                                           m_minAcceptableRowHeight, \ 
3755                                           m_rowBottoms, m_numRows, TRUE) 
3756 ///////////////////////////////////////////////////////////////////// 
3758 IMPLEMENT_DYNAMIC_CLASS( wxGrid
, wxScrolledWindow 
) 
3760 BEGIN_EVENT_TABLE( wxGrid
, wxScrolledWindow 
) 
3761     EVT_PAINT( wxGrid::OnPaint 
) 
3762     EVT_SIZE( wxGrid::OnSize 
) 
3763     EVT_KEY_DOWN( wxGrid::OnKeyDown 
) 
3764     EVT_KEY_UP( wxGrid::OnKeyUp 
) 
3765     EVT_ERASE_BACKGROUND( wxGrid::OnEraseBackground 
) 
3768 wxGrid::wxGrid( wxWindow 
*parent
, 
3773                  const wxString
& name 
) 
3774   : wxScrolledWindow( parent
, id
, pos
, size
, (style 
| wxWANTS_CHARS
), name 
), 
3775     m_colMinWidths(GRID_HASH_SIZE
), 
3776     m_rowMinHeights(GRID_HASH_SIZE
) 
3784     // Must do this or ~wxScrollHelper will pop the wrong event handler 
3785     SetTargetWindow(this); 
3787     wxSafeDecRef(m_defaultCellAttr
); 
3789 #ifdef DEBUG_ATTR_CACHE 
3790     size_t total 
= gs_nAttrCacheHits 
+ gs_nAttrCacheMisses
; 
3791     wxPrintf(_T("wxGrid attribute cache statistics: " 
3792                 "total: %u, hits: %u (%u%%)\n"), 
3793              total
, gs_nAttrCacheHits
, 
3794              total 
? (gs_nAttrCacheHits
*100) / total 
: 0); 
3800     delete m_typeRegistry
; 
3806 // ----- internal init and update functions 
3809 void wxGrid::Create() 
3811     m_created 
= FALSE
;    // set to TRUE by CreateGrid 
3813     m_table        
= (wxGridTableBase 
*) NULL
; 
3816     m_cellEditCtrlEnabled 
= FALSE
; 
3818     m_defaultCellAttr 
= new wxGridCellAttr(); 
3820     // Set default cell attributes 
3821     m_defaultCellAttr
->SetDefAttr(m_defaultCellAttr
); 
3822     m_defaultCellAttr
->SetKind(wxGridCellAttr::Default
); 
3823     m_defaultCellAttr
->SetFont(GetFont()); 
3824     m_defaultCellAttr
->SetAlignment(wxALIGN_LEFT
, wxALIGN_TOP
); 
3825     m_defaultCellAttr
->SetTextColour( 
3826         wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT
)); 
3827     m_defaultCellAttr
->SetBackgroundColour( 
3828         wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW
)); 
3829     m_defaultCellAttr
->SetRenderer(new wxGridCellStringRenderer
); 
3830     m_defaultCellAttr
->SetEditor(new wxGridCellTextEditor
); 
3835     m_currentCellCoords 
= wxGridNoCellCoords
; 
3837     m_rowLabelWidth 
= WXGRID_DEFAULT_ROW_LABEL_WIDTH
; 
3838     m_colLabelHeight 
= WXGRID_DEFAULT_COL_LABEL_HEIGHT
; 
3840     // create the type registry 
3841     m_typeRegistry 
= new wxGridTypeRegistry
; 
3844     // subwindow components that make up the wxGrid 
3845     m_cornerLabelWin 
= new wxGridCornerLabelWindow( this, 
3850     m_rowLabelWin 
= new wxGridRowLabelWindow( this, 
3855     m_colLabelWin 
= new wxGridColLabelWindow( this, 
3860     m_gridWin 
= new wxGridWindow( this, 
3867     SetTargetWindow( m_gridWin 
); 
3873 bool wxGrid::CreateGrid( int numRows
, int numCols
, 
3874                          wxGrid::wxGridSelectionModes selmode 
) 
3876     wxCHECK_MSG( !m_created
, 
3878                  wxT("wxGrid::CreateGrid or wxGrid::SetTable called more than once") ); 
3880     m_numRows 
= numRows
; 
3881     m_numCols 
= numCols
; 
3883     m_table 
= new wxGridStringTable( m_numRows
, m_numCols 
); 
3884     m_table
->SetView( this ); 
3886     m_selection 
= new wxGridSelection( this, selmode 
); 
3895 void wxGrid::SetSelectionMode(wxGrid::wxGridSelectionModes selmode
) 
3897     wxCHECK_RET( m_created
, 
3898                  wxT("Called wxGrid::SetSelectionMode() before calling CreateGrid()") ); 
3900     m_selection
->SetSelectionMode( selmode 
); 
3903 wxGrid::wxGridSelectionModes 
wxGrid::GetSelectionMode() const 
3905     wxCHECK_MSG( m_created
, wxGrid::wxGridSelectCells
, 
3906                  wxT("Called wxGrid::GetSelectionMode() before calling CreateGrid()") ); 
3908     return m_selection
->GetSelectionMode(); 
3911 bool wxGrid::SetTable( wxGridTableBase 
*table
, bool takeOwnership
, 
3912                        wxGrid::wxGridSelectionModes selmode 
) 
3916         // RD: Actually, this should probably be allowed.  I think it would be 
3917         //     nice to be able to switch multiple Tables in and out of a single 
3918         //     View at runtime.  Is there anything in the implementation that 
3919         //     would prevent this? 
3921         // At least, you now have to cope with m_selection 
3922         wxFAIL_MSG( wxT("wxGrid::CreateGrid or wxGrid::SetTable called more than once") ); 
3927         m_numRows 
= table
->GetNumberRows(); 
3928         m_numCols 
= table
->GetNumberCols(); 
3931         m_table
->SetView( this ); 
3934         m_selection 
= new wxGridSelection( this, selmode 
); 
3947     m_rowLabelWidth  
= WXGRID_DEFAULT_ROW_LABEL_WIDTH
; 
3948     m_colLabelHeight 
= WXGRID_DEFAULT_COL_LABEL_HEIGHT
; 
3950     if ( m_rowLabelWin 
) 
3952         m_labelBackgroundColour 
= m_rowLabelWin
->GetBackgroundColour(); 
3956         m_labelBackgroundColour 
= wxColour( _T("WHITE") ); 
3959     m_labelTextColour 
= wxColour( _T("BLACK") ); 
3962     m_attrCache
.row 
= -1; 
3963     m_attrCache
.col 
= -1; 
3964     m_attrCache
.attr 
= NULL
; 
3966     // TODO: something better than this ? 
3968     m_labelFont 
= this->GetFont(); 
3969     m_labelFont
.SetWeight( wxBOLD 
); 
3971     m_rowLabelHorizAlign 
= wxALIGN_CENTRE
; 
3972     m_rowLabelVertAlign  
= wxALIGN_CENTRE
; 
3974     m_colLabelHorizAlign 
= wxALIGN_CENTRE
; 
3975     m_colLabelVertAlign  
= wxALIGN_CENTRE
; 
3976     m_colLabelTextOrientation 
= wxHORIZONTAL
; 
3978     m_defaultColWidth  
= WXGRID_DEFAULT_COL_WIDTH
; 
3979     m_defaultRowHeight 
= m_gridWin
->GetCharHeight(); 
3981     m_minAcceptableColWidth  
= WXGRID_MIN_COL_WIDTH
; 
3982     m_minAcceptableRowHeight 
= WXGRID_MIN_ROW_HEIGHT
; 
3984 #if defined(__WXMOTIF__) || defined(__WXGTK__)  // see also text ctrl sizing in ShowCellEditControl() 
3985     m_defaultRowHeight 
+= 8; 
3987     m_defaultRowHeight 
+= 4; 
3990     m_gridLineColour 
= wxColour( 192,192,192 ); 
3991     m_gridLinesEnabled 
= TRUE
; 
3992     m_cellHighlightColour 
= *wxBLACK
; 
3993     m_cellHighlightPenWidth 
= 2; 
3994     m_cellHighlightROPenWidth 
= 1; 
3996     m_cursorMode  
= WXGRID_CURSOR_SELECT_CELL
; 
3997     m_winCapture 
= (wxWindow 
*)NULL
; 
3998     m_canDragRowSize 
= TRUE
; 
3999     m_canDragColSize 
= TRUE
; 
4000     m_canDragGridSize 
= TRUE
; 
4002     m_dragRowOrCol 
= -1; 
4003     m_isDragging 
= FALSE
; 
4004     m_startDragPos 
= wxDefaultPosition
; 
4006     m_waitForSlowClick 
= FALSE
; 
4008     m_rowResizeCursor 
= wxCursor( wxCURSOR_SIZENS 
); 
4009     m_colResizeCursor 
= wxCursor( wxCURSOR_SIZEWE 
); 
4011     m_currentCellCoords 
= wxGridNoCellCoords
; 
4013     m_selectingTopLeft 
= wxGridNoCellCoords
; 
4014     m_selectingBottomRight 
= wxGridNoCellCoords
; 
4015     m_selectionBackground 
= wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHT
); 
4016     m_selectionForeground 
= wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHTTEXT
); 
4018     m_editable 
= TRUE
;  // default for whole grid 
4020     m_inOnKeyDown 
= FALSE
; 
4027 // ---------------------------------------------------------------------------- 
4028 // the idea is to call these functions only when necessary because they create 
4029 // quite big arrays which eat memory mostly unnecessary - in particular, if 
4030 // default widths/heights are used for all rows/columns, we may not use these 
4033 // with some extra code, it should be possible to only store the 
4034 // widths/heights different from default ones but this will be done later... 
4035 // ---------------------------------------------------------------------------- 
4037 void wxGrid::InitRowHeights() 
4039     m_rowHeights
.Empty(); 
4040     m_rowBottoms
.Empty(); 
4042     m_rowHeights
.Alloc( m_numRows 
); 
4043     m_rowBottoms
.Alloc( m_numRows 
); 
4047     m_rowHeights
.Add( m_defaultRowHeight
, m_numRows 
); 
4049     for ( int i 
= 0;  i 
< m_numRows
;  i
++ ) 
4051         rowBottom 
+= m_defaultRowHeight
; 
4052         m_rowBottoms
.Add( rowBottom 
); 
4056 void wxGrid::InitColWidths() 
4058     m_colWidths
.Empty(); 
4059     m_colRights
.Empty(); 
4061     m_colWidths
.Alloc( m_numCols 
); 
4062     m_colRights
.Alloc( m_numCols 
); 
4065     m_colWidths
.Add( m_defaultColWidth
, m_numCols 
); 
4067     for ( int i 
= 0;  i 
< m_numCols
;  i
++ ) 
4069         colRight 
+= m_defaultColWidth
; 
4070         m_colRights
.Add( colRight 
); 
4074 int wxGrid::GetColWidth(int col
) const 
4076     return m_colWidths
.IsEmpty() ? m_defaultColWidth 
: m_colWidths
[col
]; 
4079 int wxGrid::GetColLeft(int col
) const 
4081     return m_colRights
.IsEmpty() ? col 
* m_defaultColWidth
 
4082                                  : m_colRights
[col
] - m_colWidths
[col
]; 
4085 int wxGrid::GetColRight(int col
) const 
4087     return m_colRights
.IsEmpty() ? (col 
+ 1) * m_defaultColWidth
 
4091 int wxGrid::GetRowHeight(int row
) const 
4093     return m_rowHeights
.IsEmpty() ? m_defaultRowHeight 
: m_rowHeights
[row
]; 
4096 int wxGrid::GetRowTop(int row
) const 
4098     return m_rowBottoms
.IsEmpty() ? row 
* m_defaultRowHeight
 
4099                                   : m_rowBottoms
[row
] - m_rowHeights
[row
]; 
4102 int wxGrid::GetRowBottom(int row
) const 
4104     return m_rowBottoms
.IsEmpty() ? (row 
+ 1) * m_defaultRowHeight
 
4105                                   : m_rowBottoms
[row
]; 
4108 void wxGrid::CalcDimensions() 
4111     GetClientSize( &cw
, &ch 
); 
4113     if ( m_rowLabelWin
->IsShown() ) 
4114         cw 
-= m_rowLabelWidth
; 
4115     if ( m_colLabelWin
->IsShown() ) 
4116         ch 
-= m_colLabelHeight
; 
4119     int w 
= m_numCols 
> 0 ? GetColRight(m_numCols 
- 1) + m_extraWidth 
+ 1 : 0; 
4120     int h 
= m_numRows 
> 0 ? GetRowBottom(m_numRows 
- 1) + m_extraHeight 
+ 1 : 0; 
4122     // take into account editor if shown 
4123     if( IsCellEditControlShown() ) 
4126       int r 
= m_currentCellCoords
.GetRow(); 
4127       int c 
= m_currentCellCoords
.GetCol(); 
4128       int x 
= GetColLeft(c
); 
4129       int y 
= GetRowTop(r
); 
4131       // how big is the editor 
4132       wxGridCellAttr
* attr 
= GetCellAttr(r
, c
); 
4133       wxGridCellEditor
* editor 
= attr
->GetEditor(this, r
, c
); 
4134       editor
->GetControl()->GetSize(&w2
, &h2
); 
4137       if( w2 
> w 
) w 
= w2
; 
4138       if( h2 
> h 
) h 
= h2
; 
4143     // preserve (more or less) the previous position 
4145     GetViewStart( &x
, &y 
); 
4147     // maybe we don't need scrollbars at all? 
4149     // also adjust the position to be valid for the new scroll rangs 
4170     // do set scrollbar parameters 
4171     SetScrollbars( GRID_SCROLL_LINE_X
, GRID_SCROLL_LINE_Y
, 
4172                    GetScrollX(w
), GetScrollY(h
), x
, y
, 
4173                    GetBatchCount() != 0); 
4175     // if our OnSize() hadn't been called (it would if we have scrollbars), we 
4176     // still must reposition the children 
4181 void wxGrid::CalcWindowSizes() 
4184     GetClientSize( &cw
, &ch 
); 
4186     if ( m_cornerLabelWin
->IsShown() ) 
4187         m_cornerLabelWin
->SetSize( 0, 0, m_rowLabelWidth
, m_colLabelHeight 
); 
4189     if ( m_colLabelWin
->IsShown() ) 
4190         m_colLabelWin
->SetSize( m_rowLabelWidth
, 0, cw
-m_rowLabelWidth
, m_colLabelHeight
); 
4192     if ( m_rowLabelWin
->IsShown() ) 
4193         m_rowLabelWin
->SetSize( 0, m_colLabelHeight
, m_rowLabelWidth
, ch
-m_colLabelHeight
); 
4195     if ( m_gridWin
->IsShown() ) 
4196         m_gridWin
->SetSize( m_rowLabelWidth
, m_colLabelHeight
, cw
-m_rowLabelWidth
, ch
-m_colLabelHeight
); 
4200 // this is called when the grid table sends a message to say that it 
4201 // has been redimensioned 
4203 bool wxGrid::Redimension( wxGridTableMessage
& msg 
) 
4206     bool result 
= FALSE
; 
4208     // Clear the attribute cache as the attribute might refer to a different 
4209     // cell than stored in the cache after adding/removing rows/columns. 
4211     // By the same reasoning, the editor should be dismissed if columns are 
4212     // added or removed. And for consistency, it should IMHO always be 
4213     // removed, not only if the cell "underneath" it actually changes. 
4214     // For now, I intentionally do not save the editor's content as the 
4215     // cell it might want to save that stuff to might no longer exist. 
4216     HideCellEditControl(); 
4218     // if we were using the default widths/heights so far, we must change them 
4220     if ( m_colWidths
.IsEmpty() ) 
4225     if ( m_rowHeights
.IsEmpty() ) 
4231     switch ( msg
.GetId() ) 
4233         case wxGRIDTABLE_NOTIFY_ROWS_INSERTED
: 
4235             size_t pos 
= msg
.GetCommandInt(); 
4236             int numRows 
= msg
.GetCommandInt2(); 
4238             m_numRows 
+= numRows
; 
4240             if ( !m_rowHeights
.IsEmpty() ) 
4242                 m_rowHeights
.Insert( m_defaultRowHeight
, pos
, numRows 
); 
4243                 m_rowBottoms
.Insert( 0, pos
, numRows 
); 
4246                 if ( pos 
> 0 ) bottom 
= m_rowBottoms
[pos
-1]; 
4248                 for ( i 
= pos
;  i 
< m_numRows
;  i
++ ) 
4250                     bottom 
+= m_rowHeights
[i
]; 
4251                     m_rowBottoms
[i
] = bottom
; 
4254             if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
4256                 // if we have just inserted cols into an empty grid the current 
4257                 // cell will be undefined... 
4259                 SetCurrentCell( 0, 0 ); 
4263                 m_selection
->UpdateRows( pos
, numRows 
); 
4264             wxGridCellAttrProvider 
* attrProvider 
= m_table
->GetAttrProvider(); 
4266                 attrProvider
->UpdateAttrRows( pos
, numRows 
); 
4268             if ( !GetBatchCount() ) 
4271                 m_rowLabelWin
->Refresh(); 
4277         case wxGRIDTABLE_NOTIFY_ROWS_APPENDED
: 
4279             int numRows 
= msg
.GetCommandInt(); 
4280             int oldNumRows 
= m_numRows
; 
4281             m_numRows 
+= numRows
; 
4283             if ( !m_rowHeights
.IsEmpty() ) 
4285                 m_rowHeights
.Add( m_defaultRowHeight
, numRows 
); 
4286                 m_rowBottoms
.Add( 0, numRows 
); 
4289                 if ( oldNumRows 
> 0 ) bottom 
= m_rowBottoms
[oldNumRows
-1]; 
4291                 for ( i 
= oldNumRows
;  i 
< m_numRows
;  i
++ ) 
4293                     bottom 
+= m_rowHeights
[i
]; 
4294                     m_rowBottoms
[i
] = bottom
; 
4297             if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
4299                 // if we have just inserted cols into an empty grid the current 
4300                 // cell will be undefined... 
4302                 SetCurrentCell( 0, 0 ); 
4304             if ( !GetBatchCount() ) 
4307                 m_rowLabelWin
->Refresh(); 
4313         case wxGRIDTABLE_NOTIFY_ROWS_DELETED
: 
4315             size_t pos 
= msg
.GetCommandInt(); 
4316             int numRows 
= msg
.GetCommandInt2(); 
4317             m_numRows 
-= numRows
; 
4319             if ( !m_rowHeights
.IsEmpty() ) 
4321                 m_rowHeights
.RemoveAt( pos
, numRows 
); 
4322                 m_rowBottoms
.RemoveAt( pos
, numRows 
); 
4325                 for ( i 
= 0;  i 
< m_numRows
;  i
++ ) 
4327                     h 
+= m_rowHeights
[i
]; 
4328                     m_rowBottoms
[i
] = h
; 
4333                 m_currentCellCoords 
= wxGridNoCellCoords
; 
4337                 if ( m_currentCellCoords
.GetRow() >= m_numRows 
) 
4338                     m_currentCellCoords
.Set( 0, 0 ); 
4342                 m_selection
->UpdateRows( pos
, -((int)numRows
) ); 
4343             wxGridCellAttrProvider 
* attrProvider 
= m_table
->GetAttrProvider(); 
4345                 attrProvider
->UpdateAttrRows( pos
, -((int)numRows
) ); 
4346 // ifdef'd out following patch from Paul Gammans 
4348                 // No need to touch column attributes, unless we 
4349                 // removed _all_ rows, in this case, we remove 
4350                 // all column attributes. 
4351                 // I hate to do this here, but the 
4352                 // needed data is not available inside UpdateAttrRows. 
4353                 if ( !GetNumberRows() ) 
4354                     attrProvider
->UpdateAttrCols( 0, -GetNumberCols() ); 
4357             if ( !GetBatchCount() ) 
4360                 m_rowLabelWin
->Refresh(); 
4366         case wxGRIDTABLE_NOTIFY_COLS_INSERTED
: 
4368             size_t pos 
= msg
.GetCommandInt(); 
4369             int numCols 
= msg
.GetCommandInt2(); 
4370             m_numCols 
+= numCols
; 
4372             if ( !m_colWidths
.IsEmpty() ) 
4374                 m_colWidths
.Insert( m_defaultColWidth
, pos
, numCols 
); 
4375                 m_colRights
.Insert( 0, pos
, numCols 
); 
4378                 if ( pos 
> 0 ) right 
= m_colRights
[pos
-1]; 
4380                 for ( i 
= pos
;  i 
< m_numCols
;  i
++ ) 
4382                     right 
+= m_colWidths
[i
]; 
4383                     m_colRights
[i
] = right
; 
4386             if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
4388                 // if we have just inserted cols into an empty grid the current 
4389                 // cell will be undefined... 
4391                 SetCurrentCell( 0, 0 ); 
4395                 m_selection
->UpdateCols( pos
, numCols 
); 
4396             wxGridCellAttrProvider 
* attrProvider 
= m_table
->GetAttrProvider(); 
4398                 attrProvider
->UpdateAttrCols( pos
, numCols 
); 
4399             if ( !GetBatchCount() ) 
4402                 m_colLabelWin
->Refresh(); 
4409         case wxGRIDTABLE_NOTIFY_COLS_APPENDED
: 
4411             int numCols 
= msg
.GetCommandInt(); 
4412             int oldNumCols 
= m_numCols
; 
4413             m_numCols 
+= numCols
; 
4414             if ( !m_colWidths
.IsEmpty() ) 
4416                 m_colWidths
.Add( m_defaultColWidth
, numCols 
); 
4417                 m_colRights
.Add( 0, numCols 
); 
4420                 if ( oldNumCols 
> 0 ) right 
= m_colRights
[oldNumCols
-1]; 
4422                 for ( i 
= oldNumCols
;  i 
< m_numCols
;  i
++ ) 
4424                     right 
+= m_colWidths
[i
]; 
4425                     m_colRights
[i
] = right
; 
4428             if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
4430                 // if we have just inserted cols into an empty grid the current 
4431                 // cell will be undefined... 
4433                 SetCurrentCell( 0, 0 ); 
4435             if ( !GetBatchCount() ) 
4438                 m_colLabelWin
->Refresh(); 
4444         case wxGRIDTABLE_NOTIFY_COLS_DELETED
: 
4446             size_t pos 
= msg
.GetCommandInt(); 
4447             int numCols 
= msg
.GetCommandInt2(); 
4448             m_numCols 
-= numCols
; 
4450             if ( !m_colWidths
.IsEmpty() ) 
4452                 m_colWidths
.RemoveAt( pos
, numCols 
); 
4453                 m_colRights
.RemoveAt( pos
, numCols 
); 
4456                 for ( i 
= 0;  i 
< m_numCols
;  i
++ ) 
4458                     w 
+= m_colWidths
[i
]; 
4464                 m_currentCellCoords 
= wxGridNoCellCoords
; 
4468                 if ( m_currentCellCoords
.GetCol() >= m_numCols 
) 
4469                   m_currentCellCoords
.Set( 0, 0 ); 
4473                 m_selection
->UpdateCols( pos
, -((int)numCols
) ); 
4474             wxGridCellAttrProvider 
* attrProvider 
= m_table
->GetAttrProvider(); 
4476                 attrProvider
->UpdateAttrCols( pos
, -((int)numCols
) ); 
4477 // ifdef'd out following patch from Paul Gammans 
4479                 // No need to touch row attributes, unless we 
4480                 // removed _all_ columns, in this case, we remove 
4481                 // all row attributes. 
4482                 // I hate to do this here, but the 
4483                 // needed data is not available inside UpdateAttrCols. 
4484                 if ( !GetNumberCols() ) 
4485                     attrProvider
->UpdateAttrRows( 0, -GetNumberRows() ); 
4488             if ( !GetBatchCount() ) 
4491                 m_colLabelWin
->Refresh(); 
4498     if (result 
&& !GetBatchCount() ) 
4499         m_gridWin
->Refresh(); 
4504 wxArrayInt 
wxGrid::CalcRowLabelsExposed( const wxRegion
& reg 
) 
4506     wxRegionIterator 
iter( reg 
); 
4509     wxArrayInt  rowlabels
; 
4516         // TODO: remove this when we can... 
4517         // There is a bug in wxMotif that gives garbage update 
4518         // rectangles if you jump-scroll a long way by clicking the 
4519         // scrollbar with middle button.  This is a work-around 
4521 #if defined(__WXMOTIF__) 
4523         m_gridWin
->GetClientSize( &cw
, &ch 
); 
4524         if ( r
.GetTop() > ch 
) r
.SetTop( 0 ); 
4525         r
.SetBottom( wxMin( r
.GetBottom(), ch 
) ); 
4528         // logical bounds of update region 
4531         CalcUnscrolledPosition( 0, r
.GetTop(), &dummy
, &top 
); 
4532         CalcUnscrolledPosition( 0, r
.GetBottom(), &dummy
, &bottom 
); 
4534         // find the row labels within these bounds 
4537         for ( row 
= internalYToRow(top
);  row 
< m_numRows
;  row
++ ) 
4539             if ( GetRowBottom(row
) < top 
) 
4542             if ( GetRowTop(row
) > bottom 
) 
4545             rowlabels
.Add( row 
); 
4555 wxArrayInt 
wxGrid::CalcColLabelsExposed( const wxRegion
& reg 
) 
4557     wxRegionIterator 
iter( reg 
); 
4560     wxArrayInt colLabels
; 
4567         // TODO: remove this when we can... 
4568         // There is a bug in wxMotif that gives garbage update 
4569         // rectangles if you jump-scroll a long way by clicking the 
4570         // scrollbar with middle button.  This is a work-around 
4572 #if defined(__WXMOTIF__) 
4574         m_gridWin
->GetClientSize( &cw
, &ch 
); 
4575         if ( r
.GetLeft() > cw 
) r
.SetLeft( 0 ); 
4576         r
.SetRight( wxMin( r
.GetRight(), cw 
) ); 
4579         // logical bounds of update region 
4582         CalcUnscrolledPosition( r
.GetLeft(), 0, &left
, &dummy 
); 
4583         CalcUnscrolledPosition( r
.GetRight(), 0, &right
, &dummy 
); 
4585         // find the cells within these bounds 
4588         for ( col 
= internalXToCol(left
);  col 
< m_numCols
;  col
++ ) 
4590             if ( GetColRight(col
) < left 
) 
4593             if ( GetColLeft(col
) > right 
) 
4596             colLabels
.Add( col 
); 
4605 wxGridCellCoordsArray 
wxGrid::CalcCellsExposed( const wxRegion
& reg 
) 
4607     wxRegionIterator 
iter( reg 
); 
4610     wxGridCellCoordsArray  cellsExposed
; 
4612     int left
, top
, right
, bottom
; 
4617         // TODO: remove this when we can... 
4618         // There is a bug in wxMotif that gives garbage update 
4619         // rectangles if you jump-scroll a long way by clicking the 
4620         // scrollbar with middle button.  This is a work-around 
4622 #if defined(__WXMOTIF__) 
4624         m_gridWin
->GetClientSize( &cw
, &ch 
); 
4625         if ( r
.GetTop() > ch 
) r
.SetTop( 0 ); 
4626         if ( r
.GetLeft() > cw 
) r
.SetLeft( 0 ); 
4627         r
.SetRight( wxMin( r
.GetRight(), cw 
) ); 
4628         r
.SetBottom( wxMin( r
.GetBottom(), ch 
) ); 
4631         // logical bounds of update region 
4633         CalcUnscrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top 
); 
4634         CalcUnscrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom 
); 
4636         // find the cells within these bounds 
4639         for ( row 
= internalYToRow(top
);  row 
< m_numRows
;  row
++ ) 
4641             if ( GetRowBottom(row
) <= top 
) 
4644             if ( GetRowTop(row
) > bottom 
) 
4647             for ( col 
= internalXToCol(left
);  col 
< m_numCols
;  col
++ ) 
4649                 if ( GetColRight(col
) <= left 
) 
4652                 if ( GetColLeft(col
) > right 
) 
4655                 cellsExposed
.Add( wxGridCellCoords( row
, col 
) ); 
4662     return cellsExposed
; 
4666 void wxGrid::ProcessRowLabelMouseEvent( wxMouseEvent
& event 
) 
4669     wxPoint 
pos( event
.GetPosition() ); 
4670     CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y 
); 
4672     if ( event
.Dragging() ) 
4676             m_isDragging 
= TRUE
; 
4677             m_rowLabelWin
->CaptureMouse(); 
4680         if ( event
.LeftIsDown() ) 
4682             switch( m_cursorMode 
) 
4684                 case WXGRID_CURSOR_RESIZE_ROW
: 
4686                     int cw
, ch
, left
, dummy
; 
4687                     m_gridWin
->GetClientSize( &cw
, &ch 
); 
4688                     CalcUnscrolledPosition( 0, 0, &left
, &dummy 
); 
4690                     wxClientDC 
dc( m_gridWin 
); 
4693                                GetRowTop(m_dragRowOrCol
) + 
4694                                GetRowMinimalHeight(m_dragRowOrCol
) ); 
4695                     dc
.SetLogicalFunction(wxINVERT
); 
4696                     if ( m_dragLastPos 
>= 0 ) 
4698                         dc
.DrawLine( left
, m_dragLastPos
, left
+cw
, m_dragLastPos 
); 
4700                     dc
.DrawLine( left
, y
, left
+cw
, y 
); 
4705                 case WXGRID_CURSOR_SELECT_ROW
: 
4706                     if ( (row 
= YToRow( y 
)) >= 0 ) 
4710                             m_selection
->SelectRow( row
, 
4711                                                     event
.ControlDown(), 
4718                 // default label to suppress warnings about "enumeration value 
4719                 // 'xxx' not handled in switch 
4727     if ( m_isDragging 
&& (event
.Entering() || event
.Leaving()) ) 
4732         if (m_rowLabelWin
->HasCapture()) m_rowLabelWin
->ReleaseMouse(); 
4733         m_isDragging 
= FALSE
; 
4736     // ------------ Entering or leaving the window 
4738     if ( event
.Entering() || event
.Leaving() ) 
4740         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_rowLabelWin
); 
4744     // ------------ Left button pressed 
4746     else if ( event
.LeftDown() ) 
4748         // don't send a label click event for a hit on the 
4749         // edge of the row label - this is probably the user 
4750         // wanting to resize the row 
4752         if ( YToEdgeOfRow(y
) < 0 ) 
4756                  !SendEvent( wxEVT_GRID_LABEL_LEFT_CLICK
, row
, -1, event 
) ) 
4758                 if ( !event
.ShiftDown() && !event
.ControlDown() ) 
4762                     if ( event
.ShiftDown() ) 
4764                         m_selection
->SelectBlock( m_currentCellCoords
.GetRow(), 
4767                                                   GetNumberCols() - 1, 
4768                                                   event
.ControlDown(), 
4775                         m_selection
->SelectRow( row
, 
4776                                                 event
.ControlDown(), 
4783                 ChangeCursorMode(WXGRID_CURSOR_SELECT_ROW
, m_rowLabelWin
); 
4788             // starting to drag-resize a row 
4790             if ( CanDragRowSize() ) 
4791                 ChangeCursorMode(WXGRID_CURSOR_RESIZE_ROW
, m_rowLabelWin
); 
4796     // ------------ Left double click 
4798     else if (event
.LeftDClick() ) 
4800         int row 
= YToEdgeOfRow(y
); 
4805                  !SendEvent( wxEVT_GRID_LABEL_LEFT_DCLICK
, row
, -1, event 
) ) 
4807                 // no default action at the moment 
4812             // adjust row height depending on label text 
4813             AutoSizeRowLabelSize( row 
); 
4815             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
); 
4821     // ------------ Left button released 
4823     else if ( event
.LeftUp() ) 
4825         if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_ROW 
) 
4827             DoEndDragResizeRow(); 
4829             // Note: we are ending the event *after* doing 
4830             // default processing in this case 
4832             SendEvent( wxEVT_GRID_ROW_SIZE
, m_dragRowOrCol
, -1, event 
); 
4835         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_rowLabelWin
); 
4840     // ------------ Right button down 
4842     else if ( event
.RightDown() ) 
4846             !SendEvent( wxEVT_GRID_LABEL_RIGHT_CLICK
, row
, -1, event 
) ) 
4848             // no default action at the moment 
4853     // ------------ Right double click 
4855     else if ( event
.RightDClick() ) 
4859              !SendEvent( wxEVT_GRID_LABEL_RIGHT_DCLICK
, row
, -1, event 
) ) 
4861             // no default action at the moment 
4866     // ------------ No buttons down and mouse moving 
4868     else if ( event
.Moving() ) 
4870         m_dragRowOrCol 
= YToEdgeOfRow( y 
); 
4871         if ( m_dragRowOrCol 
>= 0 ) 
4873             if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
4875                 // don't capture the mouse yet 
4876                 if ( CanDragRowSize() ) 
4877                     ChangeCursorMode(WXGRID_CURSOR_RESIZE_ROW
, m_rowLabelWin
, FALSE
); 
4880         else if ( m_cursorMode 
!= WXGRID_CURSOR_SELECT_CELL 
) 
4882             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_rowLabelWin
, FALSE
); 
4888 void wxGrid::ProcessColLabelMouseEvent( wxMouseEvent
& event 
) 
4891     wxPoint 
pos( event
.GetPosition() ); 
4892     CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y 
); 
4894     if ( event
.Dragging() ) 
4898             m_isDragging 
= TRUE
; 
4899             m_colLabelWin
->CaptureMouse(); 
4902         if ( event
.LeftIsDown() ) 
4904             switch( m_cursorMode 
) 
4906                 case WXGRID_CURSOR_RESIZE_COL
: 
4908                     int cw
, ch
, dummy
, top
; 
4909                     m_gridWin
->GetClientSize( &cw
, &ch 
); 
4910                     CalcUnscrolledPosition( 0, 0, &dummy
, &top 
); 
4912                     wxClientDC 
dc( m_gridWin 
); 
4915                     x 
= wxMax( x
, GetColLeft(m_dragRowOrCol
) + 
4916                                   GetColMinimalWidth(m_dragRowOrCol
)); 
4917                     dc
.SetLogicalFunction(wxINVERT
); 
4918                     if ( m_dragLastPos 
>= 0 ) 
4920                         dc
.DrawLine( m_dragLastPos
, top
, m_dragLastPos
, top
+ch 
); 
4922                     dc
.DrawLine( x
, top
, x
, top
+ch 
); 
4927                 case WXGRID_CURSOR_SELECT_COL
: 
4928                     if ( (col 
= XToCol( x 
)) >= 0 ) 
4932                             m_selection
->SelectCol( col
, 
4933                                                     event
.ControlDown(), 
4940                 // default label to suppress warnings about "enumeration value 
4941                 // 'xxx' not handled in switch 
4949     if ( m_isDragging 
&& (event
.Entering() || event
.Leaving()) ) 
4954         if (m_colLabelWin
->HasCapture()) m_colLabelWin
->ReleaseMouse(); 
4955         m_isDragging 
= FALSE
; 
4958     // ------------ Entering or leaving the window 
4960     if ( event
.Entering() || event
.Leaving() ) 
4962         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
); 
4966     // ------------ Left button pressed 
4968     else if ( event
.LeftDown() ) 
4970         // don't send a label click event for a hit on the 
4971         // edge of the col label - this is probably the user 
4972         // wanting to resize the col 
4974         if ( XToEdgeOfCol(x
) < 0 ) 
4978                  !SendEvent( wxEVT_GRID_LABEL_LEFT_CLICK
, -1, col
, event 
) ) 
4980                 if ( !event
.ShiftDown() && !event
.ControlDown() ) 
4984                     if ( event
.ShiftDown() ) 
4986                         m_selection
->SelectBlock( 0, 
4987                                                   m_currentCellCoords
.GetCol(), 
4988                                                   GetNumberRows() - 1, col
, 
4989                                                   event
.ControlDown(), 
4996                         m_selection
->SelectCol( col
, 
4997                                                 event
.ControlDown(), 
5004                 ChangeCursorMode(WXGRID_CURSOR_SELECT_COL
, m_colLabelWin
); 
5009             // starting to drag-resize a col 
5011             if ( CanDragColSize() ) 
5012                 ChangeCursorMode(WXGRID_CURSOR_RESIZE_COL
, m_colLabelWin
); 
5017     // ------------ Left double click 
5019     if ( event
.LeftDClick() ) 
5021         int col 
= XToEdgeOfCol(x
); 
5026                  ! SendEvent( wxEVT_GRID_LABEL_LEFT_DCLICK
, -1, col
, event 
) ) 
5028                 // no default action at the moment 
5033             // adjust column width depending on label text 
5034             AutoSizeColLabelSize( col 
); 
5036             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
); 
5042     // ------------ Left button released 
5044     else if ( event
.LeftUp() ) 
5046         if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_COL 
) 
5048             DoEndDragResizeCol(); 
5050             // Note: we are ending the event *after* doing 
5051             // default processing in this case 
5053             SendEvent( wxEVT_GRID_COL_SIZE
, -1, m_dragRowOrCol
, event 
); 
5056         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
); 
5061     // ------------ Right button down 
5063     else if ( event
.RightDown() ) 
5067              !SendEvent( wxEVT_GRID_LABEL_RIGHT_CLICK
, -1, col
, event 
) ) 
5069             // no default action at the moment 
5074     // ------------ Right double click 
5076     else if ( event
.RightDClick() ) 
5080              !SendEvent( wxEVT_GRID_LABEL_RIGHT_DCLICK
, -1, col
, event 
) ) 
5082             // no default action at the moment 
5087     // ------------ No buttons down and mouse moving 
5089     else if ( event
.Moving() ) 
5091         m_dragRowOrCol 
= XToEdgeOfCol( x 
); 
5092         if ( m_dragRowOrCol 
>= 0 ) 
5094             if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
5096                 // don't capture the cursor yet 
5097                 if ( CanDragColSize() ) 
5098                     ChangeCursorMode(WXGRID_CURSOR_RESIZE_COL
, m_colLabelWin
, FALSE
); 
5101         else if ( m_cursorMode 
!= WXGRID_CURSOR_SELECT_CELL 
) 
5103             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
, m_colLabelWin
, FALSE
); 
5109 void wxGrid::ProcessCornerLabelMouseEvent( wxMouseEvent
& event 
) 
5111     if ( event
.LeftDown() ) 
5113         // indicate corner label by having both row and 
5116         if ( !SendEvent( wxEVT_GRID_LABEL_LEFT_CLICK
, -1, -1, event 
) ) 
5122     else if ( event
.LeftDClick() ) 
5124         SendEvent( wxEVT_GRID_LABEL_LEFT_DCLICK
, -1, -1, event 
); 
5127     else if ( event
.RightDown() ) 
5129         if ( !SendEvent( wxEVT_GRID_LABEL_RIGHT_CLICK
, -1, -1, event 
) ) 
5131             // no default action at the moment 
5135     else if ( event
.RightDClick() ) 
5137         if ( !SendEvent( wxEVT_GRID_LABEL_RIGHT_DCLICK
, -1, -1, event 
) ) 
5139             // no default action at the moment 
5144 void wxGrid::ChangeCursorMode(CursorMode mode
, 
5149     static const wxChar 
*cursorModes
[] = 
5158     wxLogTrace(_T("grid"), 
5159                _T("wxGrid cursor mode (mouse capture for %s): %s -> %s"), 
5160                win 
== m_colLabelWin 
? _T("colLabelWin") 
5161                                     : win 
? _T("rowLabelWin") 
5163                cursorModes
[m_cursorMode
], cursorModes
[mode
]); 
5164 #endif // __WXDEBUG__ 
5166     if ( mode 
== m_cursorMode 
&& 
5167          win 
== m_winCapture 
&& 
5168          captureMouse 
== (m_winCapture 
!= NULL
)) 
5173         // by default use the grid itself 
5179         if (m_winCapture
->HasCapture()) m_winCapture
->ReleaseMouse(); 
5180         m_winCapture 
= (wxWindow 
*)NULL
; 
5183     m_cursorMode 
= mode
; 
5185     switch ( m_cursorMode 
) 
5187         case WXGRID_CURSOR_RESIZE_ROW
: 
5188             win
->SetCursor( m_rowResizeCursor 
); 
5191         case WXGRID_CURSOR_RESIZE_COL
: 
5192             win
->SetCursor( m_colResizeCursor 
); 
5196             win
->SetCursor( *wxSTANDARD_CURSOR 
); 
5199     // we need to capture mouse when resizing 
5200     bool resize 
= m_cursorMode 
== WXGRID_CURSOR_RESIZE_ROW 
|| 
5201                   m_cursorMode 
== WXGRID_CURSOR_RESIZE_COL
; 
5203     if ( captureMouse 
&& resize 
) 
5205         win
->CaptureMouse(); 
5210 void wxGrid::ProcessGridCellMouseEvent( wxMouseEvent
& event 
) 
5213     wxPoint 
pos( event
.GetPosition() ); 
5214     CalcUnscrolledPosition( pos
.x
, pos
.y
, &x
, &y 
); 
5216     wxGridCellCoords coords
; 
5217     XYToCell( x
, y
, coords 
); 
5219     int cell_rows
, cell_cols
; 
5220     GetCellSize( coords
.GetRow(), coords
.GetCol(), &cell_rows
, &cell_cols 
); 
5221     if ((cell_rows 
< 0) || (cell_cols 
< 0)) 
5223         coords
.SetRow(coords
.GetRow() + cell_rows
); 
5224         coords
.SetCol(coords
.GetCol() + cell_cols
); 
5227     if ( event
.Dragging() ) 
5229         //wxLogDebug("pos(%d, %d) coords(%d, %d)", pos.x, pos.y, coords.GetRow(), coords.GetCol()); 
5231         // Don't start doing anything until the mouse has been drug at 
5232         // least 3 pixels in any direction... 
5235             if (m_startDragPos 
== wxDefaultPosition
) 
5237                 m_startDragPos 
= pos
; 
5240             if (abs(m_startDragPos
.x 
- pos
.x
) < 4 && abs(m_startDragPos
.y 
- pos
.y
) < 4) 
5244         m_isDragging 
= TRUE
; 
5245         if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
5247             // Hide the edit control, so it 
5248             // won't interfer with drag-shrinking. 
5249             if ( IsCellEditControlShown() ) 
5251                 HideCellEditControl(); 
5252                 SaveEditControlValue(); 
5255             // Have we captured the mouse yet? 
5258                 m_winCapture 
= m_gridWin
; 
5259                 m_winCapture
->CaptureMouse(); 
5262             if ( coords 
!= wxGridNoCellCoords 
) 
5264                 if ( event
.ControlDown() ) 
5266                     if ( m_selectingKeyboard 
== wxGridNoCellCoords
) 
5267                         m_selectingKeyboard 
= coords
; 
5268                     HighlightBlock ( m_selectingKeyboard
, coords 
); 
5272                     if ( !IsSelection() ) 
5274                         HighlightBlock( coords
, coords 
); 
5278                         HighlightBlock( m_currentCellCoords
, coords 
); 
5282                 if (! IsVisible(coords
)) 
5284                     MakeCellVisible(coords
); 
5285                     // TODO: need to introduce a delay or something here.  The 
5286                     // scrolling is way to fast, at least on MSW - also on GTK. 
5290         else if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_ROW 
) 
5292             int cw
, ch
, left
, dummy
; 
5293             m_gridWin
->GetClientSize( &cw
, &ch 
); 
5294             CalcUnscrolledPosition( 0, 0, &left
, &dummy 
); 
5296             wxClientDC 
dc( m_gridWin 
); 
5298             y 
= wxMax( y
, GetRowTop(m_dragRowOrCol
) + 
5299                           GetRowMinimalHeight(m_dragRowOrCol
) ); 
5300             dc
.SetLogicalFunction(wxINVERT
); 
5301             if ( m_dragLastPos 
>= 0 ) 
5303                 dc
.DrawLine( left
, m_dragLastPos
, left
+cw
, m_dragLastPos 
); 
5305             dc
.DrawLine( left
, y
, left
+cw
, y 
); 
5308         else if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_COL 
) 
5310             int cw
, ch
, dummy
, top
; 
5311             m_gridWin
->GetClientSize( &cw
, &ch 
); 
5312             CalcUnscrolledPosition( 0, 0, &dummy
, &top 
); 
5314             wxClientDC 
dc( m_gridWin 
); 
5316             x 
= wxMax( x
, GetColLeft(m_dragRowOrCol
) + 
5317                           GetColMinimalWidth(m_dragRowOrCol
) ); 
5318             dc
.SetLogicalFunction(wxINVERT
); 
5319             if ( m_dragLastPos 
>= 0 ) 
5321                 dc
.DrawLine( m_dragLastPos
, top
, m_dragLastPos
, top
+ch 
); 
5323             dc
.DrawLine( x
, top
, x
, top
+ch 
); 
5330     m_isDragging 
= FALSE
; 
5331     m_startDragPos 
= wxDefaultPosition
; 
5333     // VZ: if we do this, the mode is reset to WXGRID_CURSOR_SELECT_CELL 
5334     //     immediately after it becomes WXGRID_CURSOR_RESIZE_ROW/COL under 
5337     if ( event
.Entering() || event
.Leaving() ) 
5339         ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5340         m_gridWin
->SetCursor( *wxSTANDARD_CURSOR 
); 
5345     // ------------ Left button pressed 
5347     if ( event
.LeftDown() && coords 
!= wxGridNoCellCoords 
) 
5349         if ( !SendEvent( wxEVT_GRID_CELL_LEFT_CLICK
, 
5354             if ( !event
.ControlDown() ) 
5356             if ( event
.ShiftDown() ) 
5360                     m_selection
->SelectBlock( m_currentCellCoords
.GetRow(), 
5361                                               m_currentCellCoords
.GetCol(), 
5364                                               event
.ControlDown(), 
5370             else if ( XToEdgeOfCol(x
) < 0  && 
5371                       YToEdgeOfRow(y
) < 0 ) 
5373                 DisableCellEditControl(); 
5374                 MakeCellVisible( coords 
); 
5376                 if ( event
.ControlDown() ) 
5380                         m_selection
->ToggleCellSelection( coords
.GetRow(), 
5382                                                           event
.ControlDown(), 
5387                     m_selectingTopLeft 
= wxGridNoCellCoords
; 
5388                     m_selectingBottomRight 
= wxGridNoCellCoords
; 
5389                     m_selectingKeyboard 
= coords
; 
5393                     m_waitForSlowClick 
= m_currentCellCoords 
== coords 
&& coords 
!= wxGridNoCellCoords
; 
5394                     SetCurrentCell( coords 
); 
5397                         if ( m_selection
->GetSelectionMode() != 
5398                                 wxGrid::wxGridSelectCells 
) 
5400                             HighlightBlock( coords
, coords 
); 
5409     // ------------ Left double click 
5411     else if ( event
.LeftDClick() && coords 
!= wxGridNoCellCoords 
) 
5413         DisableCellEditControl(); 
5415         if ( XToEdgeOfCol(x
) < 0  &&  YToEdgeOfRow(y
) < 0 ) 
5417             SendEvent( wxEVT_GRID_CELL_LEFT_DCLICK
, 
5425     // ------------ Left button released 
5427     else if ( event
.LeftUp() ) 
5429         if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
5433                 if (m_winCapture
->HasCapture()) m_winCapture
->ReleaseMouse(); 
5434                 m_winCapture 
= NULL
; 
5437             if ( coords 
== m_currentCellCoords 
&& m_waitForSlowClick 
&& CanEnableCellControl()) 
5440                 EnableCellEditControl(); 
5442                 wxGridCellAttr
* attr 
= GetCellAttr(coords
); 
5443                 wxGridCellEditor 
*editor 
= attr
->GetEditor(this, coords
.GetRow(), coords
.GetCol()); 
5444                 editor
->StartingClick(); 
5448                 m_waitForSlowClick 
= FALSE
; 
5450             else if ( m_selectingTopLeft 
!= wxGridNoCellCoords 
&& 
5451                  m_selectingBottomRight 
!= wxGridNoCellCoords 
) 
5455                     m_selection
->SelectBlock( m_selectingTopLeft
.GetRow(), 
5456                                               m_selectingTopLeft
.GetCol(), 
5457                                               m_selectingBottomRight
.GetRow(), 
5458                                               m_selectingBottomRight
.GetCol(), 
5459                                               event
.ControlDown(), 
5465                 m_selectingTopLeft 
= wxGridNoCellCoords
; 
5466                 m_selectingBottomRight 
= wxGridNoCellCoords
; 
5468                 // Show the edit control, if it has been hidden for 
5470                 ShowCellEditControl(); 
5473         else if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_ROW 
) 
5475             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5476             DoEndDragResizeRow(); 
5478             // Note: we are ending the event *after* doing 
5479             // default processing in this case 
5481             SendEvent( wxEVT_GRID_ROW_SIZE
, m_dragRowOrCol
, -1, event 
); 
5483         else if ( m_cursorMode 
== WXGRID_CURSOR_RESIZE_COL 
) 
5485             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5486             DoEndDragResizeCol(); 
5488             // Note: we are ending the event *after* doing 
5489             // default processing in this case 
5491             SendEvent( wxEVT_GRID_COL_SIZE
, -1, m_dragRowOrCol
, event 
); 
5498     // ------------ Right button down 
5500     else if ( event
.RightDown() && coords 
!= wxGridNoCellCoords 
) 
5502         DisableCellEditControl(); 
5503         if ( !SendEvent( wxEVT_GRID_CELL_RIGHT_CLICK
, 
5508             // no default action at the moment 
5513     // ------------ Right double click 
5515     else if ( event
.RightDClick() && coords 
!= wxGridNoCellCoords 
) 
5517         DisableCellEditControl(); 
5518         if ( !SendEvent( wxEVT_GRID_CELL_RIGHT_DCLICK
, 
5523             // no default action at the moment 
5527     // ------------ Moving and no button action 
5529     else if ( event
.Moving() && !event
.IsButton() ) 
5531         if( coords
.GetRow() < 0 || coords
.GetCol() < 0 ) 
5533             // out of grid cell area 
5534             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5538         int dragRow 
= YToEdgeOfRow( y 
); 
5539         int dragCol 
= XToEdgeOfCol( x 
); 
5541         // Dragging on the corner of a cell to resize in both 
5542         // directions is not implemented yet... 
5544         if ( dragRow 
>= 0  &&  dragCol 
>= 0 ) 
5546             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5552             m_dragRowOrCol 
= dragRow
; 
5554             if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
5556                 if ( CanDragRowSize() && CanDragGridSize() ) 
5557                     ChangeCursorMode(WXGRID_CURSOR_RESIZE_ROW
); 
5562                 m_dragRowOrCol 
= dragCol
; 
5570             m_dragRowOrCol 
= dragCol
; 
5572             if ( m_cursorMode 
== WXGRID_CURSOR_SELECT_CELL 
) 
5574                 if ( CanDragColSize() && CanDragGridSize() ) 
5575                     ChangeCursorMode(WXGRID_CURSOR_RESIZE_COL
); 
5581         // Neither on a row or col edge 
5583         if ( m_cursorMode 
!= WXGRID_CURSOR_SELECT_CELL 
) 
5585             ChangeCursorMode(WXGRID_CURSOR_SELECT_CELL
); 
5591 void wxGrid::DoEndDragResizeRow() 
5593     if ( m_dragLastPos 
>= 0 ) 
5595         // erase the last line and resize the row 
5597         int cw
, ch
, left
, dummy
; 
5598         m_gridWin
->GetClientSize( &cw
, &ch 
); 
5599         CalcUnscrolledPosition( 0, 0, &left
, &dummy 
); 
5601         wxClientDC 
dc( m_gridWin 
); 
5603         dc
.SetLogicalFunction( wxINVERT 
); 
5604         dc
.DrawLine( left
, m_dragLastPos
, left
+cw
, m_dragLastPos 
); 
5605         HideCellEditControl(); 
5606         SaveEditControlValue(); 
5608         int rowTop 
= GetRowTop(m_dragRowOrCol
); 
5609         SetRowSize( m_dragRowOrCol
, 
5610                     wxMax( m_dragLastPos 
- rowTop
, m_minAcceptableRowHeight 
) ); 
5612         if ( !GetBatchCount() ) 
5614             // Only needed to get the correct rect.y: 
5615             wxRect 
rect ( CellToRect( m_dragRowOrCol
, 0 ) ); 
5617             CalcScrolledPosition(0, rect
.y
, &dummy
, &rect
.y
); 
5618             rect
.width 
= m_rowLabelWidth
; 
5619             rect
.height 
= ch 
- rect
.y
; 
5620             m_rowLabelWin
->Refresh( TRUE
, &rect 
); 
5622             // if there is a multicell block, paint all of it 
5625                 int i
, cell_rows
, cell_cols
, subtract_rows 
= 0; 
5626                 int leftCol 
= XToCol(left
); 
5627                 int rightCol 
= internalXToCol(left
+cw
); 
5630                     for (i
=leftCol
; i
<rightCol
; i
++) 
5632                         GetCellSize(m_dragRowOrCol
, i
, &cell_rows
, &cell_cols
); 
5633                         if (cell_rows 
< subtract_rows
) 
5634                             subtract_rows 
= cell_rows
; 
5636                     rect
.y 
= GetRowTop(m_dragRowOrCol 
+ subtract_rows
); 
5637                     CalcScrolledPosition(0, rect
.y
, &dummy
, &rect
.y
); 
5638                     rect
.height 
= ch 
- rect
.y
; 
5641             m_gridWin
->Refresh( FALSE
, &rect 
); 
5644         ShowCellEditControl(); 
5649 void wxGrid::DoEndDragResizeCol() 
5651     if ( m_dragLastPos 
>= 0 ) 
5653         // erase the last line and resize the col 
5655         int cw
, ch
, dummy
, top
; 
5656         m_gridWin
->GetClientSize( &cw
, &ch 
); 
5657         CalcUnscrolledPosition( 0, 0, &dummy
, &top 
); 
5659         wxClientDC 
dc( m_gridWin 
); 
5661         dc
.SetLogicalFunction( wxINVERT 
); 
5662         dc
.DrawLine( m_dragLastPos
, top
, m_dragLastPos
, top
+ch 
); 
5663         HideCellEditControl(); 
5664         SaveEditControlValue(); 
5666         int colLeft 
= GetColLeft(m_dragRowOrCol
); 
5667         SetColSize( m_dragRowOrCol
, 
5668                     wxMax( m_dragLastPos 
- colLeft
, 
5669                            GetColMinimalWidth(m_dragRowOrCol
) ) ); 
5671         if ( !GetBatchCount() ) 
5673             // Only needed to get the correct rect.x: 
5674             wxRect 
rect ( CellToRect( 0, m_dragRowOrCol 
) ); 
5676             CalcScrolledPosition(rect
.x
, 0, &rect
.x
, &dummy
); 
5677             rect
.width 
= cw 
- rect
.x
; 
5678             rect
.height 
= m_colLabelHeight
; 
5679             m_colLabelWin
->Refresh( TRUE
, &rect 
); 
5681             // if there is a multicell block, paint all of it 
5684                 int i
, cell_rows
, cell_cols
, subtract_cols 
= 0; 
5685                 int topRow 
= YToRow(top
); 
5686                 int bottomRow 
= internalYToRow(top
+cw
); 
5689                     for (i
=topRow
; i
<bottomRow
; i
++) 
5691                         GetCellSize(i
, m_dragRowOrCol
, &cell_rows
, &cell_cols
); 
5692                         if (cell_cols 
< subtract_cols
) 
5693                             subtract_cols 
= cell_cols
; 
5695                     rect
.x 
= GetColLeft(m_dragRowOrCol 
+ subtract_cols
); 
5696                     CalcScrolledPosition(rect
.x
, 0, &rect
.x
, &dummy
); 
5697                     rect
.width 
= cw 
- rect
.x
; 
5700             m_gridWin
->Refresh( FALSE
, &rect 
); 
5703         ShowCellEditControl(); 
5710 // ------ interaction with data model 
5712 bool wxGrid::ProcessTableMessage( wxGridTableMessage
& msg 
) 
5714     switch ( msg
.GetId() ) 
5716         case wxGRIDTABLE_REQUEST_VIEW_GET_VALUES
: 
5717             return GetModelValues(); 
5719         case wxGRIDTABLE_REQUEST_VIEW_SEND_VALUES
: 
5720             return SetModelValues(); 
5722         case wxGRIDTABLE_NOTIFY_ROWS_INSERTED
: 
5723         case wxGRIDTABLE_NOTIFY_ROWS_APPENDED
: 
5724         case wxGRIDTABLE_NOTIFY_ROWS_DELETED
: 
5725         case wxGRIDTABLE_NOTIFY_COLS_INSERTED
: 
5726         case wxGRIDTABLE_NOTIFY_COLS_APPENDED
: 
5727         case wxGRIDTABLE_NOTIFY_COLS_DELETED
: 
5728             return Redimension( msg 
); 
5737 // The behaviour of this function depends on the grid table class 
5738 // Clear() function.  For the default wxGridStringTable class the 
5739 // behavious is to replace all cell contents with wxEmptyString but 
5740 // not to change the number of rows or cols. 
5742 void wxGrid::ClearGrid() 
5746         if (IsCellEditControlEnabled()) 
5747             DisableCellEditControl(); 
5750         if ( !GetBatchCount() ) m_gridWin
->Refresh(); 
5755 bool wxGrid::InsertRows( int pos
, int numRows
, bool WXUNUSED(updateLabels
) ) 
5757     // TODO: something with updateLabels flag 
5761         wxFAIL_MSG( wxT("Called wxGrid::InsertRows() before calling CreateGrid()") ); 
5767         if (IsCellEditControlEnabled()) 
5768             DisableCellEditControl(); 
5770         return m_table
->InsertRows( pos
, numRows 
); 
5772         // the table will have sent the results of the insert row 
5773         // operation to this view object as a grid table message 
5779 bool wxGrid::AppendRows( int numRows
, bool WXUNUSED(updateLabels
) ) 
5781     // TODO: something with updateLabels flag 
5785         wxFAIL_MSG( wxT("Called wxGrid::AppendRows() before calling CreateGrid()") ); 
5789     return ( m_table 
&& m_table
->AppendRows( numRows 
) ); 
5790     // the table will have sent the results of the append row 
5791     // operation to this view object as a grid table message 
5795 bool wxGrid::DeleteRows( int pos
, int numRows
, bool WXUNUSED(updateLabels
) ) 
5797     // TODO: something with updateLabels flag 
5801         wxFAIL_MSG( wxT("Called wxGrid::DeleteRows() before calling CreateGrid()") ); 
5807         if (IsCellEditControlEnabled()) 
5808             DisableCellEditControl(); 
5810         return (m_table
->DeleteRows( pos
, numRows 
)); 
5811         // the table will have sent the results of the delete row 
5812         // operation to this view object as a grid table message 
5818 bool wxGrid::InsertCols( int pos
, int numCols
, bool WXUNUSED(updateLabels
) ) 
5820     // TODO: something with updateLabels flag 
5824         wxFAIL_MSG( wxT("Called wxGrid::InsertCols() before calling CreateGrid()") ); 
5830         if (IsCellEditControlEnabled()) 
5831             DisableCellEditControl(); 
5833         return m_table
->InsertCols( pos
, numCols 
); 
5834         // the table will have sent the results of the insert col 
5835         // operation to this view object as a grid table message 
5841 bool wxGrid::AppendCols( int numCols
, bool WXUNUSED(updateLabels
) ) 
5843     // TODO: something with updateLabels flag 
5847         wxFAIL_MSG( wxT("Called wxGrid::AppendCols() before calling CreateGrid()") ); 
5851     return ( m_table 
&& m_table
->AppendCols( numCols 
) ); 
5852     // the table will have sent the results of the append col 
5853     // operation to this view object as a grid table message 
5857 bool wxGrid::DeleteCols( int pos
, int numCols
, bool WXUNUSED(updateLabels
) ) 
5859     // TODO: something with updateLabels flag 
5863         wxFAIL_MSG( wxT("Called wxGrid::DeleteCols() before calling CreateGrid()") ); 
5869         if (IsCellEditControlEnabled()) 
5870             DisableCellEditControl(); 
5872         return ( m_table
->DeleteCols( pos
, numCols 
) ); 
5873         // the table will have sent the results of the delete col 
5874         // operation to this view object as a grid table message 
5882 // ----- event handlers 
5885 // Generate a grid event based on a mouse event and 
5886 // return the result of ProcessEvent() 
5888 int wxGrid::SendEvent( const wxEventType type
, 
5890                         wxMouseEvent
& mouseEv 
) 
5895    if ( type 
== wxEVT_GRID_ROW_SIZE 
|| type 
== wxEVT_GRID_COL_SIZE 
) 
5897        int rowOrCol 
= (row 
== -1 ? col 
: row
); 
5899        wxGridSizeEvent 
gridEvt( GetId(), 
5903                mouseEv
.GetX() + GetRowLabelSize(), 
5904                mouseEv
.GetY() + GetColLabelSize(), 
5905                mouseEv
.ControlDown(), 
5906                mouseEv
.ShiftDown(), 
5908                mouseEv
.MetaDown() ); 
5910        claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
5911        vetoed 
= !gridEvt
.IsAllowed(); 
5913    else if ( type 
== wxEVT_GRID_RANGE_SELECT 
) 
5915        // Right now, it should _never_ end up here! 
5916        wxGridRangeSelectEvent 
gridEvt( GetId(), 
5920                m_selectingBottomRight
, 
5922                mouseEv
.ControlDown(), 
5923                mouseEv
.ShiftDown(), 
5925                mouseEv
.MetaDown() ); 
5927        claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
5928        vetoed 
= !gridEvt
.IsAllowed(); 
5932        wxGridEvent 
gridEvt( GetId(), 
5936                mouseEv
.GetX() + GetRowLabelSize(), 
5937                mouseEv
.GetY() + GetColLabelSize(), 
5939                mouseEv
.ControlDown(), 
5940                mouseEv
.ShiftDown(), 
5942                mouseEv
.MetaDown() ); 
5943        claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
5944        vetoed 
= !gridEvt
.IsAllowed(); 
5947    // A Veto'd event may not be `claimed' so test this first 
5948    if (vetoed
) return -1; 
5949    return claimed 
? 1 : 0; 
5953 // Generate a grid event of specified type and return the result 
5954 // of ProcessEvent(). 
5956 int wxGrid::SendEvent( const wxEventType type
, 
5962     if ( type 
== wxEVT_GRID_ROW_SIZE 
|| type 
== wxEVT_GRID_COL_SIZE 
) 
5964         int rowOrCol 
= (row 
== -1 ? col 
: row
); 
5966         wxGridSizeEvent 
gridEvt( GetId(), 
5971         claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
5972         vetoed  
= !gridEvt
.IsAllowed(); 
5976         wxGridEvent 
gridEvt( GetId(), 
5981         claimed 
= GetEventHandler()->ProcessEvent(gridEvt
); 
5982         vetoed  
= !gridEvt
.IsAllowed(); 
5985     // A Veto'd event may not be `claimed' so test this first 
5986     if (vetoed
) return -1; 
5987     return claimed 
? 1 : 0; 
5991 void wxGrid::OnPaint( wxPaintEvent
& WXUNUSED(event
) ) 
5993     wxPaintDC 
dc(this);  // needed to prevent zillions of paint events on MSW 
5996 void wxGrid::Refresh(bool eraseb
, const wxRect
* rect
) 
5998     // Don't do anything if between Begin/EndBatch... 
5999     // EndBatch() will do all this on the last nested one anyway. 
6000     if (! GetBatchCount()) 
6002         // Refresh to get correct scrolled position: 
6003         wxScrolledWindow::Refresh(eraseb
,rect
); 
6007             int rect_x
, rect_y
, rectWidth
, rectHeight
; 
6008             int width_label
, width_cell
, height_label
, height_cell
; 
6011             //Copy rectangle can get scroll offsets.. 
6012             rect_x 
= rect
->GetX(); 
6013             rect_y 
= rect
->GetY(); 
6014             rectWidth 
= rect
->GetWidth(); 
6015             rectHeight 
= rect
->GetHeight(); 
6017             width_label 
= m_rowLabelWidth 
- rect_x
; 
6018             if (width_label 
> rectWidth
) width_label 
= rectWidth
; 
6020             height_label 
= m_colLabelHeight 
- rect_y
; 
6021             if (height_label 
> rectHeight
) height_label 
= rectHeight
; 
6023             if (rect_x 
> m_rowLabelWidth
) 
6025                 x 
= rect_x 
- m_rowLabelWidth
; 
6026                 width_cell 
= rectWidth
; 
6031                 width_cell 
= rectWidth 
- (m_rowLabelWidth 
- rect_x
); 
6034             if (rect_y 
> m_colLabelHeight
) 
6036                 y 
= rect_y 
- m_colLabelHeight
; 
6037                 height_cell 
= rectHeight
; 
6042                 height_cell 
= rectHeight 
- (m_colLabelHeight 
- rect_y
); 
6045             // Paint corner label part intersecting rect. 
6046             if ( width_label 
> 0 && height_label 
> 0 ) 
6048                 wxRect 
anotherrect(rect_x
, rect_y
, width_label
, height_label
); 
6049                 m_cornerLabelWin
->Refresh(eraseb
, &anotherrect
); 
6052             // Paint col labels part intersecting rect. 
6053             if ( width_cell 
> 0 && height_label 
> 0 ) 
6055                 wxRect 
anotherrect(x
, rect_y
, width_cell
, height_label
); 
6056                 m_colLabelWin
->Refresh(eraseb
, &anotherrect
); 
6059             // Paint row labels part intersecting rect. 
6060             if ( width_label 
> 0 && height_cell 
> 0 ) 
6062                 wxRect 
anotherrect(rect_x
, y
, width_label
, height_cell
); 
6063                 m_rowLabelWin
->Refresh(eraseb
, &anotherrect
); 
6066             // Paint cell area part intersecting rect. 
6067             if ( width_cell 
> 0 && height_cell 
> 0 ) 
6069                 wxRect 
anotherrect(x
, y
, width_cell
, height_cell
); 
6070                 m_gridWin
->Refresh(eraseb
, &anotherrect
); 
6075             m_cornerLabelWin
->Refresh(eraseb
, NULL
); 
6076             m_colLabelWin
->Refresh(eraseb
, NULL
); 
6077             m_rowLabelWin
->Refresh(eraseb
, NULL
); 
6078             m_gridWin
->Refresh(eraseb
, NULL
); 
6083 void wxGrid::OnSize( wxSizeEvent
& event 
) 
6085     // position the child windows 
6088     // don't call CalcDimensions() from here, the base class handles the size 
6094 void wxGrid::OnKeyDown( wxKeyEvent
& event 
) 
6096     if ( m_inOnKeyDown 
) 
6098         // shouldn't be here - we are going round in circles... 
6100         wxFAIL_MSG( wxT("wxGrid::OnKeyDown called while already active") ); 
6103     m_inOnKeyDown 
= TRUE
; 
6105     // propagate the event up and see if it gets processed 
6107     wxWindow 
*parent 
= GetParent(); 
6108     wxKeyEvent 
keyEvt( event 
); 
6109     keyEvt
.SetEventObject( parent 
); 
6111     if ( !parent
->GetEventHandler()->ProcessEvent( keyEvt 
) ) 
6114         // try local handlers 
6116         switch ( event
.GetKeyCode() ) 
6119                 if ( event
.ControlDown() ) 
6121                     MoveCursorUpBlock( event
.ShiftDown() ); 
6125                     MoveCursorUp( event
.ShiftDown() ); 
6130                 if ( event
.ControlDown() ) 
6132                     MoveCursorDownBlock( event
.ShiftDown() ); 
6136                     MoveCursorDown( event
.ShiftDown() ); 
6141                 if ( event
.ControlDown() ) 
6143                     MoveCursorLeftBlock( event
.ShiftDown() ); 
6147                     MoveCursorLeft( event
.ShiftDown() ); 
6152                 if ( event
.ControlDown() ) 
6154                     MoveCursorRightBlock( event
.ShiftDown() ); 
6158                     MoveCursorRight( event
.ShiftDown() ); 
6163             case WXK_NUMPAD_ENTER
: 
6164                 if ( event
.ControlDown() ) 
6166                     event
.Skip();  // to let the edit control have the return 
6170                     if ( GetGridCursorRow() < GetNumberRows()-1 ) 
6172                         MoveCursorDown( event
.ShiftDown() ); 
6176                         // at the bottom of a column 
6177                         HideCellEditControl(); 
6178                         SaveEditControlValue(); 
6188                 if (event
.ShiftDown()) 
6190                     if ( GetGridCursorCol() > 0 ) 
6192                         MoveCursorLeft( FALSE 
); 
6197                         HideCellEditControl(); 
6198                         SaveEditControlValue(); 
6203                     if ( GetGridCursorCol() < GetNumberCols()-1 ) 
6205                         MoveCursorRight( FALSE 
); 
6210                         HideCellEditControl(); 
6211                         SaveEditControlValue(); 
6217                 if ( event
.ControlDown() ) 
6219                     MakeCellVisible( 0, 0 ); 
6220                     SetCurrentCell( 0, 0 ); 
6229                 if ( event
.ControlDown() ) 
6231                     MakeCellVisible( m_numRows
-1, m_numCols
-1 ); 
6232                     SetCurrentCell( m_numRows
-1, m_numCols
-1 ); 
6249                 if ( event
.ControlDown() ) 
6253                         m_selection
->ToggleCellSelection( m_currentCellCoords
.GetRow(), 
6254                                                           m_currentCellCoords
.GetCol(), 
6255                                                           event
.ControlDown(), 
6262                 if ( !IsEditable() ) 
6264                     MoveCursorRight( FALSE 
); 
6267                 // Otherwise fall through to default 
6270                 // is it possible to edit the current cell at all? 
6271                 if ( !IsCellEditControlEnabled() && CanEnableCellControl() ) 
6273                     // yes, now check whether the cells editor accepts the key 
6274                     int row 
= m_currentCellCoords
.GetRow(); 
6275                     int col 
= m_currentCellCoords
.GetCol(); 
6276                     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
6277                     wxGridCellEditor 
*editor 
= attr
->GetEditor(this, row
, col
); 
6279                     // <F2> is special and will always start editing, for 
6280                     // other keys - ask the editor itself 
6281                     if ( (event
.GetKeyCode() == WXK_F2 
&& !event
.HasModifiers()) 
6282                          || editor
->IsAcceptedKey(event
) ) 
6284                         // ensure cell is visble 
6285                         MakeCellVisible(row
, col
); 
6286                         EnableCellEditControl(); 
6288                         // a problem can arise if the cell is not completely 
6289                         // visible (even after calling MakeCellVisible the 
6290                         // control is not created and calling StartingKey will 
6292                         if( editor
->IsCreated() && m_cellEditCtrlEnabled 
) editor
->StartingKey(event
); 
6304                     // let others process char events with modifiers or all 
6305                     // char events for readonly cells 
6312     m_inOnKeyDown 
= FALSE
; 
6315 void wxGrid::OnKeyUp( wxKeyEvent
& event 
) 
6317     // try local handlers 
6319     if ( event
.GetKeyCode() == WXK_SHIFT 
) 
6321         if ( m_selectingTopLeft 
!= wxGridNoCellCoords 
&& 
6322              m_selectingBottomRight 
!= wxGridNoCellCoords 
) 
6326                 m_selection
->SelectBlock( m_selectingTopLeft
.GetRow(), 
6327                                           m_selectingTopLeft
.GetCol(), 
6328                                           m_selectingBottomRight
.GetRow(), 
6329                                           m_selectingBottomRight
.GetCol(), 
6330                                           event
.ControlDown(), 
6337         m_selectingTopLeft 
= wxGridNoCellCoords
; 
6338         m_selectingBottomRight 
= wxGridNoCellCoords
; 
6339         m_selectingKeyboard 
= wxGridNoCellCoords
; 
6343 void wxGrid::OnEraseBackground(wxEraseEvent
&) 
6347 void wxGrid::SetCurrentCell( const wxGridCellCoords
& coords 
) 
6349     if ( SendEvent( wxEVT_GRID_SELECT_CELL
, coords
.GetRow(), coords
.GetCol() ) ) 
6351         // the event has been intercepted - do nothing 
6355     wxClientDC 
dc(m_gridWin
); 
6358     if ( m_currentCellCoords 
!= wxGridNoCellCoords 
) 
6360         HideCellEditControl(); 
6361         DisableCellEditControl(); 
6363         if ( IsVisible( m_currentCellCoords
, FALSE 
) ) 
6366             r 
= BlockToDeviceRect(m_currentCellCoords
, m_currentCellCoords
); 
6367             if ( !m_gridLinesEnabled 
) 
6375             wxGridCellCoordsArray cells 
= CalcCellsExposed( r 
); 
6377             // Otherwise refresh redraws the highlight! 
6378             m_currentCellCoords 
= coords
; 
6380             DrawGridCellArea(dc
,cells
); 
6381             DrawAllGridLines( dc
, r 
); 
6385     m_currentCellCoords 
= coords
; 
6387     wxGridCellAttr
* attr 
= GetCellAttr(coords
); 
6388     DrawCellHighlight(dc
, attr
); 
6393 void wxGrid::HighlightBlock( int topRow
, int leftCol
, int bottomRow
, int rightCol 
) 
6396     wxGridCellCoords updateTopLeft
, updateBottomRight
; 
6400         if ( m_selection
->GetSelectionMode() == wxGrid::wxGridSelectRows 
) 
6403             rightCol 
= GetNumberCols() - 1; 
6405         else if ( m_selection
->GetSelectionMode() == wxGrid::wxGridSelectColumns 
) 
6408             bottomRow 
= GetNumberRows() - 1; 
6412     if ( topRow 
> bottomRow 
) 
6419     if ( leftCol 
> rightCol 
) 
6426     updateTopLeft 
= wxGridCellCoords( topRow
, leftCol 
); 
6427     updateBottomRight 
= wxGridCellCoords( bottomRow
, rightCol 
); 
6429     // First the case that we selected a completely new area 
6430     if ( m_selectingTopLeft 
== wxGridNoCellCoords 
|| 
6431          m_selectingBottomRight 
== wxGridNoCellCoords 
) 
6434         rect 
= BlockToDeviceRect( wxGridCellCoords ( topRow
, leftCol 
), 
6435                                   wxGridCellCoords ( bottomRow
, rightCol 
) ); 
6436         m_gridWin
->Refresh( FALSE
, &rect 
); 
6438     // Now handle changing an existing selection area. 
6439     else if ( m_selectingTopLeft 
!= updateTopLeft 
|| 
6440               m_selectingBottomRight 
!= updateBottomRight 
) 
6442         // Compute two optimal update rectangles: 
6443         // Either one rectangle is a real subset of the 
6444         // other, or they are (almost) disjoint! 
6446         bool    need_refresh
[4]; 
6450         need_refresh
[3] = FALSE
; 
6453         // Store intermediate values 
6454         wxCoord oldLeft   
= m_selectingTopLeft
.GetCol(); 
6455         wxCoord oldTop    
= m_selectingTopLeft
.GetRow(); 
6456         wxCoord oldRight  
= m_selectingBottomRight
.GetCol(); 
6457         wxCoord oldBottom 
= m_selectingBottomRight
.GetRow(); 
6459         // Determine the outer/inner coordinates. 
6460         if (oldLeft 
> leftCol
) 
6466         if (oldTop 
> topRow 
) 
6472         if (oldRight 
< rightCol 
) 
6475             oldRight 
= rightCol
; 
6478         if (oldBottom 
< bottomRow
) 
6481             oldBottom 
= bottomRow
; 
6485         // Now, either the stuff marked old is the outer 
6486         // rectangle or we don't have a situation where one 
6487         // is contained in the other. 
6489         if ( oldLeft 
< leftCol 
) 
6491             // Refresh the newly selected or deselected 
6492             // area to the left of the old or new selection. 
6493             need_refresh
[0] = TRUE
; 
6494             rect
[0] = BlockToDeviceRect( wxGridCellCoords ( oldTop
, 
6496                                          wxGridCellCoords ( oldBottom
, 
6500         if ( oldTop  
< topRow 
) 
6502             // Refresh the newly selected or deselected 
6503             // area above the old or new selection. 
6504             need_refresh
[1] = TRUE
; 
6505             rect
[1] = BlockToDeviceRect( wxGridCellCoords ( oldTop
, 
6507                                          wxGridCellCoords ( topRow 
- 1, 
6511         if ( oldRight 
> rightCol 
) 
6513             // Refresh the newly selected or deselected 
6514             // area to the right of the old or new selection. 
6515             need_refresh
[2] = TRUE
; 
6516             rect
[2] = BlockToDeviceRect( wxGridCellCoords ( oldTop
, 
6518                                          wxGridCellCoords ( oldBottom
, 
6522         if ( oldBottom 
> bottomRow 
) 
6524             // Refresh the newly selected or deselected 
6525             // area below the old or new selection. 
6526             need_refresh
[3] = TRUE
; 
6527             rect
[3] = BlockToDeviceRect( wxGridCellCoords ( bottomRow 
+ 1, 
6529                                          wxGridCellCoords ( oldBottom
, 
6533         // various Refresh() calls 
6534         for (i 
= 0; i 
< 4; i
++ ) 
6535             if ( need_refresh
[i
] && rect
[i
] != wxGridNoCellRect 
) 
6536                 m_gridWin
->Refresh( FALSE
, &(rect
[i
]) ); 
6539     m_selectingTopLeft 
= updateTopLeft
; 
6540     m_selectingBottomRight 
= updateBottomRight
; 
6544 // ------ functions to get/send data (see also public functions) 
6547 bool wxGrid::GetModelValues() 
6549     // Hide the editor, so it won't hide a changed value. 
6550     HideCellEditControl(); 
6554         // all we need to do is repaint the grid 
6556         m_gridWin
->Refresh(); 
6564 bool wxGrid::SetModelValues() 
6568     // Disable the editor, so it won't hide a changed value. 
6569     // Do we also want to save the current value of the editor first? 
6571     DisableCellEditControl(); 
6575         for ( row 
= 0;  row 
< m_numRows
;  row
++ ) 
6577             for ( col 
= 0;  col 
< m_numCols
;  col
++ ) 
6579                 m_table
->SetValue( row
, col
, GetCellValue(row
, col
) ); 
6591 // Note - this function only draws cells that are in the list of 
6592 // exposed cells (usually set from the update region by 
6593 // CalcExposedCells) 
6595 void wxGrid::DrawGridCellArea( wxDC
& dc
, const wxGridCellCoordsArray
& cells 
) 
6597     if ( !m_numRows 
|| !m_numCols 
) return; 
6599     int i
, numCells 
= cells
.GetCount(); 
6600     int row
, col
, cell_rows
, cell_cols
; 
6601     wxGridCellCoordsArray redrawCells
; 
6603     for ( i 
= numCells
-1; i 
>= 0;  i
-- ) 
6605         row 
= cells
[i
].GetRow(); 
6606         col 
= cells
[i
].GetCol(); 
6607         GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
6609         // If this cell is part of a multicell block, find owner for repaint 
6610         if ( cell_rows 
<= 0 || cell_cols 
<= 0 ) 
6612             wxGridCellCoords 
cell(row
+cell_rows
, col
+cell_cols
); 
6613             bool marked 
= FALSE
; 
6614             for ( int j 
= 0;  j 
< numCells
;  j
++ ) 
6616                 if ( cell 
== cells
[j
] ) 
6624                 int count 
= redrawCells
.GetCount(); 
6625                 for (int j 
= 0; j 
< count
; j
++) 
6627                     if ( cell 
== redrawCells
[j
] ) 
6633                 if (!marked
) redrawCells
.Add( cell 
); 
6635             continue; // don't bother drawing this cell 
6638         // If this cell is empty, find cell to left that might want to overflow 
6639         if (m_table 
&& m_table
->IsEmptyCell(row
, col
)) 
6641             for ( int l 
= 0; l 
< cell_rows
; l
++ ) 
6643                 // find a cell in this row to left alreay marked for repaint 
6645                 for (int k 
= 0; k 
< int(redrawCells
.GetCount()); k
++) 
6646                     if ((redrawCells
[k
].GetCol() < left
) && 
6647                         (redrawCells
[k
].GetRow() == row
)) 
6648                         left
=redrawCells
[k
].GetCol(); 
6650                 if (left 
== col
) left 
= 0; // oh well 
6652                 for (int j 
= col
-1; j 
>= left
; j
--) 
6654                     if (!m_table
->IsEmptyCell(row
+l
, j
)) 
6656                         if (GetCellOverflow(row
+l
, j
)) 
6658                             wxGridCellCoords 
cell(row
+l
, j
); 
6659                             bool marked 
= FALSE
; 
6661                             for (int k 
= 0; k 
< numCells
; k
++) 
6663                                 if ( cell 
== cells
[k
] ) 
6671                                 int count 
= redrawCells
.GetCount(); 
6672                                 for (int k 
= 0; k 
< count
; k
++) 
6674                                     if ( cell 
== redrawCells
[k
] ) 
6680                                 if (!marked
) redrawCells
.Add( cell 
); 
6688         DrawCell( dc
, cells
[i
] ); 
6691     numCells 
= redrawCells
.GetCount(); 
6693     for ( i 
= numCells 
- 1; i 
>= 0;  i
-- ) 
6695         DrawCell( dc
, redrawCells
[i
] ); 
6700 void wxGrid::DrawGridSpace( wxDC
& dc 
) 
6703   m_gridWin
->GetClientSize( &cw
, &ch 
); 
6706   CalcUnscrolledPosition( cw
, ch
, &right
, &bottom 
); 
6708   int rightCol 
= m_numCols 
> 0 ? GetColRight(m_numCols 
- 1) : 0; 
6709   int bottomRow 
= m_numRows 
> 0 ? GetRowBottom(m_numRows 
- 1) : 0 ; 
6711   if ( right 
> rightCol 
|| bottom 
> bottomRow 
) 
6714       CalcUnscrolledPosition( 0, 0, &left
, &top 
); 
6716       dc
.SetBrush( wxBrush(GetDefaultCellBackgroundColour(), wxSOLID
) ); 
6717       dc
.SetPen( *wxTRANSPARENT_PEN 
); 
6719       if ( right 
> rightCol 
) 
6721           dc
.DrawRectangle( rightCol
, top
, right 
- rightCol
, ch
); 
6724       if ( bottom 
> bottomRow 
) 
6726           dc
.DrawRectangle( left
, bottomRow
, cw
, bottom 
- bottomRow
); 
6732 void wxGrid::DrawCell( wxDC
& dc
, const wxGridCellCoords
& coords 
) 
6734     int row 
= coords
.GetRow(); 
6735     int col 
= coords
.GetCol(); 
6737     if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
6740     // we draw the cell border ourselves 
6741 #if !WXGRID_DRAW_LINES 
6742     if ( m_gridLinesEnabled 
) 
6743         DrawCellBorder( dc
, coords 
); 
6746     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
6748     bool isCurrent 
= coords 
== m_currentCellCoords
; 
6750     wxRect rect 
= CellToRect( row
, col 
); 
6752     // if the editor is shown, we should use it and not the renderer 
6753     // Note: However, only if it is really _shown_, i.e. not hidden! 
6754     if ( isCurrent 
&& IsCellEditControlShown() ) 
6756         wxGridCellEditor 
*editor 
= attr
->GetEditor(this, row
, col
); 
6757         editor
->PaintBackground(rect
, attr
); 
6762         // but all the rest is drawn by the cell renderer and hence may be 
6764         wxGridCellRenderer 
*renderer 
= attr
->GetRenderer(this, row
, col
); 
6765         renderer
->Draw(*this, *attr
, dc
, rect
, row
, col
, IsInSelection(coords
)); 
6772 void wxGrid::DrawCellHighlight( wxDC
& dc
, const wxGridCellAttr 
*attr 
) 
6774     int row 
= m_currentCellCoords
.GetRow(); 
6775     int col 
= m_currentCellCoords
.GetCol(); 
6777     if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
6780     wxRect rect 
= CellToRect(row
, col
); 
6782     // hmmm... what could we do here to show that the cell is disabled? 
6783     // for now, I just draw a thinner border than for the other ones, but 
6784     // it doesn't look really good 
6786     int penWidth 
= attr
->IsReadOnly() ? m_cellHighlightROPenWidth 
: m_cellHighlightPenWidth
; 
6790         // The center of th drawn line is where the position/width/height of 
6791         // the rectangle is actually at, (on wxMSW atr least,) so we will 
6792         // reduce the size of the rectangle to compensate for the thickness of 
6793         // the line.  If this is too strange on non wxMSW platforms then 
6794         // please #ifdef this appropriately. 
6795         rect
.x 
+= penWidth
/2; 
6796         rect
.y 
+= penWidth
/2; 
6797         rect
.width 
-= penWidth
-1; 
6798         rect
.height 
-= penWidth
-1; 
6801         // Now draw the rectangle 
6802         // use the cellHighlightColour if the cell is inside a selection, this 
6803         // will ensure the cell is always visible. 
6804         dc
.SetPen(wxPen(IsInSelection(row
,col
)?m_selectionForeground
:m_cellHighlightColour
, penWidth
, wxSOLID
)); 
6805         dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
6806         dc
.DrawRectangle(rect
); 
6810         // VZ: my experiments with 3d borders... 
6812         // how to properly set colours for arbitrary bg? 
6813         wxCoord x1 
= rect
.x
, 
6815                 x2 
= rect
.x 
+ rect
.width 
-1, 
6816                 y2 
= rect
.y 
+ rect
.height 
-1; 
6818         dc
.SetPen(*wxWHITE_PEN
); 
6819         dc
.DrawLine(x1
, y1
, x2
, y1
); 
6820         dc
.DrawLine(x1
, y1
, x1
, y2
); 
6822         dc
.DrawLine(x1 
+ 1, y2 
- 1, x2 
- 1, y2 
- 1); 
6823         dc
.DrawLine(x2 
- 1, y1 
+ 1, x2 
- 1, y2 
); 
6825         dc
.SetPen(*wxBLACK_PEN
); 
6826         dc
.DrawLine(x1
, y2
, x2
, y2
); 
6827         dc
.DrawLine(x2
, y1
, x2
, y2
+1); 
6832 void wxGrid::DrawCellBorder( wxDC
& dc
, const wxGridCellCoords
& coords 
) 
6834     int row 
= coords
.GetRow(); 
6835     int col 
= coords
.GetCol(); 
6836     if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
6839     dc
.SetPen( wxPen(GetGridLineColour(), 1, wxSOLID
) ); 
6841     wxRect rect 
= CellToRect( row
, col 
); 
6843     // right hand border 
6845     dc
.DrawLine( rect
.x 
+ rect
.width
, rect
.y
, 
6846                  rect
.x 
+ rect
.width
, rect
.y 
+ rect
.height 
+ 1 ); 
6850     dc
.DrawLine( rect
.x
,              rect
.y 
+ rect
.height
, 
6851                  rect
.x 
+ rect
.width
, rect
.y 
+ rect
.height
); 
6854 void wxGrid::DrawHighlight(wxDC
& dc
,const  wxGridCellCoordsArray
& cells
) 
6856     // This if block was previously in wxGrid::OnPaint but that doesn't 
6857     // seem to get called under wxGTK - MB 
6859     if ( m_currentCellCoords 
== wxGridNoCellCoords  
&& 
6860          m_numRows 
&& m_numCols 
) 
6862         m_currentCellCoords
.Set(0, 0); 
6865     if ( IsCellEditControlShown() ) 
6867         // don't show highlight when the edit control is shown 
6871     // if the active cell was repainted, repaint its highlight too because it 
6872     // might have been damaged by the grid lines 
6873     size_t count 
= cells
.GetCount(); 
6874     for ( size_t n 
= 0; n 
< count
; n
++ ) 
6876         if ( cells
[n
] == m_currentCellCoords 
) 
6878             wxGridCellAttr
* attr 
= GetCellAttr(m_currentCellCoords
); 
6879             DrawCellHighlight(dc
, attr
); 
6887 // TODO: remove this ??? 
6888 // This is used to redraw all grid lines e.g. when the grid line colour 
6891 void wxGrid::DrawAllGridLines( wxDC
& dc
, const wxRegion 
& WXUNUSED(reg
) ) 
6893 #if !WXGRID_DRAW_LINES 
6897     if ( !m_gridLinesEnabled 
|| 
6899          !m_numCols 
) return; 
6901     int top
, bottom
, left
, right
; 
6903 #if 0  //#ifndef __WXGTK__ 
6907       m_gridWin
->GetClientSize(&cw
, &ch
); 
6909       // virtual coords of visible area 
6911       CalcUnscrolledPosition( 0, 0, &left
, &top 
); 
6912       CalcUnscrolledPosition( cw
, ch
, &right
, &bottom 
); 
6917       reg
.GetBox(x
, y
, w
, h
); 
6918       CalcUnscrolledPosition( x
, y
, &left
, &top 
); 
6919       CalcUnscrolledPosition( x 
+ w
, y 
+ h
, &right
, &bottom 
); 
6923       m_gridWin
->GetClientSize(&cw
, &ch
); 
6924       CalcUnscrolledPosition( 0, 0, &left
, &top 
); 
6925       CalcUnscrolledPosition( cw
, ch
, &right
, &bottom 
); 
6928     // avoid drawing grid lines past the last row and col 
6930     right 
= wxMin( right
, GetColRight(m_numCols 
- 1) ); 
6931     bottom 
= wxMin( bottom
, GetRowBottom(m_numRows 
- 1) ); 
6933     // no gridlines inside multicells, clip them out 
6934     int leftCol   
= internalXToCol(left
); 
6935     int topRow    
= internalYToRow(top
); 
6936     int rightCol  
= internalXToCol(right
); 
6937     int bottomRow 
= internalYToRow(bottom
); 
6938     wxRegion 
clippedcells(0, 0, cw
, ch
); 
6941     int i
, j
, cell_rows
, cell_cols
; 
6944     for (j
=topRow
; j
<bottomRow
; j
++) 
6946         for (i
=leftCol
; i
<rightCol
; i
++) 
6948             GetCellSize( j
, i
, &cell_rows
, &cell_cols 
); 
6949             if ((cell_rows 
> 1) || (cell_cols 
> 1)) 
6951                 rect 
= CellToRect(j
,i
); 
6952                 CalcScrolledPosition( rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
6953                 clippedcells
.Subtract(rect
); 
6955             else if ((cell_rows 
< 0) || (cell_cols 
< 0)) 
6957                 rect 
= CellToRect(j
+cell_rows
, i
+cell_cols
); 
6958                 CalcScrolledPosition( rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
6959                 clippedcells
.Subtract(rect
); 
6963     dc
.SetClippingRegion( clippedcells 
); 
6965     dc
.SetPen( wxPen(GetGridLineColour(), 1, wxSOLID
) ); 
6967     // horizontal grid lines 
6969     // already declared above - int i; 
6970     for ( i 
= internalYToRow(top
); i 
< m_numRows
; i
++ ) 
6972         int bot 
= GetRowBottom(i
) - 1; 
6981             dc
.DrawLine( left
, bot
, right
, bot 
); 
6986     // vertical grid lines 
6988     for ( i 
= internalXToCol(left
); i 
< m_numCols
; i
++ ) 
6990         int colRight 
= GetColRight(i
) - 1; 
6991         if ( colRight 
> right 
) 
6996         if ( colRight 
>= left 
) 
6998             dc
.DrawLine( colRight
, top
, colRight
, bottom 
); 
7001     dc
.DestroyClippingRegion(); 
7005 void wxGrid::DrawRowLabels( wxDC
& dc 
,const wxArrayInt
& rows
) 
7007     if ( !m_numRows 
) return; 
7010     size_t numLabels 
= rows
.GetCount(); 
7012     for ( i 
= 0;  i 
< numLabels
;  i
++ ) 
7014         DrawRowLabel( dc
, rows
[i
] ); 
7019 void wxGrid::DrawRowLabel( wxDC
& dc
, int row 
) 
7021     if ( GetRowHeight(row
) <= 0 ) 
7024     int rowTop 
= GetRowTop(row
), 
7025         rowBottom 
= GetRowBottom(row
) - 1; 
7027     dc
.SetPen( wxPen(wxSystemSettings::GetColour(wxSYS_COLOUR_3DDKSHADOW
),1, wxSOLID
) ); 
7028     dc
.DrawLine( m_rowLabelWidth
-1, rowTop
, 
7029                  m_rowLabelWidth
-1, rowBottom 
); 
7031     dc
.DrawLine( 0, rowTop
, 0, rowBottom 
); 
7033     dc
.DrawLine( 0, rowBottom
, m_rowLabelWidth
, rowBottom 
); 
7035     dc
.SetPen( *wxWHITE_PEN 
); 
7036     dc
.DrawLine( 1, rowTop
, 1, rowBottom 
); 
7037     dc
.DrawLine( 1, rowTop
, m_rowLabelWidth
-1, rowTop 
); 
7039     dc
.SetBackgroundMode( wxTRANSPARENT 
); 
7040     dc
.SetTextForeground( GetLabelTextColour() ); 
7041     dc
.SetFont( GetLabelFont() ); 
7044     GetRowLabelAlignment( &hAlign
, &vAlign 
); 
7048     rect
.SetY( GetRowTop(row
) + 2 ); 
7049     rect
.SetWidth( m_rowLabelWidth 
- 4 ); 
7050     rect
.SetHeight( GetRowHeight(row
) - 4 ); 
7051     DrawTextRectangle( dc
, GetRowLabelValue( row 
), rect
, hAlign
, vAlign 
); 
7055 void wxGrid::DrawColLabels( wxDC
& dc
,const wxArrayInt
& cols 
) 
7057     if ( !m_numCols 
) return; 
7060     size_t numLabels 
= cols
.GetCount(); 
7062     for ( i 
= 0;  i 
< numLabels
;  i
++ ) 
7064         DrawColLabel( dc
, cols
[i
] ); 
7069 void wxGrid::DrawColLabel( wxDC
& dc
, int col 
) 
7071     if ( GetColWidth(col
) <= 0 ) 
7074     int colLeft 
= GetColLeft(col
), 
7075         colRight 
= GetColRight(col
) - 1; 
7077     dc
.SetPen( wxPen(wxSystemSettings::GetColour(wxSYS_COLOUR_3DDKSHADOW
),1, wxSOLID
) ); 
7078     dc
.DrawLine( colRight
, 0, 
7079                  colRight
, m_colLabelHeight
-1 ); 
7081     dc
.DrawLine( colLeft
, 0, colRight
, 0 ); 
7083     dc
.DrawLine( colLeft
, m_colLabelHeight
-1, 
7084                  colRight
+1, m_colLabelHeight
-1 ); 
7086     dc
.SetPen( *wxWHITE_PEN 
); 
7087     dc
.DrawLine( colLeft
, 1, colLeft
, m_colLabelHeight
-1 ); 
7088     dc
.DrawLine( colLeft
, 1, colRight
, 1 ); 
7090     dc
.SetBackgroundMode( wxTRANSPARENT 
); 
7091     dc
.SetTextForeground( GetLabelTextColour() ); 
7092     dc
.SetFont( GetLabelFont() ); 
7094     int hAlign
, vAlign
, orient
; 
7095     GetColLabelAlignment( &hAlign
, &vAlign 
); 
7096     orient 
= GetColLabelTextOrientation(); 
7099     rect
.SetX( colLeft 
+ 2 ); 
7101     rect
.SetWidth( GetColWidth(col
) - 4 ); 
7102     rect
.SetHeight( m_colLabelHeight 
- 4 ); 
7103     DrawTextRectangle( dc
, GetColLabelValue( col 
), rect
, hAlign
, vAlign
, orient 
); 
7106 void wxGrid::DrawTextRectangle( wxDC
& dc
, 
7107                                 const wxString
& value
, 
7111                                 int textOrientation 
) 
7113     wxArrayString lines
; 
7115     StringToLines( value
, lines 
); 
7118     //Forward to new API. 
7119     DrawTextRectangle(  dc
, 
7128 void wxGrid::DrawTextRectangle( wxDC
& dc
, 
7129                                const wxArrayString
& lines
, 
7133                                int textOrientation 
) 
7135     long textWidth
, textHeight
; 
7136     long lineWidth
, lineHeight
; 
7139     dc
.SetClippingRegion( rect 
); 
7141     nLines 
= lines
.GetCount(); 
7145         float x 
= 0.0, y 
= 0.0; 
7147         if( textOrientation 
== wxHORIZONTAL 
) 
7148             GetTextBoxSize(dc
, lines
, &textWidth
, &textHeight
); 
7150             GetTextBoxSize( dc
, lines
, &textHeight
, &textWidth 
); 
7154         case wxALIGN_BOTTOM
: 
7155             if( textOrientation 
== wxHORIZONTAL 
) 
7156                 y 
= rect
.y 
+ (rect
.height 
- textHeight 
- 1); 
7158                 x 
= rect
.x 
+ rect
.width 
- textWidth
; 
7161         case wxALIGN_CENTRE
: 
7162             if( textOrientation 
== wxHORIZONTAL 
) 
7163                 y 
= rect
.y 
+ ((rect
.height 
- textHeight
)/2); 
7165                 x 
= rect
.x 
+ ((rect
.width 
- textWidth
)/2); 
7170             if( textOrientation 
== wxHORIZONTAL 
) 
7177         // Align each line of a multi-line label 
7178         for( l 
= 0; l 
< nLines
; l
++ ) 
7180             dc
.GetTextExtent(lines
[l
], &lineWidth
, &lineHeight
); 
7182             switch( horizAlign 
) 
7185                 if( textOrientation 
== wxHORIZONTAL 
) 
7186                     x 
= rect
.x 
+ (rect
.width 
- lineWidth 
- 1); 
7188                     y 
= rect
.y 
+ lineWidth 
+ 1; 
7191             case wxALIGN_CENTRE
: 
7192                 if( textOrientation 
== wxHORIZONTAL 
) 
7193                     x 
= rect
.x 
+ ((rect
.width 
- lineWidth
)/2); 
7195                     y 
= rect
.y 
+ rect
.height 
- ((rect
.height 
- lineWidth
)/2); 
7200                 if( textOrientation 
== wxHORIZONTAL 
) 
7203                     y 
= rect
.y 
+ rect
.height 
- 1; 
7207             if( textOrientation 
== wxHORIZONTAL 
) 
7209                 dc
.DrawText( lines
[l
], (int)x
, (int)y 
); 
7214                 dc
.DrawRotatedText( lines
[l
], (int)x
, (int)y
, 90.0 ); 
7219     dc
.DestroyClippingRegion(); 
7223 // Split multi line text up into an array of strings.  Any existing 
7224 // contents of the string array are preserved. 
7226 void wxGrid::StringToLines( const wxString
& value
, wxArrayString
& lines 
) 
7230     wxString eol 
= wxTextFile::GetEOL( wxTextFileType_Unix 
); 
7231     wxString tVal 
= wxTextFile::Translate( value
, wxTextFileType_Unix 
); 
7233     while ( startPos 
< (int)tVal
.Length() ) 
7235         pos 
= tVal
.Mid(startPos
).Find( eol 
); 
7240         else if ( pos 
== 0 ) 
7242             lines
.Add( wxEmptyString 
); 
7246             lines
.Add( value
.Mid(startPos
, pos
) ); 
7250     if ( startPos 
< (int)value
.Length() ) 
7252         lines
.Add( value
.Mid( startPos 
) ); 
7257 void wxGrid::GetTextBoxSize( wxDC
& dc
, 
7258                              const wxArrayString
& lines
, 
7259                              long *width
, long *height 
) 
7266     for ( i 
= 0;  i 
< lines
.GetCount();  i
++ ) 
7268         dc
.GetTextExtent( lines
[i
], &lineW
, &lineH 
); 
7269         w 
= wxMax( w
, lineW 
); 
7278 // ------ Batch processing. 
7280 void wxGrid::EndBatch() 
7282     if ( m_batchCount 
> 0 ) 
7285         if ( !m_batchCount 
) 
7288             m_rowLabelWin
->Refresh(); 
7289             m_colLabelWin
->Refresh(); 
7290             m_cornerLabelWin
->Refresh(); 
7291             m_gridWin
->Refresh(); 
7296 // Use this, rather than wxWindow::Refresh(), to force an immediate 
7297 // repainting of the grid. Has no effect if you are already inside a 
7298 // BeginBatch / EndBatch block. 
7300 void wxGrid::ForceRefresh() 
7308 // ------ Edit control functions 
7312 void wxGrid::EnableEditing( bool edit 
) 
7314     // TODO: improve this ? 
7316     if ( edit 
!= m_editable 
) 
7318         if(!edit
) EnableCellEditControl(edit
); 
7324 void wxGrid::EnableCellEditControl( bool enable 
) 
7329     if ( m_currentCellCoords 
== wxGridNoCellCoords 
) 
7330         SetCurrentCell( 0, 0 ); 
7332     if ( enable 
!= m_cellEditCtrlEnabled 
) 
7336             if (SendEvent( wxEVT_GRID_EDITOR_SHOWN
) <0) 
7339             // this should be checked by the caller! 
7340             wxASSERT_MSG( CanEnableCellControl(), 
7341                           _T("can't enable editing for this cell!") ); 
7343             // do it before ShowCellEditControl() 
7344             m_cellEditCtrlEnabled 
= enable
; 
7346             ShowCellEditControl(); 
7350             //FIXME:add veto support 
7351             SendEvent( wxEVT_GRID_EDITOR_HIDDEN
); 
7353             HideCellEditControl(); 
7354             SaveEditControlValue(); 
7356             // do it after HideCellEditControl() 
7357             m_cellEditCtrlEnabled 
= enable
; 
7362 bool wxGrid::IsCurrentCellReadOnly() const 
7365     wxGridCellAttr
* attr 
= ((wxGrid 
*)this)->GetCellAttr(m_currentCellCoords
); 
7366     bool readonly 
= attr
->IsReadOnly(); 
7372 bool wxGrid::CanEnableCellControl() const 
7374     return m_editable 
&& !IsCurrentCellReadOnly(); 
7377 bool wxGrid::IsCellEditControlEnabled() const 
7379     // the cell edit control might be disable for all cells or just for the 
7380     // current one if it's read only 
7381     return m_cellEditCtrlEnabled 
? !IsCurrentCellReadOnly() : FALSE
; 
7384 bool wxGrid::IsCellEditControlShown() const 
7386     bool isShown 
= FALSE
; 
7388     if ( m_cellEditCtrlEnabled 
) 
7390         int row 
= m_currentCellCoords
.GetRow(); 
7391         int col 
= m_currentCellCoords
.GetCol(); 
7392         wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
7393         wxGridCellEditor
* editor 
= attr
->GetEditor((wxGrid
*) this, row
, col
); 
7398             if ( editor
->IsCreated() ) 
7400                 isShown 
= editor
->GetControl()->IsShown(); 
7410 void wxGrid::ShowCellEditControl() 
7412     if ( IsCellEditControlEnabled() ) 
7414         if ( !IsVisible( m_currentCellCoords 
) ) 
7416             m_cellEditCtrlEnabled 
= FALSE
; 
7421             wxRect rect 
= CellToRect( m_currentCellCoords 
); 
7422             int row 
= m_currentCellCoords
.GetRow(); 
7423             int col 
= m_currentCellCoords
.GetCol(); 
7425             // if this is part of a multicell, find owner (topleft) 
7426             int cell_rows
, cell_cols
; 
7427             GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
7428             if ( cell_rows 
<= 0 || cell_cols 
<= 0 ) 
7432                 m_currentCellCoords
.SetRow( row 
); 
7433                 m_currentCellCoords
.SetCol( col 
); 
7436             // convert to scrolled coords 
7438             CalcScrolledPosition( rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
7440             // done in PaintBackground() 
7442             // erase the highlight and the cell contents because the editor 
7443             // might not cover the entire cell 
7444             wxClientDC 
dc( m_gridWin 
); 
7446             dc
.SetBrush(*wxLIGHT_GREY_BRUSH
); //wxBrush(attr->GetBackgroundColour(), wxSOLID)); 
7447             dc
.SetPen(*wxTRANSPARENT_PEN
); 
7448             dc
.DrawRectangle(rect
); 
7451             // cell is shifted by one pixel 
7452             // However, don't allow x or y to become negative 
7453             // since the SetSize() method interprets that as 
7460             wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
7461             wxGridCellEditor
* editor 
= attr
->GetEditor(this, row
, col
); 
7462             if ( !editor
->IsCreated() ) 
7464                 editor
->Create(m_gridWin
, -1, 
7465                                new wxGridCellEditorEvtHandler(this, editor
)); 
7467                 wxGridEditorCreatedEvent 
evt(GetId(), 
7468                                              wxEVT_GRID_EDITOR_CREATED
, 
7472                                              editor
->GetControl()); 
7473                 GetEventHandler()->ProcessEvent(evt
); 
7477             // resize editor to overflow into righthand cells if allowed 
7478             int maxWidth 
= rect
.width
; 
7479             wxString value 
= GetCellValue(row
, col
); 
7480             if ( (value 
!= wxEmptyString
) && (attr
->GetOverflow()) ) 
7483                 GetTextExtent(value
, &maxWidth
, &y
, 
7484                         NULL
, NULL
, &attr
->GetFont()); 
7485                 if (maxWidth 
< rect
.width
) maxWidth 
= rect
.width
; 
7487             int client_right 
= m_gridWin
->GetClientSize().GetWidth(); 
7488             if (rect
.x
+maxWidth 
> client_right
) 
7489                 maxWidth 
= client_right 
- rect
.x
; 
7491             if ((maxWidth 
> rect
.width
) && (col 
< m_numCols
) && m_table
) 
7493                 GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
7494                 // may have changed earlier 
7495                 for (int i 
= col
+cell_cols
; i 
< m_numCols
; i
++) 
7498                     GetCellSize( row
, i
, &c_rows
, &c_cols 
); 
7499                     // looks weird going over a multicell 
7500                     if (m_table
->IsEmptyCell(row
,i
) && 
7501                             (rect
.width 
< maxWidth
) && (c_rows 
== 1)) 
7502                         rect
.width 
+= GetColWidth(i
); 
7506                 if (rect
.GetRight() > client_right
) 
7507                     rect
.SetRight(client_right
-1); 
7510             editor
->SetCellAttr(attr
); 
7511             editor
->SetSize( rect 
); 
7512             editor
->Show( TRUE
, attr 
); 
7514             // recalc dimensions in case we need to 
7515             // expand the scrolled window to account for editor 
7518             editor
->BeginEdit(row
, col
, this); 
7519             editor
->SetCellAttr(NULL
); 
7528 void wxGrid::HideCellEditControl() 
7530     if ( IsCellEditControlEnabled() ) 
7532         int row 
= m_currentCellCoords
.GetRow(); 
7533         int col 
= m_currentCellCoords
.GetCol(); 
7535         wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
7536         wxGridCellEditor 
*editor 
= attr
->GetEditor(this, row
, col
); 
7537         editor
->Show( FALSE 
); 
7540         m_gridWin
->SetFocus(); 
7541         // refresh whole row to the right 
7542         wxRect 
rect( CellToRect(row
, col
) ); 
7543         CalcScrolledPosition(rect
.x
, rect
.y
, &rect
.x
, &rect
.y 
); 
7544         rect
.width 
= m_gridWin
->GetClientSize().GetWidth() - rect
.x
; 
7545         m_gridWin
->Refresh( FALSE
, &rect 
); 
7550 void wxGrid::SaveEditControlValue() 
7552     if ( IsCellEditControlEnabled() ) 
7554         int row 
= m_currentCellCoords
.GetRow(); 
7555         int col 
= m_currentCellCoords
.GetCol(); 
7557         wxString oldval 
= GetCellValue(row
,col
); 
7559         wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
7560         wxGridCellEditor
* editor 
= attr
->GetEditor(this, row
, col
); 
7561         bool changed 
= editor
->EndEdit(row
, col
, this); 
7568             if ( SendEvent( wxEVT_GRID_CELL_CHANGE
, 
7569                        m_currentCellCoords
.GetRow(), 
7570                        m_currentCellCoords
.GetCol() ) < 0 ) { 
7572                 // Event has been vetoed, set the data back. 
7573                 SetCellValue(row
,col
,oldval
); 
7581 // ------ Grid location functions 
7582 //  Note that all of these functions work with the logical coordinates of 
7583 //  grid cells and labels so you will need to convert from device 
7584 //  coordinates for mouse events etc. 
7587 void wxGrid::XYToCell( int x
, int y
, wxGridCellCoords
& coords 
) 
7589     int row 
= YToRow(y
); 
7590     int col 
= XToCol(x
); 
7592     if ( row 
== -1  ||  col 
== -1 ) 
7594         coords 
= wxGridNoCellCoords
; 
7598         coords
.Set( row
, col 
); 
7603 // Internal Helper function for computing row or column from some 
7604 // (unscrolled) coordinate value, using either 
7605 // m_defaultRowHeight/m_defaultColWidth or binary search on array 
7606 // of m_rowBottoms/m_ColRights to speed up the search! 
7608 static int CoordToRowOrCol(int coord
, int defaultDist
, int minDist
, 
7609                            const wxArrayInt
& BorderArray
, int nMax
, 
7614         return clipToMinMax 
&& (nMax 
> 0) ? 0 : -1; 
7620     size_t i_max 
= coord 
/ defaultDist
, 
7623     if (BorderArray
.IsEmpty()) 
7625         if((int) i_max 
< nMax
) 
7627         return clipToMinMax 
? nMax 
- 1 : -1; 
7630     if ( i_max 
>= BorderArray
.GetCount()) 
7631         i_max 
= BorderArray
.GetCount() - 1; 
7634         if ( coord 
>= BorderArray
[i_max
]) 
7637             i_max 
= coord 
/ minDist
; 
7639         if ( i_max 
>= BorderArray
.GetCount()) 
7640             i_max 
= BorderArray
.GetCount() - 1; 
7642     if ( coord 
>= BorderArray
[i_max
]) 
7643         return clipToMinMax 
? (int)i_max 
: -1; 
7644     if ( coord 
< BorderArray
[0] ) 
7647     while ( i_max 
- i_min 
> 0 ) 
7649         wxCHECK_MSG(BorderArray
[i_min
] <= coord 
&& coord 
< BorderArray
[i_max
], 
7650                     0, _T("wxGrid: internal error in CoordToRowOrCol")); 
7651         if (coord 
>=  BorderArray
[ i_max 
- 1]) 
7655         int median 
= i_min 
+ (i_max 
- i_min 
+ 1) / 2; 
7656         if (coord 
< BorderArray
[median
]) 
7664 int wxGrid::YToRow( int y 
) 
7666     return CoordToRowOrCol(y
, m_defaultRowHeight
, 
7667                            m_minAcceptableRowHeight
, m_rowBottoms
, m_numRows
, FALSE
); 
7671 int wxGrid::XToCol( int x 
) 
7673     return CoordToRowOrCol(x
, m_defaultColWidth
, 
7674                            m_minAcceptableColWidth
, m_colRights
, m_numCols
, FALSE
); 
7678 // return the row number that that the y coord is near the edge of, or 
7679 // -1 if not near an edge 
7681 int wxGrid::YToEdgeOfRow( int y 
) 
7684     i 
= internalYToRow(y
); 
7686     if ( GetRowHeight(i
) > WXGRID_LABEL_EDGE_ZONE 
) 
7688         // We know that we are in row i, test whether we are 
7689         // close enough to lower or upper border, respectively. 
7690         if ( abs(GetRowBottom(i
) - y
) < WXGRID_LABEL_EDGE_ZONE 
) 
7692         else if( i 
> 0 && y 
- GetRowTop(i
) < WXGRID_LABEL_EDGE_ZONE 
) 
7700 // return the col number that that the x coord is near the edge of, or 
7701 // -1 if not near an edge 
7703 int wxGrid::XToEdgeOfCol( int x 
) 
7706     i 
= internalXToCol(x
); 
7708     if ( GetColWidth(i
) > WXGRID_LABEL_EDGE_ZONE 
) 
7710         // We know that we are in column i,  test whether we are 
7711         // close enough to right or left border, respectively. 
7712         if ( abs(GetColRight(i
) - x
) < WXGRID_LABEL_EDGE_ZONE 
) 
7714         else if( i 
> 0 && x 
- GetColLeft(i
) < WXGRID_LABEL_EDGE_ZONE 
) 
7722 wxRect 
wxGrid::CellToRect( int row
, int col 
) 
7724     wxRect 
rect( -1, -1, -1, -1 ); 
7726     if ( row 
>= 0  &&  row 
< m_numRows  
&& 
7727          col 
>= 0  &&  col 
< m_numCols 
) 
7729         int i
, cell_rows
, cell_cols
; 
7730         rect
.width 
= rect
.height 
= 0; 
7731         GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
7732         // if negative then find multicell owner 
7733         if (cell_rows 
< 0) row 
+= cell_rows
; 
7734         if (cell_cols 
< 0) col 
+= cell_cols
; 
7735         GetCellSize( row
, col
, &cell_rows
, &cell_cols 
); 
7737         rect
.x 
= GetColLeft(col
); 
7738         rect
.y 
= GetRowTop(row
); 
7739         for (i
=col
; i
<col
+cell_cols
; i
++) 
7740             rect
.width  
+= GetColWidth(i
); 
7741         for (i
=row
; i
<row
+cell_rows
; i
++) 
7742             rect
.height 
+= GetRowHeight(i
); 
7745     // if grid lines are enabled, then the area of the cell is a bit smaller 
7746     if (m_gridLinesEnabled
) { 
7754 bool wxGrid::IsVisible( int row
, int col
, bool wholeCellVisible 
) 
7756     // get the cell rectangle in logical coords 
7758     wxRect 
r( CellToRect( row
, col 
) ); 
7760     // convert to device coords 
7762     int left
, top
, right
, bottom
; 
7763     CalcScrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top 
); 
7764     CalcScrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom 
); 
7766     // check against the client area of the grid window 
7769     m_gridWin
->GetClientSize( &cw
, &ch 
); 
7771     if ( wholeCellVisible 
) 
7773         // is the cell wholly visible ? 
7775         return ( left 
>= 0  &&  right 
<= cw  
&& 
7776                  top 
>= 0  &&  bottom 
<= ch 
); 
7780         // is the cell partly visible ? 
7782         return ( ((left 
>=0 && left 
< cw
) || (right 
> 0 && right 
<= cw
))  && 
7783                  ((top 
>=0 && top 
< ch
) || (bottom 
> 0 && bottom 
<= ch
)) ); 
7788 // make the specified cell location visible by doing a minimal amount 
7791 void wxGrid::MakeCellVisible( int row
, int col 
) 
7795     int xpos 
= -1, ypos 
= -1; 
7797     if ( row 
>= 0  &&  row 
< m_numRows  
&& 
7798          col 
>= 0  &&  col 
< m_numCols 
) 
7800         // get the cell rectangle in logical coords 
7802         wxRect 
r( CellToRect( row
, col 
) ); 
7804         // convert to device coords 
7806         int left
, top
, right
, bottom
; 
7807         CalcScrolledPosition( r
.GetLeft(), r
.GetTop(), &left
, &top 
); 
7808         CalcScrolledPosition( r
.GetRight(), r
.GetBottom(), &right
, &bottom 
); 
7811         m_gridWin
->GetClientSize( &cw
, &ch 
); 
7817         else if ( bottom 
> ch 
) 
7819             int h 
= r
.GetHeight(); 
7821             for ( i 
= row
-1;  i 
>= 0;  i
-- ) 
7823                 int rowHeight 
= GetRowHeight(i
); 
7824                 if ( h 
+ rowHeight 
> ch 
) 
7831             // we divide it later by GRID_SCROLL_LINE, make sure that we don't 
7832             // have rounding errors (this is important, because if we do, we 
7833             // might not scroll at all and some cells won't be redrawn) 
7835             // Sometimes GRID_SCROLL_LINE/2 is not enough, so just add a full 
7837             ypos 
+= GRID_SCROLL_LINE_Y
; 
7844         else if ( right 
> cw 
) 
7846             // position the view so that the cell is on the right 
7848             CalcUnscrolledPosition(0, 0, &x0
, &y0
); 
7849             xpos 
= x0 
+ (right 
- cw
); 
7851             // see comment for ypos above 
7852             xpos 
+= GRID_SCROLL_LINE_X
; 
7855         if ( xpos 
!= -1  ||  ypos 
!= -1 ) 
7858                 xpos 
/= GRID_SCROLL_LINE_X
; 
7860                 ypos 
/= GRID_SCROLL_LINE_Y
; 
7861             Scroll( xpos
, ypos 
); 
7869 // ------ Grid cursor movement functions 
7872 bool wxGrid::MoveCursorUp( bool expandSelection 
) 
7874     if ( m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
7875          m_currentCellCoords
.GetRow() >= 0 ) 
7877         if ( expandSelection
) 
7879             if ( m_selectingKeyboard 
== wxGridNoCellCoords 
) 
7880                 m_selectingKeyboard 
= m_currentCellCoords
; 
7881             if ( m_selectingKeyboard
.GetRow() > 0 ) 
7883                 m_selectingKeyboard
.SetRow( m_selectingKeyboard
.GetRow() - 1 ); 
7884                 MakeCellVisible( m_selectingKeyboard
.GetRow(), 
7885                                  m_selectingKeyboard
.GetCol() ); 
7886                 HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
7889         else if ( m_currentCellCoords
.GetRow() > 0 ) 
7892             MakeCellVisible( m_currentCellCoords
.GetRow() - 1, 
7893                              m_currentCellCoords
.GetCol() ); 
7894             SetCurrentCell( m_currentCellCoords
.GetRow() - 1, 
7895                             m_currentCellCoords
.GetCol() ); 
7906 bool wxGrid::MoveCursorDown( bool expandSelection 
) 
7908     if ( m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
7909          m_currentCellCoords
.GetRow() < m_numRows 
) 
7911         if ( expandSelection 
) 
7913             if ( m_selectingKeyboard 
== wxGridNoCellCoords 
) 
7914                 m_selectingKeyboard 
= m_currentCellCoords
; 
7915             if ( m_selectingKeyboard
.GetRow() < m_numRows
-1 ) 
7917                 m_selectingKeyboard
.SetRow( m_selectingKeyboard
.GetRow() + 1 ); 
7918                 MakeCellVisible( m_selectingKeyboard
.GetRow(), 
7919                         m_selectingKeyboard
.GetCol() ); 
7920                 HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
7923         else if ( m_currentCellCoords
.GetRow() < m_numRows 
- 1 ) 
7926             MakeCellVisible( m_currentCellCoords
.GetRow() + 1, 
7927                              m_currentCellCoords
.GetCol() ); 
7928             SetCurrentCell( m_currentCellCoords
.GetRow() + 1, 
7929                             m_currentCellCoords
.GetCol() ); 
7940 bool wxGrid::MoveCursorLeft( bool expandSelection 
) 
7942     if ( m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
7943          m_currentCellCoords
.GetCol() >= 0 ) 
7945         if ( expandSelection 
) 
7947             if ( m_selectingKeyboard 
== wxGridNoCellCoords 
) 
7948                 m_selectingKeyboard 
= m_currentCellCoords
; 
7949             if ( m_selectingKeyboard
.GetCol() > 0 ) 
7951                 m_selectingKeyboard
.SetCol( m_selectingKeyboard
.GetCol() - 1 ); 
7952                 MakeCellVisible( m_selectingKeyboard
.GetRow(), 
7953                         m_selectingKeyboard
.GetCol() ); 
7954                 HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
7957         else if ( m_currentCellCoords
.GetCol() > 0 ) 
7960             MakeCellVisible( m_currentCellCoords
.GetRow(), 
7961                              m_currentCellCoords
.GetCol() - 1 ); 
7962             SetCurrentCell( m_currentCellCoords
.GetRow(), 
7963                             m_currentCellCoords
.GetCol() - 1 ); 
7974 bool wxGrid::MoveCursorRight( bool expandSelection 
) 
7976     if ( m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
7977          m_currentCellCoords
.GetCol() < m_numCols 
) 
7979         if ( expandSelection 
) 
7981             if ( m_selectingKeyboard 
== wxGridNoCellCoords 
) 
7982                 m_selectingKeyboard 
= m_currentCellCoords
; 
7983             if ( m_selectingKeyboard
.GetCol() < m_numCols 
- 1 ) 
7985                 m_selectingKeyboard
.SetCol( m_selectingKeyboard
.GetCol() + 1 ); 
7986                 MakeCellVisible( m_selectingKeyboard
.GetRow(), 
7987                         m_selectingKeyboard
.GetCol() ); 
7988                 HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
7991         else if ( m_currentCellCoords
.GetCol() < m_numCols 
- 1 ) 
7994             MakeCellVisible( m_currentCellCoords
.GetRow(), 
7995                              m_currentCellCoords
.GetCol() + 1 ); 
7996             SetCurrentCell( m_currentCellCoords
.GetRow(), 
7997                             m_currentCellCoords
.GetCol() + 1 ); 
8008 bool wxGrid::MovePageUp() 
8010     if ( m_currentCellCoords 
== wxGridNoCellCoords 
) return FALSE
; 
8012     int row 
= m_currentCellCoords
.GetRow(); 
8016         m_gridWin
->GetClientSize( &cw
, &ch 
); 
8018         int y 
= GetRowTop(row
); 
8019         int newRow 
= internalYToRow( y 
- ch 
+ 1 ); 
8021         if ( newRow 
== row 
) 
8023             //row > 0 , so newrow can never be less than 0 here. 
8027         MakeCellVisible( newRow
, m_currentCellCoords
.GetCol() ); 
8028         SetCurrentCell( newRow
, m_currentCellCoords
.GetCol() ); 
8036 bool wxGrid::MovePageDown() 
8038     if ( m_currentCellCoords 
== wxGridNoCellCoords 
) return FALSE
; 
8040     int row 
= m_currentCellCoords
.GetRow(); 
8041     if ( (row
+1) < m_numRows 
) 
8044         m_gridWin
->GetClientSize( &cw
, &ch 
); 
8046         int y 
= GetRowTop(row
); 
8047         int newRow 
= internalYToRow( y 
+ ch 
); 
8048         if ( newRow 
== row 
) 
8050             // row < m_numRows , so newrow can't overflow here. 
8054         MakeCellVisible( newRow
, m_currentCellCoords
.GetCol() ); 
8055         SetCurrentCell( newRow
, m_currentCellCoords
.GetCol() ); 
8063 bool wxGrid::MoveCursorUpBlock( bool expandSelection 
) 
8066          m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
8067          m_currentCellCoords
.GetRow() > 0 ) 
8069         int row 
= m_currentCellCoords
.GetRow(); 
8070         int col 
= m_currentCellCoords
.GetCol(); 
8072         if ( m_table
->IsEmptyCell(row
, col
) ) 
8074             // starting in an empty cell: find the next block of 
8080                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8083         else if ( m_table
->IsEmptyCell(row
-1, col
) ) 
8085             // starting at the top of a block: find the next block 
8091                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8096             // starting within a block: find the top of the block 
8101                 if ( m_table
->IsEmptyCell(row
, col
) ) 
8109         MakeCellVisible( row
, col 
); 
8110         if ( expandSelection 
) 
8112             m_selectingKeyboard 
= wxGridCellCoords( row
, col 
); 
8113             HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8118             SetCurrentCell( row
, col 
); 
8126 bool wxGrid::MoveCursorDownBlock( bool expandSelection 
) 
8129          m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
8130          m_currentCellCoords
.GetRow() < m_numRows
-1 ) 
8132         int row 
= m_currentCellCoords
.GetRow(); 
8133         int col 
= m_currentCellCoords
.GetCol(); 
8135         if ( m_table
->IsEmptyCell(row
, col
) ) 
8137             // starting in an empty cell: find the next block of 
8140             while ( row 
< m_numRows
-1 ) 
8143                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8146         else if ( m_table
->IsEmptyCell(row
+1, col
) ) 
8148             // starting at the bottom of a block: find the next block 
8151             while ( row 
< m_numRows
-1 ) 
8154                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8159             // starting within a block: find the bottom of the block 
8161             while ( row 
< m_numRows
-1 ) 
8164                 if ( m_table
->IsEmptyCell(row
, col
) ) 
8172         MakeCellVisible( row
, col 
); 
8173         if ( expandSelection 
) 
8175             m_selectingKeyboard 
= wxGridCellCoords( row
, col 
); 
8176             HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8181             SetCurrentCell( row
, col 
); 
8190 bool wxGrid::MoveCursorLeftBlock( bool expandSelection 
) 
8193          m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
8194          m_currentCellCoords
.GetCol() > 0 ) 
8196         int row 
= m_currentCellCoords
.GetRow(); 
8197         int col 
= m_currentCellCoords
.GetCol(); 
8199         if ( m_table
->IsEmptyCell(row
, col
) ) 
8201             // starting in an empty cell: find the next block of 
8207                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8210         else if ( m_table
->IsEmptyCell(row
, col
-1) ) 
8212             // starting at the left of a block: find the next block 
8218                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8223             // starting within a block: find the left of the block 
8228                 if ( m_table
->IsEmptyCell(row
, col
) ) 
8236         MakeCellVisible( row
, col 
); 
8237         if ( expandSelection 
) 
8239             m_selectingKeyboard 
= wxGridCellCoords( row
, col 
); 
8240             HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8245             SetCurrentCell( row
, col 
); 
8254 bool wxGrid::MoveCursorRightBlock( bool expandSelection 
) 
8257          m_currentCellCoords 
!= wxGridNoCellCoords  
&& 
8258          m_currentCellCoords
.GetCol() < m_numCols
-1 ) 
8260         int row 
= m_currentCellCoords
.GetRow(); 
8261         int col 
= m_currentCellCoords
.GetCol(); 
8263         if ( m_table
->IsEmptyCell(row
, col
) ) 
8265             // starting in an empty cell: find the next block of 
8268             while ( col 
< m_numCols
-1 ) 
8271                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8274         else if ( m_table
->IsEmptyCell(row
, col
+1) ) 
8276             // starting at the right of a block: find the next block 
8279             while ( col 
< m_numCols
-1 ) 
8282                 if ( !(m_table
->IsEmptyCell(row
, col
)) ) break; 
8287             // starting within a block: find the right of the block 
8289             while ( col 
< m_numCols
-1 ) 
8292                 if ( m_table
->IsEmptyCell(row
, col
) ) 
8300         MakeCellVisible( row
, col 
); 
8301         if ( expandSelection 
) 
8303             m_selectingKeyboard 
= wxGridCellCoords( row
, col 
); 
8304             HighlightBlock( m_currentCellCoords
, m_selectingKeyboard 
); 
8309             SetCurrentCell( row
, col 
); 
8321 // ------ Label values and formatting 
8324 void wxGrid::GetRowLabelAlignment( int *horiz
, int *vert 
) 
8326     *horiz 
= m_rowLabelHorizAlign
; 
8327     *vert  
= m_rowLabelVertAlign
; 
8330 void wxGrid::GetColLabelAlignment( int *horiz
, int *vert 
) 
8332     *horiz 
= m_colLabelHorizAlign
; 
8333     *vert  
= m_colLabelVertAlign
; 
8336 int wxGrid::GetColLabelTextOrientation() 
8338     return m_colLabelTextOrientation
; 
8341 wxString 
wxGrid::GetRowLabelValue( int row 
) 
8345         return m_table
->GetRowLabelValue( row 
); 
8355 wxString 
wxGrid::GetColLabelValue( int col 
) 
8359         return m_table
->GetColLabelValue( col 
); 
8370 void wxGrid::SetRowLabelSize( int width 
) 
8372     width 
= wxMax( width
, 0 ); 
8373     if ( width 
!= m_rowLabelWidth 
) 
8377             m_rowLabelWin
->Show( FALSE 
); 
8378             m_cornerLabelWin
->Show( FALSE 
); 
8380         else if ( m_rowLabelWidth 
== 0 ) 
8382             m_rowLabelWin
->Show( TRUE 
); 
8383             if ( m_colLabelHeight 
> 0 ) m_cornerLabelWin
->Show( TRUE 
); 
8386         m_rowLabelWidth 
= width
; 
8388         wxScrolledWindow::Refresh( TRUE 
); 
8393 void wxGrid::SetColLabelSize( int height 
) 
8395     height 
= wxMax( height
, 0 ); 
8396     if ( height 
!= m_colLabelHeight 
) 
8400             m_colLabelWin
->Show( FALSE 
); 
8401             m_cornerLabelWin
->Show( FALSE 
); 
8403         else if ( m_colLabelHeight 
== 0 ) 
8405             m_colLabelWin
->Show( TRUE 
); 
8406             if ( m_rowLabelWidth 
> 0 ) m_cornerLabelWin
->Show( TRUE 
); 
8409         m_colLabelHeight 
= height
; 
8411         wxScrolledWindow::Refresh( TRUE 
); 
8416 void wxGrid::SetLabelBackgroundColour( const wxColour
& colour 
) 
8418     if ( m_labelBackgroundColour 
!= colour 
) 
8420         m_labelBackgroundColour 
= colour
; 
8421         m_rowLabelWin
->SetBackgroundColour( colour 
); 
8422         m_colLabelWin
->SetBackgroundColour( colour 
); 
8423         m_cornerLabelWin
->SetBackgroundColour( colour 
); 
8425         if ( !GetBatchCount() ) 
8427             m_rowLabelWin
->Refresh(); 
8428             m_colLabelWin
->Refresh(); 
8429             m_cornerLabelWin
->Refresh(); 
8434 void wxGrid::SetLabelTextColour( const wxColour
& colour 
) 
8436     if ( m_labelTextColour 
!= colour 
) 
8438         m_labelTextColour 
= colour
; 
8439         if ( !GetBatchCount() ) 
8441             m_rowLabelWin
->Refresh(); 
8442             m_colLabelWin
->Refresh(); 
8447 void wxGrid::SetLabelFont( const wxFont
& font 
) 
8450     if ( !GetBatchCount() ) 
8452         m_rowLabelWin
->Refresh(); 
8453         m_colLabelWin
->Refresh(); 
8457 void wxGrid::SetRowLabelAlignment( int horiz
, int vert 
) 
8459     // allow old (incorrect) defs to be used 
8462         case wxLEFT
:   horiz 
= wxALIGN_LEFT
; break; 
8463         case wxRIGHT
:  horiz 
= wxALIGN_RIGHT
; break; 
8464         case wxCENTRE
: horiz 
= wxALIGN_CENTRE
; break; 
8469         case wxTOP
:    vert 
= wxALIGN_TOP
;    break; 
8470         case wxBOTTOM
: vert 
= wxALIGN_BOTTOM
; break; 
8471         case wxCENTRE
: vert 
= wxALIGN_CENTRE
; break; 
8474     if ( horiz 
== wxALIGN_LEFT 
|| horiz 
== wxALIGN_CENTRE 
|| horiz 
== wxALIGN_RIGHT 
) 
8476         m_rowLabelHorizAlign 
= horiz
; 
8479     if ( vert 
== wxALIGN_TOP 
|| vert 
== wxALIGN_CENTRE 
|| vert 
== wxALIGN_BOTTOM 
) 
8481         m_rowLabelVertAlign 
= vert
; 
8484     if ( !GetBatchCount() ) 
8486         m_rowLabelWin
->Refresh(); 
8490 void wxGrid::SetColLabelAlignment( int horiz
, int vert 
) 
8492     // allow old (incorrect) defs to be used 
8495         case wxLEFT
:   horiz 
= wxALIGN_LEFT
; break; 
8496         case wxRIGHT
:  horiz 
= wxALIGN_RIGHT
; break; 
8497         case wxCENTRE
: horiz 
= wxALIGN_CENTRE
; break; 
8502         case wxTOP
:    vert 
= wxALIGN_TOP
;    break; 
8503         case wxBOTTOM
: vert 
= wxALIGN_BOTTOM
; break; 
8504         case wxCENTRE
: vert 
= wxALIGN_CENTRE
; break; 
8507     if ( horiz 
== wxALIGN_LEFT 
|| horiz 
== wxALIGN_CENTRE 
|| horiz 
== wxALIGN_RIGHT 
) 
8509         m_colLabelHorizAlign 
= horiz
; 
8512     if ( vert 
== wxALIGN_TOP 
|| vert 
== wxALIGN_CENTRE 
|| vert 
== wxALIGN_BOTTOM 
) 
8514         m_colLabelVertAlign 
= vert
; 
8517     if ( !GetBatchCount() ) 
8519         m_colLabelWin
->Refresh(); 
8523 // Note:        under MSW, the default column label font must be changed because it 
8524 //                              does not support vertical printing 
8526 // Example: wxFont font(9, wxSWISS, wxNORMAL, wxBOLD); 
8527 //                              pGrid->SetLabelFont(font); 
8528 //              pGrid->SetColLabelTextOrientation(wxVERTICAL); 
8530 void wxGrid::SetColLabelTextOrientation( int textOrientation 
) 
8532     if( textOrientation 
== wxHORIZONTAL 
|| textOrientation 
== wxVERTICAL 
) 
8534         m_colLabelTextOrientation 
= textOrientation
; 
8537     if ( !GetBatchCount() ) 
8539         m_colLabelWin
->Refresh(); 
8543 void wxGrid::SetRowLabelValue( int row
, const wxString
& s 
) 
8547         m_table
->SetRowLabelValue( row
, s 
); 
8548         if ( !GetBatchCount() ) 
8550             wxRect rect 
= CellToRect( row
, 0); 
8551             if ( rect
.height 
> 0 ) 
8553                 CalcScrolledPosition(0, rect
.y
, &rect
.x
, &rect
.y
); 
8555                 rect
.width 
= m_rowLabelWidth
; 
8556                 m_rowLabelWin
->Refresh( TRUE
, &rect 
); 
8562 void wxGrid::SetColLabelValue( int col
, const wxString
& s 
) 
8566         m_table
->SetColLabelValue( col
, s 
); 
8567         if ( !GetBatchCount() ) 
8569             wxRect rect 
= CellToRect( 0, col 
); 
8570             if ( rect
.width 
> 0 ) 
8572                 CalcScrolledPosition(rect
.x
, 0, &rect
.x
, &rect
.y
); 
8574                 rect
.height 
= m_colLabelHeight
; 
8575                 m_colLabelWin
->Refresh( TRUE
, &rect 
); 
8581 void wxGrid::SetGridLineColour( const wxColour
& colour 
) 
8583     if ( m_gridLineColour 
!= colour 
) 
8585         m_gridLineColour 
= colour
; 
8587         wxClientDC 
dc( m_gridWin 
); 
8589         DrawAllGridLines( dc
, wxRegion() ); 
8594 void wxGrid::SetCellHighlightColour( const wxColour
& colour 
) 
8596     if ( m_cellHighlightColour 
!= colour 
) 
8598         m_cellHighlightColour 
= colour
; 
8600         wxClientDC 
dc( m_gridWin 
); 
8602         wxGridCellAttr
* attr 
= GetCellAttr(m_currentCellCoords
); 
8603         DrawCellHighlight(dc
, attr
); 
8608 void wxGrid::SetCellHighlightPenWidth(int width
) 
8610     if (m_cellHighlightPenWidth 
!= width
) { 
8611         m_cellHighlightPenWidth 
= width
; 
8613         // Just redrawing the cell highlight is not enough since that won't 
8614         // make any visible change if the the thickness is getting smaller. 
8615         int row 
= m_currentCellCoords
.GetRow(); 
8616         int col 
= m_currentCellCoords
.GetCol(); 
8617         if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
8619         wxRect rect 
= CellToRect(row
, col
); 
8620         m_gridWin
->Refresh(TRUE
, &rect
); 
8624 void wxGrid::SetCellHighlightROPenWidth(int width
) 
8626     if (m_cellHighlightROPenWidth 
!= width
) { 
8627         m_cellHighlightROPenWidth 
= width
; 
8629         // Just redrawing the cell highlight is not enough since that won't 
8630         // make any visible change if the the thickness is getting smaller. 
8631         int row 
= m_currentCellCoords
.GetRow(); 
8632         int col 
= m_currentCellCoords
.GetCol(); 
8633         if ( GetColWidth(col
) <= 0 || GetRowHeight(row
) <= 0 ) 
8635         wxRect rect 
= CellToRect(row
, col
); 
8636         m_gridWin
->Refresh(TRUE
, &rect
); 
8640 void wxGrid::EnableGridLines( bool enable 
) 
8642     if ( enable 
!= m_gridLinesEnabled 
) 
8644         m_gridLinesEnabled 
= enable
; 
8646         if ( !GetBatchCount() ) 
8650                 wxClientDC 
dc( m_gridWin 
); 
8652                 DrawAllGridLines( dc
, wxRegion() ); 
8656                 m_gridWin
->Refresh(); 
8663 int wxGrid::GetDefaultRowSize() 
8665     return m_defaultRowHeight
; 
8668 int wxGrid::GetRowSize( int row 
) 
8670     wxCHECK_MSG( row 
>= 0 && row 
< m_numRows
, 0, _T("invalid row index") ); 
8672     return GetRowHeight(row
); 
8675 int wxGrid::GetDefaultColSize() 
8677     return m_defaultColWidth
; 
8680 int wxGrid::GetColSize( int col 
) 
8682     wxCHECK_MSG( col 
>= 0 && col 
< m_numCols
, 0, _T("invalid column index") ); 
8684     return GetColWidth(col
); 
8687 // ============================================================================ 
8688 // access to the grid attributes: each of them has a default value in the grid 
8689 // itself and may be overidden on a per-cell basis 
8690 // ============================================================================ 
8692 // ---------------------------------------------------------------------------- 
8693 // setting default attributes 
8694 // ---------------------------------------------------------------------------- 
8696 void wxGrid::SetDefaultCellBackgroundColour( const wxColour
& col 
) 
8698     m_defaultCellAttr
->SetBackgroundColour(col
); 
8700     m_gridWin
->SetBackgroundColour(col
); 
8704 void wxGrid::SetDefaultCellTextColour( const wxColour
& col 
) 
8706     m_defaultCellAttr
->SetTextColour(col
); 
8709 void wxGrid::SetDefaultCellAlignment( int horiz
, int vert 
) 
8711     m_defaultCellAttr
->SetAlignment(horiz
, vert
); 
8714 void wxGrid::SetDefaultCellOverflow( bool allow 
) 
8716     m_defaultCellAttr
->SetOverflow(allow
); 
8719 void wxGrid::SetDefaultCellFont( const wxFont
& font 
) 
8721     m_defaultCellAttr
->SetFont(font
); 
8724 void wxGrid::SetDefaultRenderer(wxGridCellRenderer 
*renderer
) 
8726     m_defaultCellAttr
->SetRenderer(renderer
); 
8729 void wxGrid::SetDefaultEditor(wxGridCellEditor 
*editor
) 
8731     m_defaultCellAttr
->SetEditor(editor
); 
8734 // ---------------------------------------------------------------------------- 
8735 // access to the default attrbiutes 
8736 // ---------------------------------------------------------------------------- 
8738 wxColour 
wxGrid::GetDefaultCellBackgroundColour() 
8740     return m_defaultCellAttr
->GetBackgroundColour(); 
8743 wxColour 
wxGrid::GetDefaultCellTextColour() 
8745     return m_defaultCellAttr
->GetTextColour(); 
8748 wxFont 
wxGrid::GetDefaultCellFont() 
8750     return m_defaultCellAttr
->GetFont(); 
8753 void wxGrid::GetDefaultCellAlignment( int *horiz
, int *vert 
) 
8755     m_defaultCellAttr
->GetAlignment(horiz
, vert
); 
8758 bool wxGrid::GetDefaultCellOverflow() 
8760     return m_defaultCellAttr
->GetOverflow(); 
8763 wxGridCellRenderer 
*wxGrid::GetDefaultRenderer() const 
8765     return m_defaultCellAttr
->GetRenderer(NULL
, 0, 0); 
8768 wxGridCellEditor 
*wxGrid::GetDefaultEditor() const 
8770     return m_defaultCellAttr
->GetEditor(NULL
,0,0); 
8773 // ---------------------------------------------------------------------------- 
8774 // access to cell attributes 
8775 // ---------------------------------------------------------------------------- 
8777 wxColour 
wxGrid::GetCellBackgroundColour(int row
, int col
) 
8779     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8780     wxColour colour 
= attr
->GetBackgroundColour(); 
8785 wxColour 
wxGrid::GetCellTextColour( int row
, int col 
) 
8787     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8788     wxColour colour 
= attr
->GetTextColour(); 
8793 wxFont 
wxGrid::GetCellFont( int row
, int col 
) 
8795     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8796     wxFont font 
= attr
->GetFont(); 
8801 void wxGrid::GetCellAlignment( int row
, int col
, int *horiz
, int *vert 
) 
8803     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8804     attr
->GetAlignment(horiz
, vert
); 
8808 bool wxGrid::GetCellOverflow( int row
, int col 
) 
8810     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8811     bool allow 
= attr
->GetOverflow(); 
8816 void wxGrid::GetCellSize( int row
, int col
, int *num_rows
, int *num_cols 
) 
8818     wxGridCellAttr 
*attr 
= GetCellAttr(row
, col
); 
8819     attr
->GetSize( num_rows
, num_cols 
); 
8823 wxGridCellRenderer
* wxGrid::GetCellRenderer(int row
, int col
) 
8825     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
8826     wxGridCellRenderer
* renderer 
= attr
->GetRenderer(this, row
, col
); 
8832 wxGridCellEditor
* wxGrid::GetCellEditor(int row
, int col
) 
8834     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
8835     wxGridCellEditor
* editor 
= attr
->GetEditor(this, row
, col
); 
8841 bool wxGrid::IsReadOnly(int row
, int col
) const 
8843     wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
8844     bool isReadOnly 
= attr
->IsReadOnly(); 
8849 // ---------------------------------------------------------------------------- 
8850 // attribute support: cache, automatic provider creation, ... 
8851 // ---------------------------------------------------------------------------- 
8853 bool wxGrid::CanHaveAttributes() 
8860     return m_table
->CanHaveAttributes(); 
8863 void wxGrid::ClearAttrCache() 
8865     if ( m_attrCache
.row 
!= -1 ) 
8867         wxSafeDecRef(m_attrCache
.attr
); 
8868         m_attrCache
.attr 
= NULL
; 
8869         m_attrCache
.row 
= -1; 
8873 void wxGrid::CacheAttr(int row
, int col
, wxGridCellAttr 
*attr
) const 
8877         wxGrid 
*self 
= (wxGrid 
*)this;  // const_cast 
8879         self
->ClearAttrCache(); 
8880         self
->m_attrCache
.row 
= row
; 
8881         self
->m_attrCache
.col 
= col
; 
8882         self
->m_attrCache
.attr 
= attr
; 
8887 bool wxGrid::LookupAttr(int row
, int col
, wxGridCellAttr 
**attr
) const 
8889     if ( row 
== m_attrCache
.row 
&& col 
== m_attrCache
.col 
) 
8891         *attr 
= m_attrCache
.attr
; 
8892         wxSafeIncRef(m_attrCache
.attr
); 
8894 #ifdef DEBUG_ATTR_CACHE 
8895         gs_nAttrCacheHits
++; 
8902 #ifdef DEBUG_ATTR_CACHE 
8903         gs_nAttrCacheMisses
++; 
8909 wxGridCellAttr 
*wxGrid::GetCellAttr(int row
, int col
) const 
8911     wxGridCellAttr 
*attr 
= NULL
; 
8912     // Additional test to avoid looking at the cache e.g. for 
8913     // wxNoCellCoords, as this will confuse memory management. 
8916         if ( !LookupAttr(row
, col
, &attr
) ) 
8918             attr 
= m_table 
? m_table
->GetAttr(row
, col 
, wxGridCellAttr::Any
) 
8919                            : (wxGridCellAttr 
*)NULL
; 
8920             CacheAttr(row
, col
, attr
); 
8925         attr
->SetDefAttr(m_defaultCellAttr
); 
8929         attr 
= m_defaultCellAttr
; 
8936 wxGridCellAttr 
*wxGrid::GetOrCreateCellAttr(int row
, int col
) const 
8938     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
8940     wxCHECK_MSG( m_table
, attr
, 
8941                   _T("we may only be called if CanHaveAttributes() returned TRUE and then m_table should be !NULL") ); 
8943     attr 
= m_table
->GetAttr(row
, col
, wxGridCellAttr::Cell
); 
8946         attr 
= new wxGridCellAttr(m_defaultCellAttr
); 
8948         // artificially inc the ref count to match DecRef() in caller 
8950         m_table
->SetAttr(attr
, row
, col
); 
8956 // ---------------------------------------------------------------------------- 
8957 // setting column attributes (wrappers around SetColAttr) 
8958 // ---------------------------------------------------------------------------- 
8960 void wxGrid::SetColFormatBool(int col
) 
8962     SetColFormatCustom(col
, wxGRID_VALUE_BOOL
); 
8965 void wxGrid::SetColFormatNumber(int col
) 
8967     SetColFormatCustom(col
, wxGRID_VALUE_NUMBER
); 
8970 void wxGrid::SetColFormatFloat(int col
, int width
, int precision
) 
8972     wxString typeName 
= wxGRID_VALUE_FLOAT
; 
8973     if ( (width 
!= -1) || (precision 
!= -1) ) 
8975         typeName 
<< _T(':') << width 
<< _T(',') << precision
; 
8978     SetColFormatCustom(col
, typeName
); 
8981 void wxGrid::SetColFormatCustom(int col
, const wxString
& typeName
) 
8983     wxGridCellAttr 
*attr 
= (wxGridCellAttr 
*)NULL
; 
8985     attr 
= m_table
->GetAttr(-1, col
, wxGridCellAttr::Col 
); 
8987         attr 
= new wxGridCellAttr
; 
8988     wxGridCellRenderer 
*renderer 
= GetDefaultRendererForType(typeName
); 
8989     attr
->SetRenderer(renderer
); 
8991     SetColAttr(col
, attr
); 
8995 // ---------------------------------------------------------------------------- 
8996 // setting cell attributes: this is forwarded to the table 
8997 // ---------------------------------------------------------------------------- 
8999 void wxGrid::SetAttr(int row
, int col
, wxGridCellAttr 
*attr
) 
9001     if ( CanHaveAttributes() ) 
9003         m_table
->SetAttr(attr
, row
, col
); 
9012 void wxGrid::SetRowAttr(int row
, wxGridCellAttr 
*attr
) 
9014     if ( CanHaveAttributes() ) 
9016         m_table
->SetRowAttr(attr
, row
); 
9025 void wxGrid::SetColAttr(int col
, wxGridCellAttr 
*attr
) 
9027     if ( CanHaveAttributes() ) 
9029         m_table
->SetColAttr(attr
, col
); 
9038 void wxGrid::SetCellBackgroundColour( int row
, int col
, const wxColour
& colour 
) 
9040     if ( CanHaveAttributes() ) 
9042         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9043         attr
->SetBackgroundColour(colour
); 
9048 void wxGrid::SetCellTextColour( int row
, int col
, const wxColour
& colour 
) 
9050     if ( CanHaveAttributes() ) 
9052         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9053         attr
->SetTextColour(colour
); 
9058 void wxGrid::SetCellFont( int row
, int col
, const wxFont
& font 
) 
9060     if ( CanHaveAttributes() ) 
9062         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9063         attr
->SetFont(font
); 
9068 void wxGrid::SetCellAlignment( int row
, int col
, int horiz
, int vert 
) 
9070     if ( CanHaveAttributes() ) 
9072         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9073         attr
->SetAlignment(horiz
, vert
); 
9078 void wxGrid::SetCellOverflow( int row
, int col
, bool allow 
) 
9080     if ( CanHaveAttributes() ) 
9082         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9083         attr
->SetOverflow(allow
); 
9088 void wxGrid::SetCellSize( int row
, int col
, int num_rows
, int num_cols 
) 
9090     if ( CanHaveAttributes() ) 
9092         int cell_rows
, cell_cols
; 
9094         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9095         attr
->GetSize(&cell_rows
, &cell_cols
); 
9096         attr
->SetSize(num_rows
, num_cols
); 
9099         // Cannot set the size of a cell to 0 or negative values 
9100         // While it is perfectly legal to do that, this function cannot 
9101         // handle all the possibilies, do it by hand by getting the CellAttr. 
9102         // You can only set the size of a cell to 1,1 or greater with this fn 
9103         wxASSERT_MSG( !((cell_rows 
< 1) || (cell_cols 
< 1)), 
9104                       wxT("wxGrid::SetCellSize setting cell size that is already part of another cell")); 
9105         wxASSERT_MSG( !((num_rows 
< 1) || (num_cols 
< 1)), 
9106                       wxT("wxGrid::SetCellSize setting cell size to < 1")); 
9108         // if this was already a multicell then "turn off" the other cells first 
9109         if ((cell_rows 
> 1) || (cell_rows 
> 1)) 
9112             for (j
=row
; j
<row
+cell_rows
; j
++) 
9114                 for (i
=col
; i
<col
+cell_cols
; i
++) 
9116                     if ((i 
!= col
) || (j 
!= row
)) 
9118                         wxGridCellAttr 
*attr_stub 
= GetOrCreateCellAttr(j
, i
); 
9119                         attr_stub
->SetSize( 1, 1 ); 
9120                         attr_stub
->DecRef(); 
9126         // mark the cells that will be covered by this cell to 
9127         // negative or zero values to point back at this cell 
9128         if (((num_rows 
> 1) || (num_cols 
> 1)) && (num_rows 
>= 1) && (num_cols 
>= 1)) 
9131             for (j
=row
; j
<row
+num_rows
; j
++) 
9133                 for (i
=col
; i
<col
+num_cols
; i
++) 
9135                     if ((i 
!= col
) || (j 
!= row
)) 
9137                         wxGridCellAttr 
*attr_stub 
= GetOrCreateCellAttr(j
, i
); 
9138                         attr_stub
->SetSize( row
-j
, col
-i 
); 
9139                         attr_stub
->DecRef(); 
9147 void wxGrid::SetCellRenderer(int row
, int col
, wxGridCellRenderer 
*renderer
) 
9149     if ( CanHaveAttributes() ) 
9151         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9152         attr
->SetRenderer(renderer
); 
9157 void wxGrid::SetCellEditor(int row
, int col
, wxGridCellEditor
* editor
) 
9159     if ( CanHaveAttributes() ) 
9161         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9162         attr
->SetEditor(editor
); 
9167 void wxGrid::SetReadOnly(int row
, int col
, bool isReadOnly
) 
9169     if ( CanHaveAttributes() ) 
9171         wxGridCellAttr 
*attr 
= GetOrCreateCellAttr(row
, col
); 
9172         attr
->SetReadOnly(isReadOnly
); 
9177 // ---------------------------------------------------------------------------- 
9178 // Data type registration 
9179 // ---------------------------------------------------------------------------- 
9181 void wxGrid::RegisterDataType(const wxString
& typeName
, 
9182                               wxGridCellRenderer
* renderer
, 
9183                               wxGridCellEditor
* editor
) 
9185     m_typeRegistry
->RegisterDataType(typeName
, renderer
, editor
); 
9189 wxGridCellEditor
* wxGrid::GetDefaultEditorForCell(int row
, int col
) const 
9191     wxString typeName 
= m_table
->GetTypeName(row
, col
); 
9192     return GetDefaultEditorForType(typeName
); 
9195 wxGridCellRenderer
* wxGrid::GetDefaultRendererForCell(int row
, int col
) const 
9197     wxString typeName 
= m_table
->GetTypeName(row
, col
); 
9198     return GetDefaultRendererForType(typeName
); 
9202 wxGrid::GetDefaultEditorForType(const wxString
& typeName
) const 
9204     int index 
= m_typeRegistry
->FindOrCloneDataType(typeName
); 
9205     if ( index 
== wxNOT_FOUND 
) 
9207         wxFAIL_MSG(wxT("Unknown data type name")); 
9212     return m_typeRegistry
->GetEditor(index
); 
9216 wxGrid::GetDefaultRendererForType(const wxString
& typeName
) const 
9218     int index 
= m_typeRegistry
->FindOrCloneDataType(typeName
); 
9219     if ( index 
== wxNOT_FOUND 
) 
9221         wxFAIL_MSG(wxT("Unknown data type name")); 
9226     return m_typeRegistry
->GetRenderer(index
); 
9230 // ---------------------------------------------------------------------------- 
9232 // ---------------------------------------------------------------------------- 
9234 void wxGrid::EnableDragRowSize( bool enable 
) 
9236     m_canDragRowSize 
= enable
; 
9240 void wxGrid::EnableDragColSize( bool enable 
) 
9242     m_canDragColSize 
= enable
; 
9245 void wxGrid::EnableDragGridSize( bool enable 
) 
9247     m_canDragGridSize 
= enable
; 
9251 void wxGrid::SetDefaultRowSize( int height
, bool resizeExistingRows 
) 
9253     m_defaultRowHeight 
= wxMax( height
, m_minAcceptableRowHeight 
); 
9255     if ( resizeExistingRows 
) 
9257         // since we are resizing all rows to the default row size, 
9258         // we can simply clear the row heights and row bottoms 
9259         // arrays (which also allows us to take advantage of 
9260         // some speed optimisations) 
9261         m_rowHeights
.Empty(); 
9262         m_rowBottoms
.Empty(); 
9263         if ( !GetBatchCount() ) 
9268 void wxGrid::SetRowSize( int row
, int height 
) 
9270     wxCHECK_RET( row 
>= 0 && row 
< m_numRows
, _T("invalid row index") ); 
9272     // See comment in SetColSize 
9273     if ( height 
< GetRowMinimalAcceptableHeight()) { return; } 
9275     if ( m_rowHeights
.IsEmpty() ) 
9277         // need to really create the array 
9281     int h 
= wxMax( 0, height 
); 
9282     int diff 
= h 
- m_rowHeights
[row
]; 
9284     m_rowHeights
[row
] = h
; 
9286     for ( i 
= row
;  i 
< m_numRows
;  i
++ ) 
9288         m_rowBottoms
[i
] += diff
; 
9290     if ( !GetBatchCount() ) 
9294 void wxGrid::SetDefaultColSize( int width
, bool resizeExistingCols 
) 
9296     m_defaultColWidth 
= wxMax( width
, m_minAcceptableColWidth 
); 
9298     if ( resizeExistingCols 
) 
9300         // since we are resizing all columns to the default column size, 
9301         // we can simply clear the col widths and col rights 
9302         // arrays (which also allows us to take advantage of 
9303         // some speed optimisations) 
9304         m_colWidths
.Empty(); 
9305         m_colRights
.Empty(); 
9306         if ( !GetBatchCount() ) 
9311 void wxGrid::SetColSize( int col
, int width 
) 
9313     wxCHECK_RET( col 
>= 0 && col 
< m_numCols
, _T("invalid column index") ); 
9315     // should we check that it's bigger than GetColMinimalWidth(col) here? 
9317     // No, because it is reasonable to assume the library user know's 
9318     // what he is doing. However whe should test against the weaker  
9319     // constariant of minimalAcceptableWidth, as this breaks rendering 
9321     // This test then fixes sf.net bug #645734 
9323     if ( width 
< GetColMinimalAcceptableWidth()) { return; } 
9325     if ( m_colWidths
.IsEmpty() ) 
9327         // need to really create the array 
9331     // if < 0 calc new width from label 
9335       wxArrayString lines
; 
9336       wxClientDC 
dc(m_colLabelWin
); 
9337       dc
.SetFont(GetLabelFont()); 
9338       StringToLines(GetColLabelValue(col
), lines
); 
9339       GetTextBoxSize(dc
, lines
, &w
, &h
); 
9342     int w 
= wxMax( 0, width 
); 
9343     int diff 
= w 
- m_colWidths
[col
]; 
9344     m_colWidths
[col
] = w
; 
9347     for ( i 
= col
;  i 
< m_numCols
;  i
++ ) 
9349         m_colRights
[i
] += diff
; 
9351     if ( !GetBatchCount() ) 
9356 void wxGrid::SetColMinimalWidth( int col
, int width 
) 
9358     if (width 
> GetColMinimalAcceptableWidth()) { 
9359         m_colMinWidths
.Put(col
, width
); 
9363 void wxGrid::SetRowMinimalHeight( int row
, int width 
) 
9365     if (width 
> GetRowMinimalAcceptableHeight()) { 
9366        m_rowMinHeights
.Put(row
, width
); 
9370 int wxGrid::GetColMinimalWidth(int col
) const 
9372     long value 
= m_colMinWidths
.Get(col
); 
9373     return value 
!= wxNOT_FOUND 
? (int)value 
: m_minAcceptableColWidth
; 
9376 int wxGrid::GetRowMinimalHeight(int row
) const 
9378     long value 
= m_rowMinHeights
.Get(row
); 
9379     return value 
!= wxNOT_FOUND 
? (int)value 
: m_minAcceptableRowHeight
; 
9382 void wxGrid::SetColMinimalAcceptableWidth( int width 
) 
9386     m_minAcceptableColWidth 
= width
; 
9389 void wxGrid::SetRowMinimalAcceptableHeight( int height 
) 
9393     m_minAcceptableRowHeight 
= height
; 
9396 int  wxGrid::GetColMinimalAcceptableWidth() const 
9398     return m_minAcceptableColWidth
; 
9401 int  wxGrid::GetRowMinimalAcceptableHeight() const 
9403     return m_minAcceptableRowHeight
; 
9406 // ---------------------------------------------------------------------------- 
9408 // ---------------------------------------------------------------------------- 
9410 void wxGrid::AutoSizeColOrRow( int colOrRow
, bool setAsMin
, bool column 
) 
9412     wxClientDC 
dc(m_gridWin
); 
9414     // init both of them to avoid compiler warnings, even if weo nly need one 
9422     wxCoord extent
, extentMax 
= 0; 
9423     int max 
= column 
? m_numRows 
: m_numCols
; 
9424     for ( int rowOrCol 
= 0; rowOrCol 
< max
; rowOrCol
++ ) 
9431         wxGridCellAttr
* attr 
= GetCellAttr(row
, col
); 
9432         wxGridCellRenderer
* renderer 
= attr
->GetRenderer(this, row
, col
); 
9435             wxSize size 
= renderer
->GetBestSize(*this, *attr
, dc
, row
, col
); 
9436             extent 
= column 
? size
.x 
: size
.y
; 
9437             if ( extent 
> extentMax 
) 
9448     // now also compare with the column label extent 
9450     dc
.SetFont( GetLabelFont() ); 
9454         dc
.GetTextExtent( GetColLabelValue(col
), &w
, &h 
); 
9455         if( GetColLabelTextOrientation() == wxVERTICAL 
) 
9459         dc
.GetTextExtent( GetRowLabelValue(row
), &w
, &h 
); 
9461     extent 
= column 
? w 
: h
; 
9462     if ( extent 
> extentMax 
) 
9469         // empty column - give default extent (notice that if extentMax is less 
9470         // than default extent but != 0, it's ok) 
9471         extentMax 
= column 
? m_defaultColWidth 
: m_defaultRowHeight
; 
9477             // leave some space around text 
9488         SetColSize(col
, extentMax
); 
9489         if ( !GetBatchCount() ) 
9492             m_gridWin
->GetClientSize( &cw
, &ch 
); 
9493             wxRect 
rect ( CellToRect( 0, col 
) ); 
9495             CalcScrolledPosition(rect
.x
, 0, &rect
.x
, &dummy
); 
9496             rect
.width 
= cw 
- rect
.x
; 
9497             rect
.height 
= m_colLabelHeight
; 
9498             m_colLabelWin
->Refresh( TRUE
, &rect 
); 
9503         SetRowSize(row
, extentMax
); 
9504         if ( !GetBatchCount() ) 
9507             m_gridWin
->GetClientSize( &cw
, &ch 
); 
9508             wxRect 
rect ( CellToRect( row
, 0 ) ); 
9510             CalcScrolledPosition(0, rect
.y
, &dummy
, &rect
.y
); 
9511             rect
.width 
= m_rowLabelWidth
; 
9512             rect
.height 
= ch 
- rect
.y
; 
9513             m_rowLabelWin
->Refresh( TRUE
, &rect 
); 
9519             SetColMinimalWidth(col
, extentMax
); 
9521             SetRowMinimalHeight(row
, extentMax
); 
9525 int wxGrid::SetOrCalcColumnSizes(bool calcOnly
, bool setAsMin
) 
9527     int width 
= m_rowLabelWidth
; 
9532     for ( int col 
= 0; col 
< m_numCols
; col
++ ) 
9536             AutoSizeColumn(col
, setAsMin
); 
9539         width 
+= GetColWidth(col
); 
9548 int wxGrid::SetOrCalcRowSizes(bool calcOnly
, bool setAsMin
) 
9550     int height 
= m_colLabelHeight
; 
9555     for ( int row 
= 0; row 
< m_numRows
; row
++ ) 
9559             AutoSizeRow(row
, setAsMin
); 
9562         height 
+= GetRowHeight(row
); 
9571 void wxGrid::AutoSize() 
9575     wxSize 
size(SetOrCalcColumnSizes(FALSE
), SetOrCalcRowSizes(FALSE
)); 
9577     // round up the size to a multiple of scroll step - this ensures that we 
9578     // won't get the scrollbars if we're sized exactly to this width 
9579     // CalcDimension adds m_extraWidth + 1 etc. to calculate the necessary 
9581     wxSize 
sizeFit(GetScrollX(size
.x 
+ m_extraWidth 
+ 1) * GRID_SCROLL_LINE_X
, 
9582                    GetScrollY(size
.y 
+ m_extraHeight 
+ 1) * GRID_SCROLL_LINE_Y
); 
9584     // distribute the extra space between the columns/rows to avoid having 
9585     // extra white space 
9587     // Remove the extra m_extraWidth + 1 added above 
9588     wxCoord diff 
= sizeFit
.x 
- size
.x 
+ (m_extraWidth 
+ 1); 
9589     if ( diff 
&& m_numCols 
) 
9591         // try to resize the columns uniformly 
9592         wxCoord diffPerCol 
= diff 
/ m_numCols
; 
9595             for ( int col 
= 0; col 
< m_numCols
; col
++ ) 
9597                 SetColSize(col
, GetColWidth(col
) + diffPerCol
); 
9601         // add remaining amount to the last columns 
9602         diff 
-= diffPerCol 
* m_numCols
; 
9605             for ( int col 
= m_numCols 
- 1; col 
>= m_numCols 
- diff
; col
-- ) 
9607                 SetColSize(col
, GetColWidth(col
) + 1); 
9613     diff 
= sizeFit
.y 
- size
.y 
- (m_extraHeight 
+ 1); 
9614     if ( diff 
&& m_numRows 
) 
9616         // try to resize the columns uniformly 
9617         wxCoord diffPerRow 
= diff 
/ m_numRows
; 
9620             for ( int row 
= 0; row 
< m_numRows
; row
++ ) 
9622                 SetRowSize(row
, GetRowHeight(row
) + diffPerRow
); 
9626         // add remaining amount to the last rows 
9627         diff 
-= diffPerRow 
* m_numRows
; 
9630             for ( int row 
= m_numRows 
- 1; row 
>= m_numRows 
- diff
; row
-- ) 
9632                 SetRowSize(row
, GetRowHeight(row
) + 1); 
9639     SetClientSize(sizeFit
); 
9642 void wxGrid::AutoSizeRowLabelSize( int row 
) 
9644     wxArrayString lines
; 
9647     // Hide the edit control, so it 
9648     // won't interfer with drag-shrinking. 
9649     if( IsCellEditControlShown() ) 
9651         HideCellEditControl(); 
9652         SaveEditControlValue(); 
9655     // autosize row height depending on label text 
9656     StringToLines( GetRowLabelValue( row 
), lines 
); 
9657     wxClientDC 
dc( m_rowLabelWin 
); 
9658     GetTextBoxSize( dc
, lines
, &w
, &h
); 
9659     if( h 
< m_defaultRowHeight 
) 
9660         h 
= m_defaultRowHeight
; 
9665 void wxGrid::AutoSizeColLabelSize( int col 
) 
9667     wxArrayString lines
; 
9670     // Hide the edit control, so it 
9671     // won't interfer with drag-shrinking. 
9672     if( IsCellEditControlShown() ) 
9674         HideCellEditControl(); 
9675         SaveEditControlValue(); 
9678     // autosize column width depending on label text 
9679     StringToLines( GetColLabelValue( col 
), lines 
); 
9680     wxClientDC 
dc( m_colLabelWin 
); 
9681     if( GetColLabelTextOrientation() == wxHORIZONTAL 
) 
9682         GetTextBoxSize( dc
, lines
, &w
, &h
); 
9684         GetTextBoxSize( dc
, lines
, &h
, &w
); 
9685     if( w 
< m_defaultColWidth 
) 
9686         w 
= m_defaultColWidth
; 
9691 wxSize 
wxGrid::DoGetBestSize() const 
9693     // don't set sizes, only calculate them 
9694     wxGrid 
*self 
= (wxGrid 
*)this;  // const_cast 
9697     width 
= self
->SetOrCalcColumnSizes(TRUE
); 
9698     height 
= self
->SetOrCalcRowSizes(TRUE
); 
9700     int maxwidth
, maxheight
; 
9701     wxDisplaySize( & maxwidth
, & maxheight 
); 
9703     if ( width 
> maxwidth 
) width 
= maxwidth
; 
9704     if ( height 
> maxheight 
) height 
= maxheight
; 
9706     return wxSize( width
, height 
); 
9715 wxPen
& wxGrid::GetDividerPen() const 
9720 // ---------------------------------------------------------------------------- 
9721 // cell value accessor functions 
9722 // ---------------------------------------------------------------------------- 
9724 void wxGrid::SetCellValue( int row
, int col
, const wxString
& s 
) 
9728         m_table
->SetValue( row
, col
, s 
); 
9729         if ( !GetBatchCount() ) 
9732             wxRect 
rect( CellToRect( row
, col 
) ); 
9734             rect
.width 
= m_gridWin
->GetClientSize().GetWidth(); 
9735             CalcScrolledPosition(0, rect
.y
, &dummy
, &rect
.y
); 
9736             m_gridWin
->Refresh( FALSE
, &rect 
); 
9739         if ( m_currentCellCoords
.GetRow() == row 
&& 
9740              m_currentCellCoords
.GetCol() == col 
&& 
9741              IsCellEditControlShown()) 
9742              // Note: If we are using IsCellEditControlEnabled, 
9743              // this interacts badly with calling SetCellValue from 
9744              // an EVT_GRID_CELL_CHANGE handler. 
9746             HideCellEditControl(); 
9747             ShowCellEditControl(); // will reread data from table 
9754 // ------ Block, row and col selection 
9757 void wxGrid::SelectRow( int row
, bool addToSelected 
) 
9759     if ( IsSelection() && !addToSelected 
) 
9763         m_selection
->SelectRow( row
, FALSE
, addToSelected 
); 
9767 void wxGrid::SelectCol( int col
, bool addToSelected 
) 
9769     if ( IsSelection() && !addToSelected 
) 
9773         m_selection
->SelectCol( col
, FALSE
, addToSelected 
); 
9777 void wxGrid::SelectBlock( int topRow
, int leftCol
, int bottomRow
, int rightCol
, 
9778                           bool addToSelected 
) 
9780     if ( IsSelection() && !addToSelected 
) 
9784         m_selection
->SelectBlock( topRow
, leftCol
, bottomRow
, rightCol
, 
9785                                   FALSE
, addToSelected 
); 
9789 void wxGrid::SelectAll() 
9791     if ( m_numRows 
> 0 && m_numCols 
> 0 ) 
9794             m_selection
->SelectBlock( 0, 0, m_numRows
-1, m_numCols
-1 ); 
9799 // ------ Cell, row and col deselection 
9802 void wxGrid::DeselectRow( int row 
) 
9807     if ( m_selection
->GetSelectionMode() == wxGrid::wxGridSelectRows 
) 
9809         if ( m_selection
->IsInSelection(row
, 0 ) ) 
9810             m_selection
->ToggleCellSelection( row
, 0); 
9814         int nCols 
= GetNumberCols(); 
9815         for ( int i 
= 0; i 
< nCols 
; i
++ ) 
9817             if ( m_selection
->IsInSelection(row
, i 
) ) 
9818                 m_selection
->ToggleCellSelection( row
, i
); 
9823 void wxGrid::DeselectCol( int col 
) 
9828     if ( m_selection
->GetSelectionMode() == wxGrid::wxGridSelectColumns 
) 
9830         if ( m_selection
->IsInSelection(0, col 
) ) 
9831             m_selection
->ToggleCellSelection( 0, col
); 
9835         int nRows 
= GetNumberRows(); 
9836         for ( int i 
= 0; i 
< nRows 
; i
++ ) 
9838             if ( m_selection
->IsInSelection(i
, col 
) ) 
9839                 m_selection
->ToggleCellSelection(i
, col
); 
9844 void wxGrid::DeselectCell( int row
, int col 
) 
9846     if ( m_selection 
&& m_selection
->IsInSelection(row
, col
) ) 
9847         m_selection
->ToggleCellSelection(row
, col
); 
9850 bool wxGrid::IsSelection() 
9852     return ( m_selection 
&& (m_selection
->IsSelection() || 
9853              ( m_selectingTopLeft 
!= wxGridNoCellCoords 
&& 
9854                m_selectingBottomRight 
!= wxGridNoCellCoords
) ) ); 
9857 bool wxGrid::IsInSelection( int row
, int col 
) const 
9859     return ( m_selection 
&& (m_selection
->IsInSelection( row
, col 
) || 
9860              ( row 
>= m_selectingTopLeft
.GetRow() && 
9861                col 
>= m_selectingTopLeft
.GetCol() && 
9862                row 
<= m_selectingBottomRight
.GetRow() && 
9863                col 
<= m_selectingBottomRight
.GetCol() )) ); 
9866 wxGridCellCoordsArray 
wxGrid::GetSelectedCells() const 
9868     if (!m_selection
) { wxGridCellCoordsArray a
; return a
; } 
9869     return m_selection
->m_cellSelection
; 
9871 wxGridCellCoordsArray 
wxGrid::GetSelectionBlockTopLeft() const 
9873     if (!m_selection
) { wxGridCellCoordsArray a
; return a
; } 
9874     return m_selection
->m_blockSelectionTopLeft
; 
9876 wxGridCellCoordsArray 
wxGrid::GetSelectionBlockBottomRight() const 
9878     if (!m_selection
) { wxGridCellCoordsArray a
; return a
; } 
9879     return m_selection
->m_blockSelectionBottomRight
; 
9881 wxArrayInt 
wxGrid::GetSelectedRows() const 
9883     if (!m_selection
) { wxArrayInt a
; return a
; } 
9884     return m_selection
->m_rowSelection
; 
9886 wxArrayInt 
wxGrid::GetSelectedCols() const 
9888     if (!m_selection
) { wxArrayInt a
; return a
; } 
9889     return m_selection
->m_colSelection
; 
9893 void wxGrid::ClearSelection() 
9895     m_selectingTopLeft 
= wxGridNoCellCoords
; 
9896     m_selectingBottomRight 
= wxGridNoCellCoords
; 
9898         m_selection
->ClearSelection(); 
9902 // This function returns the rectangle that encloses the given block 
9903 // in device coords clipped to the client size of the grid window. 
9905 wxRect 
wxGrid::BlockToDeviceRect( const wxGridCellCoords 
&topLeft
, 
9906                                   const wxGridCellCoords 
&bottomRight 
) 
9908     wxRect 
rect( wxGridNoCellRect 
); 
9911     cellRect 
= CellToRect( topLeft 
); 
9912     if ( cellRect 
!= wxGridNoCellRect 
) 
9918         rect 
= wxRect( 0, 0, 0, 0 ); 
9921     cellRect 
= CellToRect( bottomRight 
); 
9922     if ( cellRect 
!= wxGridNoCellRect 
) 
9928         return wxGridNoCellRect
; 
9932     int left 
= rect
.GetLeft(); 
9933     int top 
= rect
.GetTop(); 
9934     int right 
= rect
.GetRight(); 
9935     int bottom 
= rect
.GetBottom(); 
9937     int leftCol 
= topLeft
.GetCol(); 
9938     int topRow 
= topLeft
.GetRow(); 
9939     int rightCol 
= bottomRight
.GetCol(); 
9940     int bottomRow 
= bottomRight
.GetRow(); 
9963     for ( j 
= topRow
; j 
<= bottomRow
; j
++ ) 
9965         for ( i 
= leftCol
; i 
<= rightCol
; i
++ ) 
9967             if ((j
==topRow
) || (j
==bottomRow
) || (i
==leftCol
) || (i
==rightCol
)) 
9969                 cellRect 
= CellToRect( j
, i 
); 
9971                 if (cellRect
.x 
< left
) 
9973                 if (cellRect
.y 
< top
) 
9975                 if (cellRect
.x 
+ cellRect
.width 
> right
) 
9976                     right 
= cellRect
.x 
+ cellRect
.width
; 
9977                 if (cellRect
.y 
+ cellRect
.height 
> bottom
) 
9978                     bottom 
= cellRect
.y 
+ cellRect
.height
; 
9980             else i 
= rightCol
; // jump over inner cells. 
9984     // convert to scrolled coords 
9986     CalcScrolledPosition( left
, top
, &left
, &top 
); 
9987     CalcScrolledPosition( right
, bottom
, &right
, &bottom 
); 
9990     m_gridWin
->GetClientSize( &cw
, &ch 
); 
9992     if (right 
< 0 || bottom 
< 0 || left 
> cw 
|| top 
> ch
) 
9993         return wxRect( 0, 0, 0, 0); 
9995     rect
.SetLeft( wxMax(0, left
) ); 
9996     rect
.SetTop( wxMax(0, top
) ); 
9997     rect
.SetRight( wxMin(cw
, right
) ); 
9998     rect
.SetBottom( wxMin(ch
, bottom
) ); 
10004 // ------ Grid event classes 
10007 IMPLEMENT_DYNAMIC_CLASS( wxGridEvent
, wxNotifyEvent 
) 
10009 wxGridEvent::wxGridEvent( int id
, wxEventType type
, wxObject
* obj
, 
10010                           int row
, int col
, int x
, int y
, bool sel
, 
10011                           bool control
, bool shift
, bool alt
, bool meta 
) 
10012         : wxNotifyEvent( type
, id 
) 
10019     m_control 
= control
; 
10024     SetEventObject(obj
); 
10028 IMPLEMENT_DYNAMIC_CLASS( wxGridSizeEvent
, wxNotifyEvent 
) 
10030 wxGridSizeEvent::wxGridSizeEvent( int id
, wxEventType type
, wxObject
* obj
, 
10031                                   int rowOrCol
, int x
, int y
, 
10032                                   bool control
, bool shift
, bool alt
, bool meta 
) 
10033         : wxNotifyEvent( type
, id 
) 
10035     m_rowOrCol 
= rowOrCol
; 
10038     m_control 
= control
; 
10043     SetEventObject(obj
); 
10047 IMPLEMENT_DYNAMIC_CLASS( wxGridRangeSelectEvent
, wxNotifyEvent 
) 
10049 wxGridRangeSelectEvent::wxGridRangeSelectEvent(int id
, wxEventType type
, wxObject
* obj
, 
10050                                                const wxGridCellCoords
& topLeft
, 
10051                                                const wxGridCellCoords
& bottomRight
, 
10052                                                bool sel
, bool control
, 
10053                                                bool shift
, bool alt
, bool meta 
) 
10054         : wxNotifyEvent( type
, id 
) 
10056     m_topLeft     
= topLeft
; 
10057     m_bottomRight 
= bottomRight
; 
10059     m_control     
= control
; 
10064     SetEventObject(obj
); 
10068 IMPLEMENT_DYNAMIC_CLASS(wxGridEditorCreatedEvent
, wxCommandEvent
) 
10070 wxGridEditorCreatedEvent::wxGridEditorCreatedEvent(int id
, wxEventType type
, 
10071                                                    wxObject
* obj
, int row
, 
10072                                                    int col
, wxControl
* ctrl
) 
10073     : wxCommandEvent(type
, id
) 
10075     SetEventObject(obj
); 
10081 #endif // wxUSE_GRID